Как предотвратить пустые атрибуты xmlns в выводе из XmlDocument .NET?

118

При генерации XML из XmlDocument в .NET пустой xmlnsатрибут появляется при первой вставке элемента без связанного пространства имен; как этого избежать?

Пример:

XmlDocument xml = new XmlDocument();
xml.AppendChild(xml.CreateElement("root",
    "whatever:name-space-1.0"));
xml.DocumentElement.AppendChild(xml.CreateElement("loner"));
Console.WriteLine(xml.OuterXml);

Вывод:

<root xmlns="whatever:name-space-1.0"><loner xmlns="" /></root>

Желаемый результат :

<root xmlns="whatever:name-space-1.0"><loner /></root>

Есть ли решение, применимое к XmlDocumentкоду, а не то, что происходит после преобразования документа в строку с помощью OuterXml?

Я делаю это, чтобы посмотреть, смогу ли я сопоставить стандартный XML определенного протокола с помощью XML, созданного XmlDocument. xmlnsАтрибут blank не может сломать или запутать синтаксический анализатор, но он также не присутствует ни в одном из примеров использования этого протокола.

Нил С. Обремски
источник

Ответы:

111

Благодаря ответу Джереми Лью и небольшому количеству экспериментов я понял, как удалить пустые xmlnsатрибуты: передать в пространство имен корневого узла при создании любого дочернего узла, на котором вы не хотите иметь префикс. Использование пространства имен без префикса в корне означает, что вам нужно использовать это же пространство имен для дочерних элементов, чтобы они также не имели префиксов.

Фиксированный код:

XmlDocument xml = new XmlDocument();
xml.AppendChild(xml.CreateElement("root", "whatever:name-space-1.0"));
xml.DocumentElement.AppendChild(xml.CreateElement("loner", "whatever:name-space-1.0")); 
Console.WriteLine(xml.OuterXml);

Спасибо всем за все ваши ответы, которые привели меня в правильном направлении!

Нил С. Обремски
источник
1
Точно. Помещение элемента <loner> в пространство имен «something: name-space-1.0» означает, что пустой атрибут xmlns (который не помещает его в пространство имен) не будет добавлен к нему при сериализации. Если вам нужно обновить, как работают пространства имен, взгляните на jclark.com/xml/xmlns.htm
JeniT,
2
Осторожно: элементы нуждаются в этом (или, возможно, лучше doc.DocumentElement.NamespaceURI), но не указывайте пространство имен, иначе CreateAttribute()вы получите, xmlns:psomethingдаже если это тот же самый uri.
Джейсон Клебан
87

Это вариант ответа JeniT (кстати, большое спасибо!)

XmlElement new_element = doc.CreateElement("Foo", doc.DocumentElement.NamespaceURI);

Это избавляет от необходимости везде копировать или повторять пространство имен.

C Джонсон
источник
3
По мне, лучший ответ. Нам не нужно знать, что такое пространство имен документа по умолчанию (полезно, когда мы не создаем XML-файл с нуля, т.е. в сценариях чтения и изменения).
MuiBienCarlota
11

Если <loner>элемент в вашем примере XML не имеет xmlnsобъявления пространства имен по умолчанию, тогда он будет в whatever:name-space-1.0пространстве имен, а не в пространстве имен. Если это то, что вы хотите, вам нужно создать элемент в этом пространстве имен:

xml.CreateElement("loner", "whatever:name-space-1.0")

Если вы хотите, чтобы <loner>элемент не находился в пространстве имен, то созданный XML - это именно то, что вам нужно, и вам не следует беспокоиться об xmlnsатрибуте, который был добавлен автоматически для вас.

JeniT
источник
3
Проблема заключается в несовместимых парсерах XML (обычно от Microsoft), которые не могут справиться с xmnls = "").
Craig Trader,
2
/. называется. Они хотят получить назад свой случайный комментарий о MS.
@W. Крейг Трейдер: не могу сказать, что столкнулся с этим как с проблемой. Пример?
Кев,
1
Правильно, я не хочу, чтобы у узла <loner /> было пространство имен, но я также не хочу, чтобы у него был пустой атрибут пространства имен (xmlns). Мое рассуждение - просто посмотреть, смогу ли я сопоставить вывод XML определенного протокола, который настроен таким образом.
Нил С. Обремски
5
Это не была случайная трепка. Блок приложения Microsoft Updater использует XML-манифест, чтобы определить, что доставить клиенту. К сожалению, парсер манифеста не справляется с xmlns = ""; Мне пришлось написать постпроцессор, который удалял бы пустые атрибуты xmlns.
Craig Trader
7

Поскольку root находится в пространстве имен без префикса, любой дочерний элемент root, который хочет быть без пространства имен, должен выводиться как ваш пример. Решением было бы префикс корневого элемента следующим образом:

<w:root xmlns:w="whatever:name-space-1.0">
   <loner/>
</w:root>

код:

XmlDocument doc = new XmlDocument();
XmlElement root = doc.CreateElement( "w", "root", "whatever:name-space-1.0" );
doc.AppendChild( root );
root.AppendChild( doc.CreateElement( "loner" ) );
Console.WriteLine(doc.OuterXml);
jlew
источник
Спасибо, но добавление пространства имен к фактическому корню нарушит мой XML по отношению к конкретному протоколу, с которым я работаю.
Нил С. Обремски
Ах! Я лучше понял, о чем вы говорите, и учел это при написании собственного ответа. Спасибо, Джереми
Нил С. Обремски
0

Если возможно, создайте класс сериализации, а затем выполните:

XmlSerializerNamespaces ns = new XmlSerializerNamespaces();
ns.Add("", "");
XmlSerializer serializer = new XmlSerializer(yourType);
serializer.Serialize(xmlTextWriter, someObject, ns);

Это безопаснее, и вы можете управлять пространствами имен с помощью атрибутов, если вам действительно нужно больше контроля.

ilitirit
источник
0

Я решил проблему, используя Factory Pattern. Я создал фабрику для объектов XElement. В качестве параметра для создания фабрики я указал объект XNamespace. Итак, каждый раз, когда фабрика создает XElement, пространство имен будет добавляться автоматически. Вот код фабрики:

internal class XElementFactory
{
    private readonly XNamespace currentNs;

    public XElementFactory(XNamespace ns)
    {
        this.currentNs = ns;
    }

    internal XElement CreateXElement(String name, params object[] content)
    {
        return new XElement(currentNs + name, content);
    }
}
brinke
источник
1
ОП спрашивал XmlDocument, а не XDocument.
Джон Сондерс
0

Да, вы можете предотвратить XMLNS из XmlElement. Время первого создания: вот так

<trkpt lat="30.53597" lon="-97.753324" xmlns="">
    <ele>249.118774</ele>
    <time>2006-05-05T14:34:44Z</time>
</trkpt>

Измените код: и передайте пространство имен xml следующим образом

Код C #:

XmlElement bookElement = xdoc.CreateElement("trkpt", "http://www.topografix.com/GPX/1/1");
bookElement.SetAttribute("lat", "30.53597");
bookElement.SetAttribute("lon", "97.753324");
Дебабрата Гош
источник