0

我正在使用以下 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 会很好吗?

4

4 回答 4

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"))
           };

顺便说一句,您返回的是具有布尔Children属性的对象列表。我相信你需要更多的结果数据。

于 2013-01-21T19:00:59.640 回答
1
bool flag = boolexpression ? true : false;

是相同的

bool flag = boolexpression;

因此,请停止使用三元来不必要地使您的代码复杂化。


您可以将您的选择子句编写为:

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)
于 2013-01-21T19:09:13.017 回答
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")
    )
于 2013-01-21T19:09:37.567 回答
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表示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"));
    }
}

现在我们只需要一个查询:

XElement rootElement = XElement.Load(@"test.xml");
int StyCode = 0;
var 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 GetChildFromSubject(StyCode, ltd));

好多了。

于 2013-01-21T19:13:23.387 回答