XDocument изменить все имена атрибутов - PullRequest
3 голосов
/ 15 апреля 2010

У меня есть XDocument, похожий на

<root>
     <a>
          <b foo="1" bar="2" />
          <b foo="3" bar="4" />
          <b foo="5" bar="6" />
          <b foo="7" bar="8" />
          <b foo="9" bar="10" />
     </a>
</root>

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

        string dd=LoadedXDocument.ToString();
        foreach (var s in AttributeReplacements)
            dd = dd.Replace(s.Old+"=", s.New+"=");

Ответы [ 2 ]

3 голосов
/ 15 апреля 2010

Выполнение этого с текстовым поиском и заменой должно выполняться с использованием StringBuilder, чтобы избежать обычных проблем создания строк в цикле (много мусора). Также очень трудно предотвратить ложные срабатывания (что если текст, соответствующий атрибуту, встречается в текстовом узле?)

Лучшие варианты с различными компромиссами включают в себя:

  1. Загрузка в XDocument или XmlDocument, итерация по дереву с заменой соответствующих атрибутов.
  2. Используйте XSLT
  3. Чтение из XmlReader и запись непосредственно в XmlWriter с измененными атрибутами.

Из них # 3 избегает загрузки всего документа в память. # 2 требует навыков XSLT, но легко допускает произвольное количество замен (ядром XSLT может быть шаблон с новыми, старыми парами атрибутов, введенными во время выполнения). # 1, вероятно, будет самым простым, но с целым документом в памяти и накладными расходами на обработку множественных замен.

Я бы, скорее всего, посмотрел на XSLT с подходом Xml Reader / Writer в качестве резервной копии.

Однако # 1 должен быть проще всего реализован, что-то вроде (игнорируя пространства имен XML среди других деталей):

using System.Xml.Linq;
using System.Xml.XPath;

var xdoc = XDocument.Load(....);
var nav = xdoc.CreateNavigator();

foreach (repl in replacements) {
  var found = (XPathNodeIterator) nav.Evaluate("//@" + repl.OldName);

  while (found.MoveNext()) {
    var node = found.Current;
    var val = node.Value;
    node.DeleteSelf(); // Moves ref to parent.
    node.CreateAttribute("", repl.NewName, "", val);
  }
}

Окончательный выбор будет зависеть от производительности балансировки (особенно памяти при работе с большими документами) и сложности. но только вы (и ваша команда) можете сделать этот звонок.

2 голосов
/ 15 апреля 2010

Вот полное решение XSLT :

<xsl:stylesheet version="1.0"
    xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
    xmlns:my="my:reps"
    exclude-result-prefixes="my"
>
    <xsl:output omit-xml-declaration="yes" indent="yes"/>

    <my:replacements>
      <foo1 old="foo"/>
      <bar1 old="bar"/>
    </my:replacements>

    <xsl:variable name="vReps" select=
     "document('')/*/my:replacements/*"/>

 <xsl:template match="node()|@*">
     <xsl:copy>
       <xsl:apply-templates select="node()|@*"/>
     </xsl:copy>
 </xsl:template>

 <xsl:template match="@*">
  <xsl:variable name="vRepNode" select=
   "$vReps[@old = name(current())]"/>

   <xsl:variable name="vName" select=
    "name(current()[not($vRepNode)] | $vRepNode)"/>

   <xsl:attribute name="{$vName}">
     <xsl:value-of select="."/>
   </xsl:attribute>
 </xsl:template>
</xsl:stylesheet>

Когда это преобразование применяется к предоставленному документу XML, получается желаемый результат :

<root>
   <a>
      <b foo1="1" bar1="2"/>
      <b foo1="3" bar1="4"/>
      <b foo1="5" bar1="6"/>
      <b foo1="7" bar1="8"/>
      <b foo1="9" bar1="10"/>
   </a>
</root>

Обратите внимание , что это общее решение, позволяющее указывать и изменять любой список замен без изменения кода. Замены могут быть в отдельном файле XML для простоты обслуживания.

...