我正在使用以下linq查询来处理XML文件..
XElement rootElement = XElement.Load(@"test.xml");
int StyCode;
var lv1s = from lv1 in rootElement.Descendants("Class")
where lv1.Attribute("Code").Value.Equals("002")
select new
{
Children = from ltd in lv1.Descendants("Subject")
where ltd.Attribute("Course").Value.Equals("Math")
select new
{
****//In This below section need result on condition based... ****
if(StyCode=0)
{
Children1 = ltd.Attribute("AllTeachers").Value.Equals("Y") ? true : false
}
else
{
Children2 = ltd.Attribute("SpeciaGuest").Value.Equals("Y") ? true : ltd.Elements("Topic").Attributes("Code").Where(x => x.Value.Equals("1")).FirstOrDefault() != null ? true : false
}
}
};
以下是我的XML结构..
<?xml version="1.0" encoding="utf-8" ?>
<Document>
<Class Code="001">
<Subject Course="Math" AllTeachers='N' SpeciaGuest='N'></Subject>
<Subject Course="Engish" AllTeachers='Y' SpeciaGuest='Y'></Subject>
<Subject Course="History" AllTeachers='Y' SpeciaGuest='Y'></Subject>
</Class>
<Class Code="002">
<Subject Course="Math" AllTeachers='Y' SpeciaGuest='N'>
<Topic Code="1">LAW1</Topic>
<Topic Code="2">
LAW2
</Topic>
<Topic Code="3">
LAW3</Topic>
</Subject>
<Subject Course="Engish" AllTeachers='Y' SpeciaGuest='Y'></Subject>
<Subject Course="History" AllTeachers='Y' SpeciaGuest='Y'></Subject>
</Class>
</Document>
请告诉我,如何使用条件选择多个选择结果。另请告诉我在这个linq查询中使用resharper会没问题吗?
答案 0 :(得分:1)
您无法创建具有不同属性的匿名对象。那么序列类型是什么?您可以为同一属性分配不同的值(通过三元运算符?:
)
XDocument xdoc = XDocument.Load(@"test.xml");
var lv1s = from c in xdoc.Descendants("Class")
where (string)c.Attribute("Code") == "002"
select new {
Children = (StyCode == 0) ?
((string)c.Attribute("AllTeachers") == "Y") :
((string)c.Attribute("SpeciaGuest") == "Y") ||
c.Elements("Topic") .
.Any(t => (string)t.Attribute("Code") == "1"))
};
BTW你要返回的是具有布尔Children
属性的对象列表。我相信你需要更多的数据。
答案 1 :(得分:1)
bool flag = boolexpression ? true : false;
与
相同bool flag = boolexpression;
因此,请停止使用三元数来不必要地使代码复杂化。
您可以将select子句编写为:
select new
{
Children1 = (StyCode==0) ? ltd.Attribute("AllTeachers").Value.Equals("Y") : false,
Children2 = (StyCode==0) ? false :
ltd.Attribute("SpeciaGuest").Value.Equals("Y")
|| ltd.Elements("Topic").Attributes("Code").Where(x => x.Value.Equals("1")).FirstOrDefault() != null
}
有些人可能会说重复这种情况是不好的形式。如果你在那个营地,我推荐使用它:
select (StyCode == 0) ? objectContructor1(ltd) : objectConstructor2(ltd)
答案 2 :(得分:1)
您需要使用条件运算符:
Children1 = StyCode == 0 ?
ltd.Attribute("AllTeachers").Value == "Y"
:
(ltd.Attribute("SpeciaGuest").Value == "Y"
|| ltd.Elements("Topic").Attributes("Code").Select(a => a.Value).Contains("1")
)
答案 3 :(得分:1)
由于if(StyCode=0)
不是基于序列中的当前项目,因此您可以在查询之外提取if
:
另请注意,因为两个Select
子句都选择了单个项目,所以您不需要匿名类型,只需选择该项目即可。结果如下:
XElement rootElement = XElement.Load(@"test.xml");
int StyCode = 0;
IEnumerable<IEnumerable<bool>> lv1s;
if (StyCode == 0)
{
lv1s = from lv1 in rootElement.Descendants("Class")
where lv1.Attribute("Code").Value.Equals("002")
select (from ltd in lv1.Descendants("Subject")
where ltd.Attribute("Course").Value.Equals("Math")
select ltd.Attribute("AllTeachers").Value.Equals("Y") ? true : false);
}
else
{
lv1s = from lv1 in rootElement.Descendants("Class")
where lv1.Attribute("Code").Value.Equals("002")
select (from ltd in lv1.Descendants("Subject")
where ltd.Attribute("Course").Value.Equals("Math")
select ltd.Attribute("SpeciaGuest").Value.Equals("Y") ? true : ltd.Elements("Topic").Attributes("Code").Where(x => x.Value.Equals("1")).FirstOrDefault() != null ? true : false);
}
所以这很简单,很好,但是有很多代码重复,这并不好。向下移动的另一条道路是创建一个方法,该方法采用表示XElement
的{{1}}以及整数ltd
,并返回指示适当值的布尔值。这是一个简单的写法:
StyCode
现在我们只需要一个查询:
private static bool GetChildFromSubject(int styCode, XElement subject)
{
if (styCode == 0)
return subject.Attribute("AllTeachers").Value.Equals("Y");
else
{
return subject.Attribute("SpeciaGuest").Value.Equals("Y") ||
subject.Elements("Topic").Attributes("Code")
.Any(x => x.Value.Equals("1"));
}
}
好多了。