分享
 
 
 

及原理

王朝学院·作者佚名  2016-05-20
窄屏简体版  字體: |||超大  

分享动态拼接ExPRession表达式组件及原理前言LINQ大家都知道,用起来也还不错,但有一个问题,当你用Linq进行搜索的时候,你是这样写的

var query = from user in db.Set<User>() where user.Username == "xxxx" select user; OK,看起来很好,不过····如果你要进行动态搜索的话··呵呵!其实方法还是挺多,只不过绕大弯

动态搜索是什么?顺便介绍下,假如你做了一个表格页面,有用户名、注册时间、等级三列,你希望实现动态组合搜索,只有当用户指定了要以用户名搜索的时候才把用户名这一列加入搜索条件,传统的sql是这么干的

string sql="select * from user "; if(userName!="") { sql+=" Where Username="+userName } 声明:只做示例,能不能运行不重要!(上面的代码一般情况下必须是有问题的)

那你用linq能不能这么干呢?呃····向下面这样

var query = from user in db.Set<User>() select user; var userName = string.Empty; if(!string.IsNullOrWhiteSpace(userName)) { query = query.Where(x => x.Username == userName); } 对,是可以这样,但在实际项目中一般是不会直接返回IQueryable接口的,也就没办法这么做。

所以,动态拼接linq就应运而生了,而linq的实质是表达式树,大家可以看IQueryable的Where扩展方法的签名,是以Expression开头的。

组件介绍及使用封装的最初目的,也是给自己用。我是懒人,所以我会把组件搞得越简单越好,下面是一个完整的使用示例

TestDataContext db = new TestDataContext(); var builder = new ExpressionBuilder<User>();//实例化组件,User是什么下面说 var filters = new List<SqlFilter>(); filters.Add(SqlFilter.Create("Id", Operation.Equal, 1)); //添加User的Id属性值等于1的搜索条件 filters.Add(SqlFilter.Create("LastLoginDate", Operation.GreaterThan, DateTime.Now));//添加User的LastLoginDate属性值大于现在的搜索条件 filters.Add(SqlFilter.Create("Username", Operation.Like, "aaaa"));//添加User的Username属性值like "aaaaa"的搜索条件 filters.Add(SqlFilter.Create("Id", Operation.In, new int[] { 1, 2, 3 }));//添加User的Id属性值在1、2、3之中的搜索条件,当Operation为In的时候,最后一个参数必须为集合 filters.Add(SqlFilter.Create("PassWord", Operation.NotEqual, "1")); filters.Add(SqlFilter.Create("Status", Operation.In, new int[] { 1 })); var where = builder.Build(filters, new Dictionary<string, string>());//根据上面的条件,拼接出表达式树 var results = db.Set<User>().Where(where).ToList();//地球人都知道 上面的代码拼接出来表达式树是这样的

var ids=new int[]{1,2,3}; var status=new int[]{1}; db.Set<User>().Where(x => x.Id == 1 && x.LastLoginDate > DateTime.Now && ids.Contains(x.Id) && x.Username.Contains("aaaa") && x.Password != "1" && status.Contains(x.Status));

注释已经说得比较清楚了,但第2行和第10行需要特别解释一下,第10行的Build方法原型如下

public Expression<Func<TParameter, bool>> Build(IList<SqlFilter> filters, Dictionary<string, string> filterNameMap) 返回值是一个Expression<Func<TParameter,bool>>类型的,其中有一个泛型参数,这个参数就是第2行实例化的时候传的User,在使用的时候,必须保证添加的每一个搜索条件的属性在User类里面有。

为什么要这样设计呢?这个感觉一下子说不清楚,等下讲原理的时候说

另外还可以看到有个filterNameMap的参数,这个主要是用来进行属性名的转换的,一般用于外键。简单说一下我当时的设计意图。

例如,有一个列表,展示的是权限系统中的角色信息,有角色名、描述、拥有的功能三列,其中第三列内容是来自功能表中的,其他是来自角色表的。

一般情况下可能会将第三列的属性名设置为Privileges,然后类型是string型,但如果说用户要按角色拥有的功能进行搜索,你不可能按字符串过滤吧?一般是按照功能Id也就是PrivilegeId过滤。

因为我用的是ExtJs(没用过这个的直接跳过这一段吧),所以问题来了,extjs传给我的参数名是Privileges,值是一个集合,因为我的Model类属性名是这个,但我后台用于过滤的真正属性是PrivilegeId,所以我需要将Privileges映射到PrivilegeId,告诉ExpressionBuilder,如果遇到了Privileges属性名的搜索条件,就将属性名换成PrivilegeId进行拼接

为了方便理解,下面是SqlFilter的源码,很简单

public class SqlFilter { public static SqlFilter Create(string propertyName, Operation operation, object value) { return new SqlFilter() { Name = propertyName, Operation = operation, Value = value }; } /// <summary> /// 字段名 /// </summary> public string Name { get; set; } /// <summary> /// 搜索操作,大于小于等于 /// </summary> public Xinchen.DbUtils.Operation Operation { get; set; } /// <summary> /// 搜索参数值 /// </summary> public object Value { get; set; } } 下面是Operation的

public enum Operation { GreaterThan, LessThan, GreaterThanOrEqual, LessThanOrEqual, NotEqual, Equal, Like, In } 组件原理还真没把握能说清楚···

其实就是拼接表达式树的原理

//假如我们要拼接x=>x.Id==1,假如x的类型为User var parameterExp = Expression.Parameter(typeof(User), "x"); //结果是这样:x=>,x是变量名 var propertyExp = Expression.Property(parameterExp, "Id"); //结果是这样:x=>x.Id,这句是为了构建访问属性的表达式 //上面这句第一个参数是你要取属性的对象表达式。我们要拼的表达式是x=>x.Id==1,==1这块先不管,其实就是x=>x.Id,那么其实我们就是对x进行取属性值,而x是parameterExp,所以第一个参数是parameterExp,第二个参数好说,就是属性名 var constExp = Expression.Constant(1); //结果是··没有结果,构建一个常量表达式,值为1(LINQ的世界,一切皆表达式树) //马上就是关键的一步了 var body = Expression.Equal(propertyExp, constExp); //结果是:x=>x.Id==1,这个··还需要解释么,很简单,不是么。创建一个相等的表达式,然后传入左边和右边的表达式 //当然到这儿还不能用,还需要继续 var lambda = Expression.Lambda<Func<User, bool>>(body, parameterExp); //这句和第二句是我学的时候最难理解的两个地方。这句是将我们的成果封装成能用的,第一个参数就是我们的成果,第二个参数是实现这个成果所需要的参数,那当然是parameterExp,然后泛型参数Func<User,bool>就是我们想把这个表达式封装成什么样的东西,此时,lambda的类型就是Expression<Fun<User,bool>>,这个时候就能用了 这是一个相当简单的表达式树,注释写得很清楚,下面来个复杂的

//假如我们要拼接x=>x.Username.Contains("aaa"),假如x的类型为User var parameterExp = Expression.Parameter(typeof(User), "x"); var propertyExp = Expression.Property(parameterExp, "Username"); //上面两句不再介绍 var containsMethod = typeof(string).GetMethod("Contains", new Type[] { typeof(string) }); //因为我们要拼接的表达式中调用了string类型的Username的Contains方法,所以反射获取string类型的Contains方法 var constExp = Expression.Constant("aaa"); //不再解释 var containsExp = Expression.Call(propertyExp, containsMethod, constExp); //结果是:x=>x.Username.Contains("aaa"),第一个参数,是要调用哪个实例的方法,这里是propertyExp,第二个是调用哪个方法,第三个是参数,理解了上一个示例,这个应该不难理解 var lambda = Expression.Lambda<Func<User, bool>>(containsExp, parameterExp); //不再解释 可以看到,第一句都是取了User的类型,所以我在设计ExpressionBuilder的使用了泛型,以供传入这个参数

原理就到这儿吧,要看更多的示例就直接看源码吧

后记继续广告···要了解相关技术的请进QQ群74522853,答案XLinq

 
 
 
免责声明:本文为网络用户发布,其观点仅代表作者个人观点,与本站无关,本站仅提供信息存储服务。文中陈述内容未经本站证实,其真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
2023年上半年GDP全球前十五强
 百态   2023-10-24
美众议院议长启动对拜登的弹劾调查
 百态   2023-09-13
上海、济南、武汉等多地出现不明坠落物
 探索   2023-09-06
印度或要将国名改为“巴拉特”
 百态   2023-09-06
男子为女友送行,买票不登机被捕
 百态   2023-08-20
手机地震预警功能怎么开?
 干货   2023-08-06
女子4年卖2套房花700多万做美容:不但没变美脸,面部还出现变形
 百态   2023-08-04
住户一楼被水淹 还冲来8头猪
 百态   2023-07-31
女子体内爬出大量瓜子状活虫
 百态   2023-07-25
地球连续35年收到神秘规律性信号,网友:不要回答!
 探索   2023-07-21
全球镓价格本周大涨27%
 探索   2023-07-09
钱都流向了那些不缺钱的人,苦都留给了能吃苦的人
 探索   2023-07-02
倩女手游刀客魅者强控制(强混乱强眩晕强睡眠)和对应控制抗性的关系
 百态   2020-08-20
美国5月9日最新疫情:美国确诊人数突破131万
 百态   2020-05-09
荷兰政府宣布将集体辞职
 干货   2020-04-30
倩女幽魂手游师徒任务情义春秋猜成语答案逍遥观:鹏程万里
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案神机营:射石饮羽
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案昆仑山:拔刀相助
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案天工阁:鬼斧神工
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案丝路古道:单枪匹马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:与虎谋皮
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:李代桃僵
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:指鹿为马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:小鸟依人
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:千金买邻
 干货   2019-11-12
 
推荐阅读
 
 
 
>>返回首頁<<
靜靜地坐在廢墟上,四周的荒凉一望無際,忽然覺得,淒涼也很美
© 2005- 王朝網路 版權所有