[英]Modify Child Node Value With C# XML Document Class
我真是個菜鳥,很抱歉,但是我到處搜索了,但仍然受困。
我正在編寫一個程序,該程序將在用戶選擇初始XML文檔並將其重命名后修改2個XML文檔。 一旦用戶更改了第一個XML的文件名(我們將其稱為XML#1),其他兩個XML文檔必須將新文件名插入其中的特定節點值(XML#2和#3)。
每個XML文檔都具有名稱完全相同的多個“資產”節點,我需要使用XML文檔#1中存在的唯一UUID值來區分所需的節點。
對於XML#2和#3,“ Id”節點都包含相同的唯一UUID。 我正在解析XML#1以獲取此UUID,並將其分配給名為“ cpluuid”的變量。
然后,我在XML#2和#3中搜索“ Id” =到“ cpluuid”的節點,並嘗試修改包含要插入文件名的正確節點。
XML文檔#2-又名var =打包(省略了其他“資產”節點)
<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 Doc#2。 通過以下代碼,“ OriginalFileName”節點已成功更新為變量值“ cpluuid”:
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時遇到了麻煩,因為我需要修改的節點(“路徑”)嵌套在樹的更下方:
XML文檔#3-又名var = Assetmap
<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”的同級兄弟,因此無法弄清楚如何進一步深入研究。 此代碼不會修改“路徑”值。 我嘗試使用不同的XPath語法,但我所知無用:
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方法進行各種操作,但無法使其正常工作。
更新#1
這就是我一直嘗試使用XDocument的方式。 似乎應該可以,但是總是返回對象引用錯誤。
var xml = XDocument.Load(assetmap);
var node = xml.Descendants("Asset").FirstOrDefault(asset => asset.Element("Id").Value == (cpluuid));
node.SetElementValue("Path", (outfile));
更新#2
我第二次嘗試使用XDocument是基於@Juan M.Vergara的出色反饋。 不幸的是,雖然代碼沒有錯誤地執行,但XML節點不會更新為新值。
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的較大部分。 需要更新的“路徑”節點是樹中的第二個節點:
<?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”不是“ Asset”下的“ First”元素。 我正在嘗試使用不同的語法組合,但是似乎沒有任何效果(順序不包含任何元素錯誤)
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”元素。
<?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
再次感謝Juan的幫助。 這是XML Doc#2的工作解決方案。 我知道它不是很漂亮,但是可以。
在某個時候,我想返回並完全消化您的最后一個示例,並按照您的方式組合/清理我的代碼。 謝謝!
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);
我只是嘗試了一下,然后在控制台應用程序示例中整理了一些代碼,請記住,在某些情況下我不控制所有錯誤(例如找不到元素...),但是我認為它可以給您一個思路如何使用System.Xml.Linq。
希望能幫助到你。
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();
}
}
}
這是我得到的輸出:
<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會給出一條消息並在返回中停止執行,只需用continue指令替換return語句,這樣執行就不會停止。
另一方面,您可能必須包括tha名稱空間,這是我最初的ReplacePath函數的更新代碼:
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();
}
希望這次您得到正確的答案:)。
由於新的條件,我更願意寫一個不同的答案。 假設您有這2個示例xml,它們具有不同的結構但具有共同的節點,我又嘗試了使用Descendants。
您基本上需要找到Asset元素,然后檢查ID是否匹配,如果匹配,則更新Path或OriginalFileName元素,請嘗試以下代碼:
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的修改以了解是否已完成替換:
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;
}
聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.