C#LINQ xml解析使用;PreviousNode“;
在SO的帮助下,我成功地组合了以下LINQ表达式C#LINQ xml解析使用;PreviousNode“;,c#,xml,linq,xml-parsing,C#,Xml,Linq,Xml Parsing,在SO的帮助下,我成功地组合了以下LINQ表达式 var parentids = xliff.Descendants() .Elements(xmlns + "trans-unit") .Elements(xmlns + "seg-source") .Elements(xmlns + "mrk") .Where(e => e
var parentids = xliff.Descendants()
.Elements(xmlns + "trans-unit")
.Elements(xmlns + "seg-source")
.Elements(xmlns + "mrk")
.Where(e => e.Attribute("mtype").Value == "seg")
.Select(item => (XElement)item.Parent.Parent.PreviousNode)
.Where(item => item != null)
.Select(item => item.Elements(xmlns + "source")
.Where(itema => itema != null)
.Select(itemb => itemb.Elements(xmlns + "x")
.LastOrDefault()
.Attribute("id")
.Value.ToString())).ToArray();
它所做的是定位一个mrk
标记(该标记具有@mtype=“seg”
),然后它向上移动到trans-unit
祖先(.parent.parent),检查前一个同级trans-unit
是否有子trans
,如果没有,它从source
子元素返回最后一个x
元素的@id
,否则返回null
(它必须返回null,不能不返回匹配项)
我需要补充的是,虽然下面的示例只有一个这样的前一个节点,没有trans
元素,但在现实生活中的xml中还有很多,因此我必须使用PreviousNode
以下是它使用的XML,它完美地返回了“2”
:
<?xml version="1.0" encoding="utf-8"?>
<xliff xmlns:sdl="http://sdl.com/FileTypes/SdlXliff/1.0" version="1.2" sdl:version="1.0" xmlns="urn:oasis:names:tc:xliff:document:1.2">
<file original="Pasadena_Internet_2016.xml" source-language="en-US" datatype="x-sdlfilterframework2" target-language="da-DK">
<body>
<trans-unit id="d679cb2d-ecba-47ba-acb7-1bb4a798c755" translate="no">
<source>
<x id="0" />
<x id="1" />
<x id="2" />
</source>
</trans-unit>
<trans-unit id="aed9fde2-fd1b-4eba-bfc9-06d325aa7047">
<source>
<x id="3" />Pasadena, California’s iconic Colorado Boulevard <x id="4" />has been the site of the world-famous Tournament of Roses Parade since it began in 1890.
</source>
<seg-source>
<mrk mtype="seg" mid="1">
<x id="3" />Pasadena, California’s iconic Colorado Boulevard <x id="4" />has been the site of the world-famous Tournament of Roses Parade since it began in 1890.
</mrk>
</seg-source>
<target>
<mrk mtype="seg" mid="1">
<x id="3" /><x id="4" />Pasadena, Californiens ikoniske Colorado Boulevard har været stedet for den verdensberømte Rose Bowl-parade siden den begyndte i 1890.
</mrk>
</target>
</trans-unit>
</body>
</file>
</xliff>
下面是另一个XML,它现在抛出上述代码的异常,但它应该返回“0”
:
毒品正在夺走我们孩子的未来。
每17秒就有一名青少年第一次尝试非法药物。
一个drogok megfosztják gyermekeinket一个jövőjüktől。
17 másodpercenként egyújabb tizenéves próbálja ki először a kábítószeret。
我将非常感谢您的帮助。您可以尝试使用
祖先().Last()
来查找名为“trans unit”
或“group”
的最高级别祖先,而不是根据XML结构多次使用父项来导航XML树,然后导航到上一个节点
尝试更换此部件:
.Select(item => (XElement) item.Parent.Parent.PreviousNode)
关于这一点:
.Select(item => (XElement)item.Ancestors()
.Last(o => new[]{"trans-unit","group"}.Contains(o.Name.LocalName))
.PreviousNode)
试试这个:.Where(itema=>(itema!=null)和&(itemb.Elements(xmlns+“x”)!=null))@jdweng——我猜你的意思是itema
notitemb
范围中不存在这个。虽然这可以防止异常,但它不会像应该的那样返回“0”。好的,谢谢。我尝试了这个(包括new[]{“trans-unit”,“group”}
和new[]{“group”,“trans-unit”}
但是在下一行。其中(item=>item!=null)
项的值是
,它低于组
,因此似乎第一个跨单元
被视为匹配项,然后它的前一个节点
被匹配,而不是上到组
,然后匹配前一个节点
。@ib11我的错,我想它可能是 同时存在的trans-unit
或group
没有注意到group
实际上是trans-unit
的父级。在这种情况下,请尝试使用Last()
而不是First()
.Select(item => (XElement) item.Parent.Parent.PreviousNode)
.Select(item => (XElement)item.Ancestors()
.Last(o => new[]{"trans-unit","group"}.Contains(o.Name.LocalName))
.PreviousNode)