0330LINQ学习资料整理.docx
《0330LINQ学习资料整理.docx》由会员分享,可在线阅读,更多相关《0330LINQ学习资料整理.docx(15页珍藏版)》请在冰豆网上搜索。
![0330LINQ学习资料整理.docx](https://file1.bdocx.com/fileroot1/2022-11/25/9f90390c-f2ae-4037-b0e8-32f5c8c27784/9f90390c-f2ae-4037-b0e8-32f5c8c277841.gif)
0330LINQ学习资料整理
LINQ-基础知识
1.LINQ的读法:
(1)link
(2)linq
2.LINQ的关键词:
from,select,in,where,groupby,orderby…
3.LINQ的注意点:
必须以select或者是groupby结束。
4.LINQ的语义:
from临时变量in集合对象或数据库对象
where条件表达式
[orderby条件]
select临时变量中被查询的值
[groupby条件]
LINQ的查询返回值的类型是临时变量的类型,可能是一个对象也可能是一个集合。
并且LINQ的查询表达式是在最近一次创建对象时才被编译的。
LINQ的查询一般跟var关键字一起联用(什么是var?
匿名对象)。
以下的两个查询表达式是一样的效果:
varq=fromnameinmethods
where(name.Name.Length>15)
selectname;
5.LINQ的全称:
Language-IntegratedQuery
6.LINQ的分类:
LINQtoObject,LINQtoXML,LINQtoSQL,LINQtoADO.NET
Q:
为何LINQ查询语法是以from关键字开头的,而不是以select关键字开头的?
select开头这种写法跟SQL的写法更接近,更易懂呀?
A:
简单来说,为了IDE的智能感知(Intelisence)这个功能,select关键字放在后面了。
假设你要书写这样的代码:
Selectp.Name,p.AgeFrompInpersonsWherexxx,代码是一个个字符输入的。
我们在写到pinpersons之前,p的类型是无法推测的,所以写Selectp.的时候,Name之类的属性不会弹出智能提示来。
这样就需要先去写From这句,再回来写Select。
微软IDE组经过反复考虑决定,还不如就把Select写到后面了。
于是编程语言中的写法就确定这样来写了。
我们再来看一个稍稍复杂的LINQ查询:
在我们罗列的语言字符串中,我们希望按照字符长短,分类罗列出来,实现代码如下:
staticvoidMain(string[]args)
{
//定义数组
string[]languages={"Java","C#","C++","Delphi","VB.net","VC.net","C++Builder","Kylix","Perl","Python"};
//定义查询
varquery=fromiteminlanguages
orderbyitem
groupitembyitem.LengthintolengthGroups
orderbylengthGroups.Keydescending
selectlengthGroups;
foreach(variteminquery)
{
Console.WriteLine("stringsoflength",item.Key);
foreach(varvalinitem)
{
Console.WriteLine(val);
}
}
Console.ReadLine();
}
其中的into关键字表示将前一个查询的结果视为后续查询的生成器,这里是跟groupby一起使用的。
LINQ中的Groupby不要跟SQL中的Groupby混淆,SQL由于是二维结构,Groupby的一些逻辑受二维结构的约束,无法象LINQ中的Groupby这么灵活。
LINQ-Linq的内部执行原理浅析
LINQ(LanguageIntegratedQuery)是VisualStudio2008中的领军人物。
借助于LINQ技术,我们可以使用一种类似SQL的语法来查询任何形式的数据。
目前为止LINQ所支持的数据源有SQLServer、XML以及内存中的数据集合。
开发人员也可以使用其提供的扩展框架添加更多的数据源,例如MySQL、Amazon甚至是GoogleDesktop。
一般来讲,这类查询语句的一个重要特点就是可以并行化执行。
虽然有些情况下并行可能会带来一些问题,但这种情况非常少见。
这样也就水到渠成地引出了PLINQ这个并行处理的LINQ类库。
PLINQ原名为ParallelLINQ,支持XML和内存中的数据集合。
执行于远程服务器上的查询语句(例如LINQtoSQL)显然无法实现这个功能。
将LINQ语句转换为PLINQ语句极为简单——只需要在查询语句中From子句所指定的数据源的最后添加.AsParallel()即可。
随后Where、OrderBy和Select子句将自动改为调用这个并行的LINQ版本。
据MSDNMagazine介绍,PLINQ可以以三种方式执行。
第一种是管道处理:
一个线程用来读取数据源,而其他的线程则用来处理查询语句,二者同步进行——虽然这个单一的消费线程可能并不那么容易与多个生产线程同步。
不过若是能够仔细配置好负载平衡的话,仍然会极大地减少内存占用。
第二种模式叫做“stopandgo”,用于处理结果集需要被一次返回时(例如调用ToList、ToArray或对结果排序)的情况。
在这种模式下,将依次完成各个处理过程,并将结果统一返回给消费线程。
这个模式在性能上将优于第一种模式,因为它省去了用来保持线程同步所花费的开销。
最后一种方法叫做“invertedenumeration”。
该方法并不需要实现收集到所有的输出,然后在单一的线程中处理,而是将最终调用的函数通过ForAll扩展传递到每个线程中。
这是目前为止最快的一种处理模式,不过这需要传递到ForAll中的函数是线程安全的,且最好不包含任何lock之类的互斥语句。
若是PLINQ中任意的一个线程抛出异常,那么所有的其他线程将会被终止。
若是抛出了多个异常,那么这些异常将被组合成一个MultipleFailuresException类型的异常,但每个异常的调用堆栈仍会被保留。
C#3.0新语言特性和改进
C#3.0新语言特性和改进包括:
∙自动属性(Auto-Implemented【实施】Properties)
∙隐含类型局部变量(LocalVariableTypeInference【推理】)
∙匿名类型(AnonymousTypes)
∙对象与集合初始化器(ObjectandCollectionInitializers【初始化】)
∙扩展方法(ExtensionMethods)
∙Lambda表达式和Lambda表达式树(LambdaExpressionandLambdaExpressionTrees)
自动属性(Auto-ImplementedProperties)
自动属性可以避免原来这样我们手工声明一个私有成员变量以及编写get/set逻辑,在VS2008中可以像下面这样编写一个类,编译器会自动地生成私有变量和默认的get/set操作。
你也可以分别定义get和set的“protected”等访问级别。
在.Net2.0框架下,我们可以这样写一个User类:
publicclassUser
{
privateint_id;
privatestring_name;
privateint_age;
publicintId
{
get{return_id;}
set{_id=value;}
}
publicstringName
{
get{return_name;}
set{_name=value;}
}
publicintAge
{
get{return_age;}
set{_age=value;}
}
}
现在,可以这样简化:
publicclassUser
{
publicintId{get;set;}
publicstringName{get;set;}
publicintAge{get;set;}
}
像上面这样的空的get/set属性的话,它会自动为你在类中生成一个私有成员变量,对这个变量实现一个公开的getter和setter。
我们可以使用.NET开发环境所提供的ildasm.exe(IL代码反汇编器)工具来分析程序集或者模块的内容。
隐含类型局部变量(LocalVariableTypeInference)
C#3.0引进了var这个新关键字,在声明局部变量时可用于替代原先的类型名,即当一个变量声明标识为var类型并且该范围域中没有var名称类型存在,那么这个声明就称为隐含类型局部变量。
如下(等同于//后面的显式声明):
vari=5;//int
varj=23.56;//double
vark="CSharp";//string
varx;//错误
vary=null;//错误
varz={1,2,3};//错误
在调试状态下,编译器解释如下
隐含类型局部变量要点
1.var为关键字,可以根据后面的初始化语句自动推断类型,这个类型为强类型。
2.初始化语句必须为表达式,不可以为空。
且编译时可以推断类型。
一旦初始化之后,只可以存储这种类型。
3.var声明的仅限于局部变量,不可用于字段。
亦可以用于for,foreach,using等语句中。
4.数组也可以作为隐含类型。
5.初始化语句不能是一个自身的对象或者集合初始化器,但是他可以是包含一个对象或者初始化器的一个new表达式。
6.如果局部变量声明包含了多个声明符,其类型必须相同。
匿名类型(AnonymousTypes)
匿名类型允许定义行内类型,无须显式定义类型。
常和var配合使用来声明匿名类型。
varp1=new{Id=1,Name="YJingLee",Age=22};//属性也不需要申明
varp2=new{Id=2,Name="XieQing",Age=25};
p1=p2;//p1,p2结构相同,可以互相赋值
在这里编译器会认为p1,p2相当于:
publicclassSomeType
{
publicintId{get;set;}
publicstringName{get;set;}
publicintAge{get;set;}
}
那么数组怎么定义呢?
使用"new[]"关键字来声明数组,加上数组的初始值列表。
像这样:
varintArray=new[]{2,3,5,6};
varstrArray=new[]{"Hello","World"};
varanonymousTypeArray=new[]
{
new{Name="YJingLee",Age=22},
new{Name="XieQing",Age=25}
};
vara=intArray[0];
varb=strArray[0];
varc=anonymousTypeArray[1].Name;
匿名类型要点
1.可以使用new关键字调用匿名初始化器创建一个匿名类型的对象。
2.匿名类型直接继承自System.Object。
3.匿名类型的成员是编译器根据初始化器推断而来的一些读写属性
对象与集合初始化器(ObjectandCollectionInitializers)
对象初始化器(ObjectInitializers):
.NET2.0框架中的类型非常依赖于属性。
当生成对象实例和使用新的类型时,在.Net2.0时候我们像这样写:
Useruser=newUser();
user.Id=1;
user.Name="YJingLee";
user.Age=22;
在VS2008中,编译器会自动地生成合适的属性setter代码,使得原来几行的属性赋值操作可以在一行完成。
我们可以这样简化:
像这样,
对象初始化器由一系列成员对象组成,其对象必须初始化,用逗号间隔,使用{}封闭。
Useruser=newUser{Id=1,Name="YJingLee",Age=22};
又例如,我把二个人加到一个基于泛型的类型为User的List集合中:
Listuser=newList{
newUser{Id=1,Name="YJingLee",Age=22},
newUser{Id=2,Name="XieQing",Age=25},
};
如果有相同名字和类型的两个对象初始化器将会产生相同的实例,可以相互赋值。
例如:
Useruser=newUser{Id=1,Name="YJingLee",Age=22};
Useruser2=newUser{Id=2,Name="XieQing",Age=25};
user=user2;
除了在初始化类时设置简单的属性值外,对象初始化器特性也允许我们设置更复杂的嵌套(nested)属性类型。
例如我们可以在上面定义的User类型同时拥有一个属于Address类型的叫“Address”的属性:
Useruser=newUser
{
Id=1,
Name="YJingLee",
Age=22,
Address=newAddress
{
City="NanJing",
Zip=21000
}
};
集合初始化器(CollectionInitializers):
集合初始化器由一系列集合对象组成,用逗号间隔,使用{}封闭。
集合初始化器可以简化把几个对象一起添加到一个集合,编译器会自动为你做集合插入操作。
例如把七个数加到一个基于泛型的类型为int的List集合中
Listnum=newList{0,1,2,6,7,8,9};
对象与集合初始化器要点
1.对象初始化器实际上利用了编译器对对象中对外可见的字段和属性进行按序赋值。
2.对象初始化器允许只给一部分属性赋值,包括internal访问级别
3.对象初始化器可以结合构造函数一起使用,并且构造函数初始化先于对象初始化器执行。
4.集合初始化器会对初始化器中的元素进行按序调用ICollection.Add(T)方法。
5.注意对象初始化器和集合初始化器中成员的可见性和调用顺序。
6.对象与集合初始化器同样是一种编译时技术。
扩展方法(ExtensionMethods)
往往我们需要对CLR类型进行一些操作,但苦于无法扩展CLR类型的方法,只能创建一些helper方法,或者继承类。
我们来修改上面的User类:
publicclassUser
{
publicintId{get;set;}
publicstringName{get;set;}
publicintAge{get;set;}
publicstringRead()
{
return"Id:
"+Id+"姓名:
"+Name+"年龄:
"+Age;
}
}
然后调用
varuser=new{Id=1,Name="YJingLee",Age=22};
varstr=user.Read();
现在有了扩展方法就方便多了。
扩展方法允许开发人员往一个现有的CLR类型的公开契约(contract)中添加新的方法,而不用生成子类或者重新编译原来的类型。
扩展方法有助于把今天动态语言中流行的对ducktyping的支持之灵活性,与强类型语言之性能和编译时验证融合起来。
——引用Scott博文
扩展方法是可以通过使用实例方法语法调用的静态方法。
效果上,使得附加的方法扩展已存在类型和构造类型成为可能。
他可以对现有类功能进行扩充,从而使该类型的实例具有更多的方法(功能)。
扩展方法允许我们在不改变源代码的情况下扩展(即添加不能修改)现有类型中的实例方法。
扩展方法给我们一个怎样的思路呢?
我们一步一步做一下!
首先声明扩展方法:
1.通过指定关键字this修饰方法的第一个参数。
2.注意扩展方法仅可声明在静态类中。
3.扩展方法具备所有常规静态方法的所有能力,可以使用实例方法语法来调用。
接着就可以调用扩展方法了。
下面通过一个具体的实例分析一下:
例如我们要检查一个字符串变量是否是合法的电子邮件地址?
在.Net2.0框架下像这样:
varemail="leeyongjing@";
if(EmailValidator.IsValid(email))
{
Response.Write("YJingLee提示:
这是一个正确的邮件地址");
}
而使用扩展方法的话,我可以添加“IsValidEmailAddress()”方法到string类本身中去,该方法返回当前字符串实例是否是个合法的字符串。
if(email.IsValidEmailAddress())
{
Response.Write("YJingLee提示:
这是一个正确的邮件地址");
}
我们是怎么把这个IsValidEmailAddress()方法添加到现有的string类里去的呢?
先定义一个静态类,再定义“IsValidEmailAddress”这个静态的方法来实现的。
publicstaticclassExtensions//静态类
{
publicstaticboolIsValidEmailAddress(thisstrings)
//静态方法和this
{
Regexregex=newRegex(@"^[\w-\.]+@([\w-]+\.)+[\w-]{2,4}$");
returnregex.IsMatch(s);
}
}
注意,上面的静态方法在第一个类型是string的参数变量前有个“this”关键词,这告诉编译器,这个特定的扩展方法应该添加到类型为“string”的对象中去。
然后在IsValidEmailAddress()方法实现里,我可以访问调用该方法的实际string实例的所有公开属性/方法/事件,取决于它是否是合法电子邮件地址来返回true/false。
扩展方法不仅能够应用到个别类型上,也能应用到.NET框架中任何基类或接口上。
即可用于整个.NET框架丰富的可组合的框架层扩展。
扩展方法要点
1.扩展方法的本质为将实例方法调用在编译期改变为静态类中的静态方法调用。
事实上,它确实拥有静态方法所具有的所有功能。
2.扩展方法的作用域是整个namespace可见的,并且可以通过usingnamespace来导入其它命名空间中的扩展方法。
3.扩展方法的优先级:
现有实例方法优先级最高,其次为最近的namespace下的静态类的静态方法,最后为较远的namespace下的静态类的静态方法。
4.扩展方法是一种编译时技术,注意与反射等运行时技术进行区别,并慎重使用。
Lambda表达式和Lambda表达式树(LambdaExpressionandLambdaExpressionTrees)
Lambda表达式
我们从“所有字符串查找包含YJingLee子字符串”说起。
在C#2.0中,匿名方法允许我们以内联的方式来实现委托实例,它提供强大的函数式编程语言,但是标记显得相当的冗长和带有强制性。
我们使用C#2.0中的匿名方法查找,代码如下:
varinString=list.FindAll(delegate(strings)
{returns.Indexof("YJingLee")>=0;});
现在可以使用C#3.0带来的Lambda表达式允许我们使用一种更接近人的思维、更自然的方式来实现类似于匿名方法同样的效果,看下面的代码多么简洁:
varinString=list.FindAll(s=>s.Indexof("YJingLee")>=0);
Lambda表达式格式:
(参数列表)=>表达式或语句块
具体意义:
定义Lambda接受参数列表,运行表达式或语句块返回表达式或语句块的值传给这个参数列表。
Lambda表达式参数类型可以是隐式类型或显式类型。
在显式列表中,每个参数的类型是显式指定的,
在隐式列表中,参数的类型由Lambda表达式出现的语境自动推断类型。
Lambda表达式的参数列表可以有一个或多个参数,或者无参数。
在有单一的隐型参数的lambda表达式中,圆括号可以从参数列表中省略。
例如:
(x,y)=>x*y;//多参数,隐式类型=>表达式
x=>x*10;//单参数,隐式类型=>表达式
x=>{returnx*10;};//单参数,隐式类型=>语句块
(intx)=>x*10;//单参数,显式类型=>表达式
(intx)=>{returnx*10;};//单参数,显式类型=>语句块
()=>Console.WriteLine();//无参数
下面看这个例子:
在前面的帖子中,我们写了一个User类及增加了2个人,接下来,我们使用由LINQ提供的新的Where和Average方法来返回集合中的人的一个子集,以及计算这个集合中的人的平均年龄:
Listuser=newList{
newUser{Id=1,Name="YJingLee",Age=22},
newUser{Id=2,Name="XieQing",Age=25},
};
//获取特定人时所用的过滤条件,p参数属于User类型
varresults=user.Where(p=>p.Name=="YJingLee").ToList();
//用User对象的Age值计算平均年龄
varaverage=user.Average(p=>p.Age);
效果图如下:
对这个Lambda表达式做个简要分析:
varresultsdelegate=user.Where(delegate(Userp)
{
returnp.Name=="YJingLee";//返回一个布尔值
});
varaveragedelegate=user.Average(delegate(