首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >使用C# XML文档类修改子节点值

使用C# XML文档类修改子节点值
EN

Stack Overflow用户
提问于 2018-02-22 00:05:59
回答 2查看 2.7K关注 0票数 1

我是个彻头彻尾的菜鸟,所以道歉,但我到处都找遍了,我还是被困住了。

我正在编写一个程序,它将在用户选择初始XML文档并将其重命名之后修改2个XML文档。一旦用户更改了第一个XML的文件名(让我们称之为XML #1),其他两个XML文档必须将新的文件名插入到其中的特定节点值中(XML #2和#3)。

每个XML都有多个名称完全相同的"Asset“节点,我需要使用XML #1中存在的唯一UUID值来区分我想要的节点。

对于XML #2和#3,"Id“节点包含相同的唯一UUID。我正在解析XML#1以获得这个UUID,并将其分配给一个名为“cpluuid”的变量。

然后,我在XML #2和#3中搜索具有"Id“= to "cpluuid”的节点,并试图修改包含要插入的文件名的正确节点。

XML #2 --又名var = packing (省略了额外的“资产”节点)

代码语言:javascript
复制
<Asset>
  <Id>urn:uuid:d0686356-19c7-4bf4-b915-db778c308d1c</Id>
  <AnnotationText>blah</AnnotationText>
  <Hash>5Yf4BV4GZ4qE9EjvtohZ8Rq8M2w=</Hash>
  <Size>21881</Size>
  <Type>text/xml</Type>
  <OriginalFileName>CPL_IMF_JOT_Sample_143.xml</OriginalFileName>
  <HashAlgorithm Algorithm="http://www.w3.org/2000/09/xmldsig#sha1" />
</Asset>

我已经成功地以这种方式更新了XML #2。"OriginalFileName“节点通过以下代码成功地更新为变量值"cpluuid”:

代码语言:javascript
复制
XmlDocument xmlDoc = new XmlDocument();
XmlNamespaceManager ns = new XmlNamespaceManager(xmlDoc.NameTable);
ns.AddNamespace("cplns1", "http://www.smpte-ra.org/schemas/2067-3/2016");
ns.AddNamespace("cplns2", "http://www.w3.org/2001/XMLSchema-instance");
ns.AddNamespace("pklns", "http://www.smpte-ra.org/schemas/2067-2/2016/PKL");
ns.AddNamespace("assetns", "http://www.smpte-ra.org/schemas/429-9/2007/AM");

xmlDoc.PreserveWhitespace = true;
xmlDoc.Load(output);

string cpluuid = xmlDoc.SelectSingleNode("//cplns1:CompositionPlaylist/cplns1:Id", ns).InnerText;

xmlDoc.PreserveWhitespace = true;
xmlDoc.Load(packing);

XmlNodeList nodeList;
XmlNode root = xmlDoc.DocumentElement;

nodeList = root.SelectNodes("descendant::pklns:Asset[pklns:Id=\"" + cpluuid + "\"]", ns);

foreach (XmlNode Asset in nodeList)
{
    Asset["OriginalFileName"].InnerText = (outfile);
}
xmlDoc.PreserveWhitespace = true;
xmlDoc.Save(packing);

但是,我在XML #3上遇到了问题,因为我需要修改的节点("Path")被嵌套在树的下面:

XML #3 -又名var = assetmap

代码语言:javascript
复制
<Asset>
  <Id>urn:uuid:d0686356-19c7-4bf4-b915-db778c308d1c</Id>
  <ChunkList>
    <Chunk>
      <Path>CPL_IMF_JOT_Sample.xml</Path>
      <VolumeIndex>1</VolumeIndex>
      <Offset>0</Offset>
      <Length>21881</Length>
    </Chunk>
  </ChunkList>
</Asset>

因此,我尝试使用类似的代码,但无法找到进一步深入的方法,因为"Path“不是”Asset“的兄弟。此代码不修改"Path“值。我尝试使用不同的XPath语法,但我所知道的都没有用:

代码语言:javascript
复制
xmlDoc.PreserveWhitespace = true;
xmlDoc.Load(assetmap);

XmlNodeList nodeList2;

nodeList2 = root.SelectNodes("descendant::assetns:Asset[assetns:Id=\"" + cpluuid + "\"]", ns);

foreach (XmlNode Asset in nodeList2)
{
Asset["Path"].InnerText = (outfile);
}
xmlDoc.PreserveWhitespace = true;
xmlDoc.Save(assetmap);

我也尝试过各种使用System.Xml.Linq方法的方法,但都无法让它起作用。

UPDATE#1

下面是我尝试使用XDocument的方式。看起来应该可以工作,但是总是返回一个对象引用错误。

代码语言:javascript
复制
var xml = XDocument.Load(assetmap);

var node = xml.Descendants("Asset").FirstOrDefault(asset => asset.Element("Id").Value == (cpluuid));

node.SetElementValue("Path", (outfile));

更新#2

我第二次尝试使用XDocument是基于@JuanM.Vergara的令人敬畏的反馈。不幸的是,虽然代码执行时没有错误,但是XML节点没有更新到新值。

代码语言:javascript
复制
XDocument document = XDocument.Load(assetmap);

var assetElements = document.Elements("Asset");

foreach (var asset in assetElements)
{
    var innerElements = asset.Elements("Id");
    var matchingId = innerElements.FirstOrDefault(x => x.Value.Equals(cpluuid));
    if (matchingId == null)
    {
       MessageBox.Show("UUID not found");
       return;
    }

    var chunks = asset.Elements("ChunkList").First().Elements();

    foreach (var chunk in chunks)
    {
       chunk.Elements("Path").First().SetValue(outfile);
    }
}

document.Save(assetmap);

这里还有XML的一部分,以防名称空间或其他方面出现问题。需要更新的“路径”节点是树中的第二个节点:

代码语言:javascript
复制
<?xml version="1.0" encoding="utf-8"?>
<AssetMap xmlns="http://www.smpte-ra.org/schemas/429-9/2007/AM">
  <Id>urn:uuid:dc59ba55-adfd-4395-bdaa-de54202d014d</Id>
  <Creator>Colorfront Transkoder 2017</Creator>
  <VolumeCount>1</VolumeCount>
  <IssueDate>2018-02-16T20:59:42-00:00</IssueDate>
  <Issuer>Generic</Issuer>
  <AssetList>
    <Asset>
    <Id>urn:uuid:296a656c-3610-4de1-9b08-2aa63245788d</Id>
    <PackingList>true</PackingList>
    <ChunkList>
      <Chunk>
        <Path>PKL_UUID.xml</Path>
        <VolumeIndex>1</VolumeIndex>
        <Offset>0</Offset>
        <Length>3015</Length>
      </Chunk>
    </ChunkList>
    </Asset>
    <Asset>
    <Id>urn:uuid:d0686356-19c7-4bf4-b915-db778c308d1c</Id>
    <ChunkList>
      <Chunk>
        <Path>CPL_IMF_JOT_Sample.xml</Path>
        <VolumeIndex>1</VolumeIndex>
        <Offset>0</Offset>
        <Length>21881</Length>
      </Chunk>
   </ChunkList>
   </Asset>
 </AssetList>
</AssetMap>

更新#3

我现在正在为XML#2尝试同样的方法,但它不起作用。我猜是因为"OriginalFileName“不是”资产“下的”第一个“元素。我正在尝试不同的语法组合,但似乎没有什么工作(序列不包含元素错误)

代码语言:javascript
复制
XDocument pkldoc = XDocument.Load(packing);

var pklns = pkldoc.Root.GetDefaultNamespace();
var packingList = pkldoc.Elements(pklns + "PackingList").First(); // this ignores the namespace
var pklassetList = packingList.Elements(pklns + "AssetList").First();
var pklassetElements = pklassetList.Elements(pklns + "Asset");

foreach (var pklasset in pklassetElements)
{
    var innerElements = pklasset.Elements(pklns + "Id");
    var matchingId = innerElements.FirstOrDefault(x => x.Value.Equals(cpluuid));
    if (matchingId == null)
    {
       //MessageBox.Show("UUID not found");
       continue;
    }

    var ofns = pklasset.Elements(pklns + "Asset").First().Elements();

    foreach (var ofn in ofns)
    {
       ofn.Elements(pklns + "OriginalFileName").First().SetValue(outfile);
     }
    }

更新#4

以下是XML #2的全部内容,它的结构与XML#3不同,我只想更新最后一棵“资产”树中的"OriginalFileName“元素。

代码语言:javascript
复制
<?xml version="1.0" encoding="UTF-8"?>
<PackingList xmlns="http://www.smpte-ra.org/schemas/2067-2/2016/PKL">
<Id>urn:uuid:296a656c-3610-4de1-9b08-2aa63245788d</Id>
<AnnotationText>IMF_JOT_Sample</AnnotationText>
<IssueDate>2018-02-16T20:59:42-00:00</IssueDate>
<Issuer>Generic</Issuer>
<Creator>Generic</Creator>
<AssetList>
<Asset>
  <Id>urn:uuid:744f36b7-fc7e-4179-8b75-c71c18f98156</Id>
  <AnnotationText>Video_744f36b7-fc7e-4179-8b75-c71c18f98156.mxf</AnnotationText>
  <Hash>8HhnKnLn+Lp/Ik9i94Ml4SXAxH4=</Hash>
  <Size>14568486</Size>
  <Type>application/mxf</Type>
  <OriginalFileName>Video_744f36b7-fc7e-4179-8b75-c71c18f98156.mxf</OriginalFileName>
  <HashAlgorithm Algorithm="http://www.w3.org/2000/09/xmldsig#sha1"/>
</Asset>
<Asset>
  <Id>urn:uuid:bf5438ea-ba58-4ae0-a64a-5d23cee2ebb3</Id>
  <AnnotationText>Audio_bf5438ea-ba58-4ae0-a64a-5d23cee2ebb3.mxf</AnnotationText>
  <Hash>Wg4aEAE5Ji9e14ZyGkvfUUjBwCw=</Hash>
  <Size>4341294</Size>
  <Type>application/mxf</Type>
  <OriginalFileName>Audio_bf5438ea-ba58-4ae0-a64a-5d23cee2ebb3.mxf</OriginalFileName>
  <HashAlgorithm Algorithm="http://www.w3.org/2000/09/xmldsig#sha1"/>
</Asset>
<Asset>
  <Id>urn:uuid:dd5a88d2-ccec-4b22-8584-fda51945c3ea</Id>
  <AnnotationText>Audio_dd5a88d2-ccec-4b22-8584-fda51945c3ea.mxf</AnnotationText>
  <Hash>OwjRFnWZCdKHSZ+3PBXroDhMMlY=</Hash>
  <Size>1458414</Size>
  <Type>application/mxf</Type>
  <OriginalFileName>Audio_dd5a88d2-ccec-4b22-8584-fda51945c3ea.mxf</OriginalFileName>
  <HashAlgorithm Algorithm="http://www.w3.org/2000/09/xmldsig#sha1"/>
</Asset>
<Asset>
  <Id>urn:uuid:9e11458a-71fb-4702-8609-55d2308dcc64</Id>
  <AnnotationText>Sub_9e11458a-71fb-4702-8609-55d2308dcc64.mxf</AnnotationText>
  <Hash>48KyxgwCJVXIdgAGfaNApheQN5M=</Hash>
  <Size>34509</Size>
  <Type>application/mxf</Type>
  <OriginalFileName>Sub_9e11458a-71fb-4702-8609-55d2308dcc64.mxf</OriginalFileName>
  <HashAlgorithm Algorithm="http://www.w3.org/2000/09/xmldsig#sha1"/>
</Asset>
<Asset>
  <Id>urn:uuid:3f57f474-4c81-438e-a67d-1b08fa09a10d</Id>
  <AnnotationText>IMF_JOT_Sample</AnnotationText>
  <Hash>q8TiPkg/3devlN3LXnBhrgkZ968=</Hash>
  <Size>713</Size>
  <Type>text/xml</Type>
  <OriginalFileName>OPL_3f57f474-4c81-438e-a67d-1b08fa09a10d.xml</OriginalFileName>
  <HashAlgorithm Algorithm="http://www.w3.org/2000/09/xmldsig#sha1"/>
</Asset>
<Asset>
  <Id>urn:uuid:d0686356-19c7-4bf4-b915-db778c308d1c</Id>
  <AnnotationText>CPL_IMF_JOT_Sample.xml</AnnotationText>
  <Hash>5Yf4BV4GZ4qE9EjvtohZ8Rq8M2w=</Hash>
  <Size>21881</Size>
  <Type>text/xml</Type>
  <OriginalFileName>CPL_IMF_JOT_Sample.xml</OriginalFileName>
  <HashAlgorithm Algorithm="http://www.w3.org/2000/09/xmldsig#sha1"/>
</Asset>

更新#5

再次感谢胡安的帮助。这里是XML #2的工作解决方案,我知道它不是很漂亮,但它可以工作。

在某种程度上,我希望返回并完整地消化您的最后一个示例,并按照您的方式组合/清理我的代码。谢谢!

代码语言:javascript
复制
XDocument pkldoc = XDocument.Load(packing);

var pklns = pkldoc.Root.GetDefaultNamespace();

var pklassetElements = pkldoc.Descendants(pklns + "Asset");

foreach (var pklasset in pklassetElements)
{
    var idElement = pklasset.Descendants(pklns + "Id").First();
    if (!idElement.Value.Equals(cpluuid))
    continue;


    SetNewValue(pklasset, pklns + "OriginalFileName", outfile);

}
void SetNewValue(XElement currentElement, XName elementName, string newValue)
                {
    var matchingElements = currentElement.Descendants(elementName);
    if (matchingElements.Any())
    {
        foreach (var element in matchingElements)
          element.SetValue(newValue);
                    }
                }
                pkldoc.Save(packing);
EN

回答 2

Stack Overflow用户

回答已采纳

发布于 2018-02-23 03:13:28

由于新的情况,我宁愿写一个不同的答案。如果您有这两个示例xmls,具有不同的结构,但有公共节点,那么我又尝试了使用Descendants。

基本上,您需要找到Asset,然后检查Id是否匹配,如果匹配,请更新Path或OriginalFileName元素,尝试以下代码:

代码语言:javascript
复制
    private void Run()
    {
        XDocument doc1 = XDocument.Load("xml1.xml");
        XDocument doc2 = XDocument.Load("xml2.xml");
        var id = @"urn:uuid:d0686356-19c7-4bf4-b915-db778c308d1c";

        ReplaceId(doc1, id, "new path");
        ReplaceId(doc2, id, "new path");

        doc1.Save("xml1_new.xml");
        doc2.Save("xml2_new.xml");

        Console.WriteLine("Enter to exit...");
        Console.ReadLine();
    }

    private void ReplaceId(XDocument doc, string id, string newValue)
    {
        var ns = doc.Root.GetDefaultNamespace();
        var assetElements = doc.Descendants(ns + "Asset");

        foreach (var element in assetElements)
        {
            var idElement = element.Descendants(ns + "Id").First();
            if (!idElement.Value.Equals(id))
                continue;

            // for xml model #1
            SetNewValue(element, ns + "Path", newValue);

            // for xml model #2
            SetNewValue(element, ns + "OriginalFileName", newValue);
        }
    }

    private void SetNewValue(XElement currentElement, XName elementName, string newValue)
    {
        var matchingElements = currentElement.Descendants(elementName);
        if (matchingElements.Any())
        {
            foreach (var element in matchingElements)
                element.SetValue(newValue);
        }
    }

现在让我来看看这个是怎么回事。

更新

对ReplaceId的修改,以了解是否已经完成了替换:

代码语言:javascript
复制
    private bool ReplaceId(XDocument doc, string id, string newValue)
    {
        var ns = doc.Root.GetDefaultNamespace();
        var assetElements = doc.Descendants(ns + "Asset");
        var result = false;

        foreach (var element in assetElements)
        {
            var idElement = element.Descendants(ns + "Id").First();
            if (!idElement.Value.Equals(id))
                continue;

            // for xml model #1
            SetNewValue(element, ns + "Path", newValue);

            // for xml model #2
            SetNewValue(element, ns + "OriginalFileName", newValue);

            result = true;
        }

        return result;
    }
票数 0
EN

Stack Overflow用户

发布于 2018-02-22 05:13:53

我只是尝试将一些代码放在控制台应用程序示例中,请记住,我并不控制al用例中的所有错误(比如元素not .),但我认为它可以让您了解如何使用System.Xml.Linq。

希望能帮上忙。

代码语言:javascript
复制
using System;
using System.Linq;
using System.Xml.Linq;

namespace ParsingXML
{

    class Program
    {
        const string _xml =
            @"<Asset>
              <Id>urn:uuid:d0686356-19c7-4bf4-b915-db778c308d1c</Id>
              <ChunkList>
                <Chunk>
                  <Path>CPL_IMF_JOT_Sample.xml</Path>
                  <VolumeIndex>1</VolumeIndex>
                  <Offset>0</Offset>
                  <Length>21881</Length>
                </Chunk>
              </ChunkList>
            </Asset>";

        static void Main(string[] args)
        {
            ReplacePath(
                @"urn:uuid:d0686356-19c7-4bf4-b915-db778c308d1c",
                @"c:\path\to\somefilename.xml"
                );

            Console.WriteLine("Enter to exit...");
            Console.ReadLine();
        }

        static void ReplacePath(string id, string pathToSet)
        {
            XDocument document = XDocument.Parse(_xml);

            var assetElements = document.Elements("Asset");

            foreach (var asset in assetElements)
            {
                var innerElements = asset.Elements("Id");
                var matchingId = innerElements.FirstOrDefault(e => e.Value.Equals(id));
                if (matchingId == null)
                {
                    Console.WriteLine("id not found");
                    return;
                }

                var chunks = asset.Elements("ChunkList").First().Elements();

                foreach (var chunk in chunks)
                {
                    chunk.Elements("Path").First().SetValue(pathToSet);
                }
            }
            var xml = document.ToString();
            Console.WriteLine(xml);

            Console.WriteLine("Enter to exit...");
            Console.ReadLine();
        }
    }
}

这是我得到的输出:

代码语言:javascript
复制
<Asset>
  <Id>urn:uuid:d0686356-19c7-4bf4-b915-db778c308d1c</Id>
  <ChunkList>
    <Chunk>
      <Path>c:\path\to\somefilename.xml</Path>
      <VolumeIndex>1</VolumeIndex>
      <Offset>0</Offset>
      <Length>21881</Length>
    </Chunk>
  </ChunkList>
</Asset>
Enter to exit...

更新

问题是,有多个资产,而且由于第一个资产与第一个资产不匹配,第一个foreach给出消息并在返回中停止执行,因此只需将返回语句替换为继续指令,这样执行就不会停止。

另一方面,您可能必须包含tha命名空间,这是我最初的ReplacePath函数的更新代码:

代码语言:javascript
复制
    static void ReplacePath(string targetFile, string id, string outfile)
    {
        //XDocument document = XDocument.Parse(_xml);
        XDocument document = XDocument.Load(targetFile);
        // get root namespace to use with rest of element names
        var ns = document.Root.GetDefaultNamespace();

        var assetMap = document.Elements(ns + "AssetMap").First(); // this ignores the namespace
        var assetList = assetMap.Elements(ns + "AssetList").First();
        var assetElements = assetList.Elements(ns + "Asset");

        foreach (var asset in assetElements)
        {
            var innerElements = asset.Elements(ns + "Id");
            var matchingId = innerElements.FirstOrDefault(e => e.Value.Equals(id));
            if (matchingId == null)
            {
                continue;
            }

            var chunks = asset.Elements(ns + "ChunkList").First().Elements();

            foreach (var chunk in chunks)
            {
                var chunkElement = chunk.Elements(ns + "Path").First();
                chunkElement.SetValue(outfile);
            }
        }
        var xml = document.ToString();
        Console.WriteLine(xml);

        Console.WriteLine("writing to file");
        document.Save(targetFile);

        Console.WriteLine("Enter to exit...");
        Console.ReadLine();
    }

我希望这次你得到了正确的答案:)。

票数 0
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/48917569

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档