2013-01-10 33 views
1

您好,我想用N多的动态,其中条件,我的解决方案..的LINQ to XML与动态n个有条件的地方

和我的情况我已经获取所有的XML元素,所有必须包含数组(arrSearchContents)的所有元素,它是通过分割strSearchContent的所有单词来构建的。

<Mobiles> 
    <Mobile Id="1"> 
    <MDetails ModelNo="1" Desc="x phone color red"/> 
    <MDetails ModelNo="2" Desc="x phone color green"/> 
    <MDetails ModelNo="3" Desc="x phone color blue"/> 
    <MDetails ModelNo="4" Desc="x phone color black"/> 
    <MDetails ModelNo="5" Desc="x phone color yellow"/> 
    <MDetails ModelNo="6" Desc="x phone color pink"/> 
    <MDetails ModelNo="7" Desc="x phone color gray"/> 
    <MDetails ModelNo="8" Desc="x phone color silver"/> 
    <MDetails ModelNo="9" Desc="x phone color orange"/> 
    </Mobile> 
    <Mobile Id="2"> 
    <MDetails ModelNo="11" Desc="y phone color red"/> 
    <MDetails ModelNo="12" Desc="y phone color green"/> 
    <MDetails ModelNo="13" Desc="y phone color blue"/> 
    <MDetails ModelNo="14" Desc="y phone color black"/> 
    <MDetails ModelNo="15" Desc="y phone color yellow"/> 
    <MDetails ModelNo="16" Desc="y phone color pink"/> 
    <MDetails ModelNo="17" Desc="y phone color gray"/> 
    <MDetails ModelNo="18" Desc="y phone color silver"/> 
    <MDetails ModelNo="19" Desc="y phone color orange"/> 
    </Mobile> 
    <Mobile Id="3"> 
    ...... 
    <Mobile> 
    ...... 
</Mobiles> 

我想要一个类似波纹管代码的解决方案。 任何机构可以整顿代码

public void doSearch(string strSearchContent) 
{ 
string[] arrSearchContents; 
arrSearchContents = strSearchContent.Split(' '); 

string fileName = HttpContext.Current.Server.MapPath(@"~\XMLFiles/Mobile.xml"); 
XDocument doc = null; 
    if (System.IO.File.Exists(fileName)) 
    { 
     doc = XDocument.Load(fileName); 

     IEnumerable<XElement> list1 = 
      from elements in doc.Descendants("MDetails") 
      where 
       for(int i=0;i<arrSearchContents.Length;i++) 
       { 
        elements.Attribute("Desc").Value.Contains(arrSearchContents[i]) && 
       } 
      select elements; 
    } 
} 

选择列表应包含搜索字符串的所有关键词。

as per example : if strSearchContent = "phone color blue" 

then as a result it should return : 

<MDetails ModelNo="3" Desc="x phone color blue"/> 
<MDetails ModelNo="13" Desc="y phone color blue"/> 

回答

4
from d in doc.Descendants("MDetails") 
let desc = (string)d.Attribute("Desc") 
where desc != null && 
     arrSearchContents.All(keyword => desc.Contains(keyword)) 
select d; 

但它也会匹配“bluephonecolor”。如果你想匹配的话,你可以用文字使用正则表达式,或分裂描述:

!arrSearchContents.Except(desc.Split()).Any() 
+0

非常感谢。它真的帮助我即兴编写我的代码... – pks

+0

@pks欢迎,顺便说一句,如果所有'MDetails'元素具有'Desc'属性,你可以省略'desc!= null'检查。还要记住,默认情况下不带参数的'String.Split()'按字串分割。如果答案为你工作,你可以接受它:) –

+1

正确答案根据我的要求。 – pks

2

你应该检查一下节点属性包含价值,而不是老虎钳诗句

var str = @"<Mobiles> 
        <Mobile Id=""1""> 
         <MDetails ModelNo=""1"" Desc=""x phone color red""/> 
         <MDetails ModelNo=""2"" Desc=""x phone color green""/> 
         <MDetails ModelNo=""3"" Desc=""x phone color blue""/> 
         <MDetails ModelNo=""4"" Desc=""x phone color black""/> 
         <MDetails ModelNo=""5"" Desc=""x phone color yellow""/> 
         <MDetails ModelNo=""6"" Desc=""x phone color pink""/> 
         <MDetails ModelNo=""7"" Desc=""x phone color gray""/> 
         <MDetails ModelNo=""8"" Desc=""x phone color silver""/> 
         <MDetails ModelNo=""9"" Desc=""x phone color orange""/> 
        </Mobile> 
        <Mobile Id=""2""> 
         <MDetails ModelNo=""11"" Desc=""y phone color red""/> 
         <MDetails ModelNo=""12"" Desc=""y phone color green""/> 
         <MDetails ModelNo=""13"" Desc=""y phone color blue""/> 
         <MDetails ModelNo=""14"" Desc=""y phone color black""/> 
         <MDetails ModelNo=""15"" Desc=""y phone color yellow""/> 
         <MDetails ModelNo=""16"" Desc=""y phone color pink""/> 
         <MDetails ModelNo=""17"" Desc=""y phone color gray""/> 
         <MDetails ModelNo=""18"" Desc=""y phone color silver""/> 
         <MDetails ModelNo=""19"" Desc=""y phone color orange""/> 
        </Mobile></Mobiles>"; 

    var doc = XDocument.Parse(str); 
    var strSearchContent = "phone color blue"; 
    IEnumerable<XElement> list1 = doc.XPathSelectElements("//MDetails") 
            .Where(node => node.Attribute("Desc").Value.Contains(strSearchContent)); 

    //LinqPad specific print call 
    list1.Dump(); 

打印

<MDetails ModelNo="3" Desc="x phone color blue" /> 
<MDetails ModelNo="13" Desc="y phone color blue" /> 

因此产生的代码看起来像这样(请注意,我改变了方法的返回类型):

public IEnumerable<XElement> doSearch(string strSearchContent) 
{ 
    string fileName = HttpContext.Current.Server.MapPath(@"~\XMLFiles/Mobile.xml"); 

    if (File.Exists(fileName)) 
    { 
     XDocument doc = XDocument.Load(fileName); 
     return doc.XPathSelectElements("//MDetails") 
        .Where(node => node.Attribute("Desc").Value.Contains(strSearchContent)); 
    } 

    return Enumerable.Empty<XElement>(); 
} 
+0

顺便说一句,你其实可以把条件转换中的XPath:“// MDetails [包含(@Desc,{0}) ]“(其中{0}是您的搜索项) – Efrain

+0

是的,当然,但我不喜欢在LINQ to XML中创建长的XPath表达式。 XPath并不是很多开发者所熟知的,所以我只用它来指定DOM中的所有元素。否则由LINQ以静态类型的方式完成。再次 - 在这里可以有很多选择。谢谢你的回应 –

+0

是真的。实际上你的LINQ代码读起来确实很容易。 – Efrain