2012-05-07 86 views
5

我有一个充满序列化对象的审计列表,我想比较它们并返回差异列表。通过'比较'我的意思是我想返回一个元素的文本已经改变的地方,或者添加了一个节点的地方(所以它不在Xml1中,但它在Xml2中 - 它不会以相反的方式发生)比较XML片段和返回差异

示例XML:

<HotelBookingView xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema"> 
    <Id>119</Id> 
    <RoomId>1</RoomId> 
    <ChangeRequested>false</ChangeRequested> 
    <CourseBookings>  
    <CourseHotelLink> 
     <Id>0</Id> 
    </CourseHotelLink> 
</CourseBookings> 
</HotelBookingView> 

的命名空间和名称/标签的情况下也不会改变。本示例中所有可以改变的是标签之间的值以及'CourseHotelLink(它的序列化列表)的数量。

我想要的最终结果是哪个节点已更改的列表 - 旧值和新值。

什么是比较它们的最佳选择?我使用.Net 4.0,所以linq是一个选项。我需要能够进行比较而不必知道所有节点的名称 - 尽管我只会比较两个相同类型的对象。我一直在尝试使用下面的代码,但我无法设法使其适应于挑选文本中的更改以及额外的节点。

XmlDocument Xml1 = new XmlDocument(); 
XmlDocument Xml2 = new XmlDocument(); 
Xml1.LoadXml(list[1].Changes); 
Xml2.LoadXml(list[2].Changes); 
foreach (XmlNode chNode in Xml2.ChildNodes) 
{ 
    CompareLower(chNode); 
} 

protected void CompareLower(XmlNode aNode) 
{ 
    foreach (XmlNode chlNode in aNode.ChildNodes) 
    { 
     string Path = CreatePath(chlNode); 
     if (chlNode.Name == "#text") 
     { 
      //all my efforts at comparing text have failed 
      continue; 
     } 
     if (Xml1.SelectNodes(Path).Count == 0) 
     { 
      XmlNode TempNode = Xml1.ImportNode(chlNode, true); 
      //node didn't used to exist, this works- though doesn't return values 
      str = str + "New Node: " + TempNode.Name + ": " + TempNode.Value; 
     } 
     else 
     { 
      CompareLower(chlNode); 
     } 
    } 
} 

它可能是我的代码尝试是英里关闭,有一个更好的方法来做,欢迎任何建议!

EDITTED要添加: 我最终使用了MS Xml Diff工具,下面的代码生成了一个大的html表格列表,其中两个xml节点以绿色突出显示。所以它可能(尽管是疯狂的)生成html,然后对其进行排序以找到文本'lightgreen'(突出显示的值),然后执行一些字符串形成以仅显示更改后的子节点。

var node1 = XElement.Parse("Xml string 1 here").CreateReader(); 
var node2 = XElement.Parse("Xml string 2 here").CreateReader(); 

MemoryStream diffgram = new MemoryStream(); 
XmlTextWriter diffgramWriter = new XmlTextWriter(new StreamWriter(diffgram)); 

XmlDiff xmlDiff = new XmlDiff(XmlDiffOptions.IgnoreChildOrder); 
xmlDiff.Algorithm = XmlDiffAlgorithm.Fast; 
xmlDiff.Compare(node1, node2,diffgramWriter); 

diffgram.Seek(0, SeekOrigin.Begin); 
XmlDiffView xmlDiffView = new Microsoft.XmlDiffPatch.XmlDiffView(); 
StringBuilder sb = new StringBuilder(); 
TextWriter resultHtml = new StringWriter(sb); 
xmlDiffView.Load("Xml string 1", new XmlTextReader(diffgram)); 

xmlDiffView.GetHtml(resultHtml); 
resultHtml.Close(); 
+1

看看这篇文章:http://stackoverflow.com/questions/167946/how-will-you-compare-two-xml-documents –

+0

我还没有弄清楚如何让MS Diff和Patch采用XML字符串 - 我的XML来自数据库,我不想每次我想要创建文件o使用它...可能只是我变得密集。 – UglyTeapot

+0

您不需要创建文件,它带有许多重载比较文件,XmlTextReader或XmlNode –

回答

7

使用XMlDiff是要走的路 - 证明它是一些工作代码。我正在使用你的XML。如果XML不同(或无效),这可能不起作用。

原文:

var xml1 = @"<HotelBookingView xmlns:xsi=""http://www.w3.org/2001/XMLSchema-instance"" xmlns:xsd=""http://www.w3.org/2001/XMLSchema""> 
<Id>119</Id> 
<RoomId>1</RoomId> 
<ChangeRequested>false</ChangeRequested> 
<CourseBookings>  
    <CourseHotelLink> 
    <Id>0</Id> 
    </CourseHotelLink> 
</CourseBookings> 
</HotelBookingView>"; 

CourseBookings不同Id值:创建读者(如果需要改变XDocument)的

var xml2 = @"<HotelBookingView xmlns:xsi=""http://www.w3.org/2001/XMLSchema-instance"" xmlns:xsd=""http://www.w3.org/2001/XMLSchema""> 
<Id>119</Id> 
<RoomId>1</RoomId> 
<ChangeRequested>false</ChangeRequested> 
<CourseBookings>  
    <CourseHotelLink> 
    <Id>1</Id> 
    </CourseHotelLink> 
</CourseBookings> 
</HotelBookingView>"; 

省力的方式:

var node1 = XElement.Parse(xml1).CreateReader(); 
var node2 = XElement.Parse(xml2).CreateReader(); 

准备结果作者:

var result = new XDocument(); 
var writer = result.CreateWriter(); 

执行DIFF:

var diff = new Microsoft.XmlDiffPatch.XmlDiff();  
diff.Compare(node1, node2, writer); 
writer.Flush(); writer.Close(); 

result现在是包含的差异汇总的XDocument

<xd:xmldiff version="1.0" srcDocHash="14506386314386767543" options="None" fragments="no" xmlns:xd="http://schemas.microsoft.com/xmltools/2002/xmldiff"> 
    <xd:node match="1"> 
    <xd:node match="4"> 
     <xd:node match="1"> 
     <xd:node match="1"> 
      <xd:change match="1">1</xd:change> 
     </xd:node> 
     </xd:node> 
    </xd:node> 
    </xd:node> 
</xd:xmldiff> 
+0

这确实有用,谢谢!所以我只需要解析diffgram就可以显示'CourseBookings Id is 0,now 1'这样的东西,这应该很有趣! – UglyTeapot

+0

不要忘记接受它是否回答原始问题:-)关于如何解释来自'xmldiff'的回报有很多信息。 – yamen

+0

已接受,谢谢!还有一些例子如何解释差异只显示差异?我的弱谷歌找不到任何东西 – UglyTeapot