gpt4 book ai didi

c# - 处理 Linq XML 和复杂查询

转载 作者:行者123 更新时间:2023-11-30 22:35:41 26 4
gpt4 key购买 nike

这对我来说是个脑筋急转弯:

假设我有类似于以下的 XML 数据:

<?xml version="1.0"?>
<site>
<open_auction id="open_auction0">
<initial>25.39</initial>
<reserve>67.91</reserve>
<bidder>
<date>09/27/1999</date>
<time>18:47:09</time>
<personref person="person308"/>
<increase>18.00</increase>
</bidder>
<bidder>
<date>06/03/2000</date>
<time>17:47:49</time>
<personref person="person942"/>
<increase>19.50</increase>
</bidder>
<bidder>
...
</bidder>
<current>202.39</current>
<itemref item="item2"/>
<seller person="person573"/>
<annotation>
<author person="person128"/>
<description>
<parlist>
<listitem>
<text>
niece <keyword> till banqueting little unkindly </keyword>
</text>
</listitem>
<listitem>
<text>
emilia jack standards
</text>
</listitem>
</parlist>
</description>
<happiness>1</happiness>
</annotation>
<quantity>1</quantity>
<type>Regular</type>
<interval>
<start>12/23/2001</start>
<end>11/11/2000</end>
</interval>
</open_auction>
<open_auction id="open_auction1">
...
</open_auction>
<open_auction id="open_auction2">
...
</open_auction>
</site>

您可以从 www.megamatz.de/auction.xml 下载完整且更大(约 5MB)的测试文件

我需要确定所有 open_auctions 的 ID,其中某个给定的人 (Person1) 在另一个给定的人 (Person2) 之前出价。在一次公开拍卖中,Person1 的“投标人”节点的含义中的“之前”出现在 Person2 的“投标人”节点之前,因此它与日期或时间无关。

这里是这个问题的 XPath-Query 也许它能更好地解释问题:

let $auction := doc("auction.xml") return
for $b in $auction/site/open_auctions/open_auction
where
some $pr1 in $b/bidder/personref[@person = "person20"],
$pr2 in $b/bidder/personref[@person = "person51"]
satisfies $pr1 << $pr2
return <history>{$b/reserve/text()}</history>

我以某种方式知道我需要某种嵌套查询,但我无法直接使用此 Linq 概念和这些 XElement(s)。因此,我不仅对解决方案感兴趣,而且还特别了解 any()、contains() 的概念,以及使用 linq 处理此类查询的其他方法。

我真的很绝望,需要一些帮助,我整晚都在寻找和尝试,但我就是无法弄清楚,但没有任何效果。

编辑添加

很抱歉再次询问,但是不行,它不起作用。我试了又试,非常好,因为我学到了很多东西。我感觉我快要找到答案了,但是当涉及到“ElementAfterSelf”时,我可以在这个复杂查询的上下文中使用一些提示和解释。

这是我的方法,是我根据从 Jon Skeets 的回答中学到的知识得出的:

List<XElement> = doc.Descendants("open_auction")
.Where(x => x.Element("bidder") != null && x.Elements("bidder")
.Any(b => b.Elements("personref")
.First(c => c.IsBidder("person540"))
.ElementsAfterSelf("bidder").Any(d => d.IsBidder("person1068"))
)).ToList();

但它也不起作用。但是……:

List<XElement> j = XMarkXML.Element("site").Element("open_auctions").Elements("open_auction")
.Where(x => x.Element("bidder") != null && x.Elements("bidder")
.Any(b => b.Elements("personref")
.Any(c => c.IsBidder("person540")) //I changed First with Any
//.ElementsAfterSelf().Any(d => d.IsBidder("person1068")) //and commented the last part out
)).ToList();

...至少提出了 3 个“open_auction”,其中“person540”是投标人。具体来说,在我的测试文件中,它是 id="open_auction11",其中“person1068”出价“在”“person540”之后。

我发自内心地问,有人可以用测试文件试一试吗?我将它上传到 www.megamatz.de/auction.xml 。在文件中,公开拍卖的路径​​是:Element("site").Element("open_auctions").Elements("open_auction")

最佳答案

编辑:好的,再次编辑。我认为这没问题,但即使有您的示例文件,如果您能给我们一个输入和预期输出的具体示例,那就太好了:

// Extension methods
internal static bool IsBidder(this XElement bidder, string person)
{
return (string) bidder.Element("personref")
.Attribute("person") == person;
}

internal static IEnumerable<XElement> ElementsAfterFirst
(this IEnumerable<XElement> source, XName name)
{
var first = source.FirstOrDefault();
if (first != null)
{
foreach (var element in first.ElementsAfterSelf(name))
{
yield return element;
}
}
}

var query = doc
.Descendants("open_auction")
.Where(x => x.Elements("bidder")
// All the person20 bids...
.Where(b => b.IsBidder("person20"))
// Now project to bids after the first one...
.ElementsAfterFirst("bidder")
// Are there any from person51? If so, include this auction.
.Any(b => b.IsBidder("person51"))
);

第二种扩展方法也可以写成:

internal static IEnumerable<XElement> ElementsAfterFirst
(this IEnumerable<XElement> source, XName name)
{
var first = source.FirstOrDefault();
return first == null ? new XElement[0] : first.ElementsAfterSelf(name);
}

它稍微改变了时间,但在这种特殊情况下应该没有任何区别......

关于c# - 处理 Linq XML 和复杂查询,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/7377198/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com