我有以下查询XML文件的方法。我需要它来返回所有文件的信息,这些文件的“System”元素与我传递给我的方法的'sys'变量相匹配。
它工作正常,但只返回1个结果,当我知道XML文件中有多个匹配时。
就好像我的Linq查询只是查找XML文件,直到找到结果然后停止,而我需要它来获取所有匹配的集合。
public ListViewItem getDMcollection(string sys)
{
XDocument doc = XDocument.Load(Form1.CSDBpath + Form1.projectName + "\\Data.xml");
var dms = from dm in doc.Descendants("dataModule")
where dm.Descendants("system").First().Value == sys
select dm;
foreach (var module in dms)
{
ListViewItem item = new ListViewItem(new string[]
{
module.Element("DMC").Value,
module.Element("techName").Value,
module.Element("infoName").Value,
module.Element("status").Value,
module.Element("currentUser").Value,
module.Element("validator").Value,
module.Element("notes").Value,
//dm.Element("size").Value + " kb",
//dm.Element("dateMod").Value
});
return item;
}
return null;
}
这是XML文件的示例:
<DMs>
<dataModule>
<DMC>DMC-AJ-A-29-13-54-00ZZZ-254Z-B_001-00.XML</DMC>
<techName>Pressure switch</techName>
<infoName>Clean mechanically</infoName>
<system>72</system>
<subsystem>13</subsystem>
<subsubsystem>60</subsubsystem>
<status>Checked Out</status>
<notes>-</notes>
<currentUser>JakeMemery</currentUser>
<validator>-</validator>
<dateMod>-</dateMod>
<size>-</size>
</dataModule>
<dataModule>
<DMC>DMC-AJ-A-30-15-62-00AAA-066A-D_001-00.XML</DMC>
<techName>Pressure switch</techName>
<infoName>Support equipment and tools data</infoName>
<system>29</system>
<subsystem>13</subsystem>
<subsubsystem>54</subsubsystem>
<status>Checked In</status>
<notes>-</notes>
<currentUser>-</currentUser>
<validator>-</validator>
<dateMod>-</dateMod>
<size>-</size>
</dataModule>
<dataModule>
<DMC>DMC-AJ-A-45-60-12-00AAA-420A-B_001-00.XML</DMC>
<techName>Pressure switch</techName>
<infoName>General fault isolation procedure</infoName>
<system>29</system>
<subsystem>20</subsystem>
<subsubsystem>10</subsubsystem>
<status>Checked In</status>
<notes>-</notes>
<currentUser>-</currentUser>
<validator>-</validator>
<dateMod>-</dateMod>
<size>-</size>
</dataModule>
</DMs>
举个例子,我可能会将29的值传递给我的方法。正如您所看到的,上面的XML文件包含两个'System'29匹配,但我的程序只返回其中一个 - 第一个。
调用上述方法并传入'sys'变量的方法是:
public ListViewItem splitSNS(string fullSNSpath)
{
string sns = new String(fullSNSpath.ToCharArray().Where(c => Char.IsDigit(c)).ToArray());
if (sns.Length.ToString() == "6")
{
string sys = sns.Substring(4, 2);
string subsys = sns.Substring(2, 2);
string unit = sns.Substring(0, 2);
ListViewItem dms = getDMcollection(sys, subsys, unit);
return dms;
}
else if (sns.Length.ToString() == "4")
{
string sys = sns.Substring(2, 2);
string subsys = sns.Substring(0, 2);
ListViewItem dms = getDMcollection(sys, subsys);
return dms;
}
else if (sns.Length.ToString() == "2")
{
string sys = sns.Substring(0, 2);
ListViewItem dms = getDMcollection(sys);
return dms;
}
return null;
}
以及调用上述方法的摘录
ListViewItem dms = newFilter.splitSNS(fullSNSpath);
if (dms != null)
{
// showfilteredList(dms);
listView1.Items.Add(dms);
showStatus(dms);
}
else
{
MessageBox.Show("There are no DMs to be displayed");
}
答案 0 :(得分:3)
正如我在评论中所提到的,删除First(),你应该没问题:
var dms = from dm in doc.Descendants("dataModule")
where dm.Element("system").Value == sys
select dm;
答案 1 :(得分:2)
您希望system
项与每个dataModule
对应,而不仅仅是第一个。
试试这个:
var dms = from dm in doc.Descendants("dataModule")
where dm.Element("system").Value == sys
select dm;
答案 2 :(得分:2)
我可以看到你的函数只返回一个元素。您尝试循环查询,但return item;
语句将始终返回查询中的第一个元素。也许您需要将其返回类型更改为IEnumerable<ListViewItem>
并将return item;
替换为yield return item;
,例如?
对于您的情况,我建议对getDMcollection
函数进行以下更改:
public IEnumerable<ListViewItem> getDMcollection(string sys)
{
XDocument doc = XDocument.Load(Form1.CSDBpath + Form1.projectName + "\\Data.xml");
var dms = from dm in doc.Descendants("dataModule")
where dm.Descendants("system").First().Value == sys
select dm;
foreach (var module in dms)
{
ListViewItem item = new ListViewItem(new string[]
{
module.Element("DMC").Value,
module.Element("techName").Value,
module.Element("infoName").Value,
module.Element("status").Value,
module.Element("currentUser").Value,
module.Element("validator").Value,
module.Element("notes").Value,
//dm.Element("size").Value + " kb",
//dm.Element("dateMod").Value
});
yield return item;
}
}
当你调用它时,你应该迭代它的结果,将它们添加到列表框
public IEnumerable<ListViewItem> splitSNS(string fullSNSpath)
{
string sns = new String(fullSNSpath.ToCharArray().Where(c => Char.IsDigit(c)).ToArray());
if (sns.Length.ToString() == "6")
{
string sys = sns.Substring(4, 2);
string subsys = sns.Substring(2, 2);
string unit = sns.Substring(0, 2);
IEnumerable<ListViewItem> dms = getDMcollection(sys, subsys, unit);
foreach(var d in dms)
yield return d;
}
else if (sns.Length.ToString() == "4")
{
string sys = sns.Substring(2, 2);
string subsys = sns.Substring(0, 2);
IEnumerable<ListViewItem> dms = getDMcollection(sys, subsys);
foreach(var d in dms)
yield return d;
}
else if (sns.Length.ToString() == "2")
{
string sys = sns.Substring(0, 2);
IEnumerable<ListViewItem> dms = getDMcollection(sys);
foreach(var d in dms)
yield return d;
}
}
最后......
IEnumerable<ListViewItem> dms = newFilter.splitSNS(fullSNSpath);
if (dms.Any())
{
// showfilteredList(dms);
foreach(var d in dms)
listView1.Items.Add(d);
showStatus(dms);
}
else
{
MessageBox.Show("There are no DMs to be displayed");
}