C# Удаление ссылок и использования пространства имен из данных XML

Я получаю сообщения Swift MX, и мне нужно удалить из них пространства имен. По сути, это два XML-документа, склеенных вместе. Чаще всего они выглядят так:

<AppHdr> содержимое заголовка </AppHdr><Document> содержимое документа </Document>

С этим легко справиться, поскольку я поставил <root> и </root> вокруг всего этого, чтобы создать XML (только один корневой узел) и прочитать его в XmlDocument:

<root><AppHdr> ... </AppHdr><Document> ... </Document></root>

Однако некоторые сообщения имеют пространства имен, и это пространство имен неизвестно во время разработки. Примером может быть:

<ns2:AppHdr xmlns:ns2 = "some stuff"> содержимое заголовка </ns2:AppHdr><ns3:Document xmlns:ns3 = "some other stuff"> содержимое документа </ns3:Document>

Обратите внимание, что все XML-дерево под AppHdr цитирует тег пространства имен AppHdr, а те, что под Document, цитируют тег Document.

Теги пространства имен могут различаться. В этом примере это ns2 и ns3, но какими они будут заранее я не знаю. Есть ли простой способ избавиться от них?

Задавая вопрос, вам необходимо предоставить минимальный воспроизводимый пример: Отредактируйте исходный вопрос и предоставьте следующее: (1) Образец правильно сформированного XML-файла. (2) Что вам нужно сделать, то есть логика, и ваш код пытается ее реализовать. (3) Желаемый результат основан на примере данных в № 1 выше.

Yitzhak Khabinsky 19.04.2024 17:32

Это можно легко сделать через XSLT.

Yitzhak Khabinsky 19.04.2024 17:41

Почти каждый запрос на удаление пространств имен вызван непониманием их назначения и того, как с ними бороться. Имейте в виду, что вы отклоняетесь от рекомендуемого пути, удаляя пространства имен и, вероятно, создавая проблемы для себя и/или других людей, работающих с полученным XML-документом.

kjhughes 19.04.2024 21:00

@kjhughes - Я слышу твое предупреждение. Мне интересно, что вы можете считать проблемой в приведенном выше примере. Я не уверен, как наличие двух разных пространств имен (и только иногда) повышает смысл документов.

Mark Roworth 22.04.2024 10:42

Дело не в том, что я воспринимаю; дело в том, что компоненты XML (элементы и атрибуты) в пространстве имен XML намеренно имеют другие имена, чем те, которые не находятся в пространстве имен. Расстояние между именами служит нескольким целям, включая предотвращение конфликтов имен и установление полномочий для определения синтаксиса/семантики. Большинство запросов на их удаление связано с незнанием их назначения и/или механизма работы с ними, а не потому, что это лучшее дизайнерское решение в целом.

kjhughes 22.04.2024 13:06
Стоит ли изучать PHP в 2026-2027 годах?
Стоит ли изучать PHP в 2026-2027 годах?
Привет всем, сегодня я хочу высказать свои соображения по поводу вопроса, который я уже много раз получал в своем сообществе: "Стоит ли изучать PHP в...
Поведение ключевого слова "this" в стрелочной функции в сравнении с нормальной функцией
Поведение ключевого слова "this" в стрелочной функции в сравнении с нормальной функцией
В JavaScript одним из самых запутанных понятий является поведение ключевого слова "this" в стрелочной и обычной функциях.
Приемы CSS-макетирования - floats и Flexbox
Приемы CSS-макетирования - floats и Flexbox
Здравствуйте, друзья-студенты! Готовы совершенствовать свои навыки веб-дизайна? Сегодня в нашем путешествии мы рассмотрим приемы CSS-верстки - в...
Тестирование функциональных ngrx-эффектов в Angular 16 с помощью Jest
В системе управления состояниями ngrx, совместимой с Angular 16, появились функциональные эффекты. Это здорово и делает код определенно легче для...
Концепция локализации и ее применение в приложениях React ⚡️
Концепция локализации и ее применение в приложениях React ⚡️
Локализация - это процесс адаптации приложения к различным языкам и культурным требованиям. Это позволяет пользователям получить опыт, соответствующий...
Пользовательский скаляр GraphQL
Пользовательский скаляр GraphQL
Листовые узлы системы типов GraphQL называются скалярами. Достигнув скалярного типа, невозможно спуститься дальше по иерархии типов. Скалярный тип...
0
5
55
1
Перейти к ответу Данный вопрос помечен как решенный

Ответы 1

Ответ принят как подходящий

Минимальный воспроизводимый пример не приведен. Вот общее решение на основе XSLT, которое будет работать с любым правильно сформированным XML-файлом.

Входной XML

<root>
   <ns2:AppHdr xmlns:ns2 = "some stuff">*contents of the header*</ns2:AppHdr>
   <ns3:Document xmlns:ns3 = "some other stuff">*contents of the document*</ns3:Document>
</root>

XSLT

<?xml version='1.0'?>
<xsl:stylesheet version = "1.0" xmlns:xsl = "http://www.w3.org/1999/XSL/Transform">
   <xsl:output method = "xml" indent = "yes" omit-xml-declaration = "yes"/>
   <xsl:strip-space elements = "*"/>

   <!-- template to copy elements -->
   <xsl:template match = "*">
      <xsl:element name = "{local-name()}">
         <xsl:apply-templates select = "@* | node()"/>
      </xsl:element>
   </xsl:template>

   <!-- template to copy attributes -->
   <xsl:template match = "@*">
      <xsl:attribute name = "{local-name()}">
         <xsl:value-of select = "."/>
      </xsl:attribute>
   </xsl:template>

   <!-- template to copy the rest of the nodes -->
   <xsl:template match = "comment() | text() | processing-instruction()">
      <xsl:copy/>
   </xsl:template>
</xsl:stylesheet>

Выходной XML

<root>
  <AppHdr>*contents of the header*</AppHdr>
  <Document>*contents of the document*</Document>
</root>

С#

void Main()
{
   const string SOURCEXMLFILE = @"e:\Temp\input.xml";
   const string XSLTFILE = @"e:\Temp\process.xslt";
   const string OUTPUTXMLFILE = @"e:\temp\output.xml";

   try
   {
      XsltArgumentList xslArg = new XsltArgumentList();

      using (XmlReader src = XmlReader.Create(SOURCEXMLFILE))
      {
         XslCompiledTransform xslt = new XslCompiledTransform();
         xslt.Load(XSLTFILE, new XsltSettings(true, true), new XmlUrlResolver());

         XmlWriterSettings settings = xslt.OutputSettings.Clone();
         settings.IndentChars = "\t";
         // to remove BOM
         settings.Encoding = new UTF8Encoding(false);

         using (XmlWriter result = XmlWriter.Create(OUTPUTXMLFILE, settings))
         {
            xslt.Transform(src, xslArg, result, new XmlUrlResolver());
            result.Close();
         }
      }
      Console.WriteLine("File '{0}' has been generated.", OUTPUTXMLFILE);
   }
   catch (Exception ex)
   {
      Console.WriteLine(ex.Message);
   }
}

Привет @Yitzhak, спасибо за ответ. Я не могу легко привести минимальный рабочий пример, поскольку это означает вставку конфиденциальных финансовых данных, что мне явно не разрешено. Я не уверен, что строки пространства имен или структура сообщений Swift являются общедоступным, отсюда и минимальный характер вопроса. Спасибо за четкий и исчерпывающий ответ. Я буду использовать это. Очень признателен.

Mark Roworth 22.04.2024 10:23

Другие вопросы по теме