C#的自定义语法糖的使用详解
语法糖(Syntacticsugar),也译为糖衣语法,是由英国计算机科学家彼得·约翰·兰达(PeterJ.Landin)发明的一个术语,指计算机语言中添加的某种语法,这种语法对语言的功能并没有影响,但是更方便程序员使用。通常来说使用语法糖能够增加程序的可读性,从而减少程序代码出错的机会。
对If...Where的封装——语法糖WhereIf(如果读者已经知晓,请自行跳过)
在做条件查询的时候,我们可能经常要写这样的代码:
ListQuery(UserqueryModel) { //定义一个演示数据集 List userList=newList { newUser{UserName="燕双鹰",Phone="10369852103",Role="正派",Sex=true}, newUser{UserName="沈七七",Phone="14785203630",Role="反派",Sex=true}, newUser{UserName="步鹰",Phone="14702021596",Role="反派",Sex=true}, newUser{UserName="小玲",Phone="19469874106",Role="正派",Sex=false}, newUser{UserName="赵一平",Phone="18502369740",Role="反派",Sex=true} }; vardata=userList.AsQueryable();//转为IQueryable类型 //条件过滤 if(!string.IsNullOrEmpty(queryModel.UserName)) { data=data.Where(u=>u.UserName==queryModel.UserName); } if(!string.IsNullOrEmpty(queryModel.Phone)) { data=data.Where(u=>u.Phone==queryModel.Phone); } if(!string.IsNullOrEmpty(queryModel.Role)) { data=data.Where(u=>u.Role==queryModel.Role); } if(queryModel.Sex!=null) { data=data.Where(u=>u.Sex==queryModel.Sex); } returndata.ToList(); }
当传入的参数不为空时,才执行查询。很明显,这里大量的If-Where语句是极为简单,且不断重复出现的代码(逻辑),可以进行封装以简化操作,以简化代码。
创建泛型扩展方法WhereIf,代码如下:
publicstaticIQueryableWhereIf (thisIQueryable query,boolcondition,Expression >predicate) { returncondition ?query.Where(predicate) :query; }
该方法实现了对If-Where的封装,使用方法如下:
ListQuery(UserqueryModel) { //定义一个演示数据集 List userList=newList { newUser{UserName="燕双鹰",Phone="10369852103",Role="正派",Sex=true}, newUser{UserName="沈七七",Phone="14785203630",Role="反派",Sex=true}, newUser{UserName="步鹰",Phone="14702021596",Role="反派",Sex=true}, newUser{UserName="小玲",Phone="19469874106",Role="正派",Sex=false}, newUser{UserName="赵一平",Phone="18502369740",Role="反派",Sex=true} }; vardata=userList.AsQueryable() .WhereIf(!string.IsNullOrEmpty(queryModel.UserName),u=>u.UserName==queryModel.UserName) .WhereIf(!string.IsNullOrEmpty(queryModel.Phone),u=>u.Phone==queryModel.Phone) .WhereIf(!string.IsNullOrEmpty(queryModel.Role),u=>u.Role==queryModel.Role) .WhereIf(queryModel.Sex!=null,u=>u.Sex==queryModel.Sex); returndata.ToList(); }
之前超过8行代码的查询代码,被精简到4行,代码行数减少超过一半,可读性大幅提高,由于只是简单的封装,运行效率几乎不变。(减少大量代码,提高可读性,功能不变,效率不变,有优无缺,因此强烈建议WhereIf来代替传统的If-Where操作。)
新的问题来了,If语句还存在一个条件不满足的情况:else,WhereIf方法只封装了IfWhere,却没有封装If-Whrere-else-Where语句,如果遇到如下的查询要求,要怎么做呢?
if(!string.IsNullOrEmpty(queryModel.UserName)) { data=data.Where(u=>u.UserName==queryModel.UserName); } else { data=data.Where(u=>u.UserName=="燕双鹰");//如果查询条件为空,就查询燕双鹰的姓名 }
有三个办法可以解决这个问题:
第一个办法,是修改WhereIf方法,增加else-Where的逻辑,使其支持If-Whrere-else-Where的逻辑:
publicstaticIQueryableWhereIf (thisIQueryable query,boolcondition,Expression >truePredicate,Expression >falsePredicate=null) =>condition?query.Where(truePredicate):falsePredicate==null?query:query.Where(falsePredicate);
这样的做的缺点也是明显的:在参数condition为false时,会进行第二次逻辑判断,缺点是减低效率,优点是代码简洁。(当然,多一个逻辑判断也减低不了多少效率)
第二个方法,避免第二次逻辑判断的方式是进行方法重载,也就是写两个WhereIf方法,在新增的这个WhereIf方法中,参数falsePredicate不再设置为可空参数:
publicstaticIQueryableWhereIf (thisIQueryable query,boolcondition,Expression >truePredicate,Expression >falsePredicate) =>condition?query.Where(truePredicate):query.Where(falsePredicate);
优点是可以在不影响效率的情况下支持If-Whrere-else-Where逻辑,因为两个WhereIf方法的逻辑是差不多的,缺点是又写了简单重复的代码,不简洁。(当然,仅仅是定义它的时候不简洁,调用时候简洁程度和方法一,是一样的)
第三个方法,完全不修改WhereIf方法,仅仅在调用的时候,通过对参数condition进行取反操作,来达到目的:
vardata2=data.WhereIf(!string.IsNullOrEmpty(queryModel.UserName),u=>u.UserName==queryModel.UserName) .WhereIf(string.IsNullOrEmpty(queryModel.UserName),u=>u.UserName=="燕双鹰");
优点:方法定义最简单,缺点:在遇到If-Whrere-else-Where逻辑时,会增加代码量。
具体选择哪一种,请读者自行斟酌,如果有更好的实现方法,就留言讨论分享出来吧^_^
对for循环的封装,语法糖For
实际开发中,很多时候对for循环的使用,仅仅是将一个操作,循环指定的次数,而且其中没有break、continue这些提前终止循环的逻辑。这种简单重复的逻辑可以进行提取封装。
publicstaticvoidFor(intcount,Actionaction) { for(inti=0;i 这里使用了C#的内置泛型委托Action,发挥的作用就是将方法作为参数去传递。参数count表示循环总次数,Action的参数int,表示正在进行的循环次数,从0开始,读者可以根据需要改成从1开始(这里从1开始好,还是从0开始好,待定)。
调用:
SyntacticSugar.For(1,p=>{inta=p+8;data2.Remove(data2[a]);});如果认为这样调用麻烦,可以在参数count前加this,使之变为扩展方法,以简化调用。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持毛票票。
声明:本文内容来源于网络,版权归原作者所有,内容由互联网用户自发贡献自行上传,本网站不拥有所有权,未作人工编辑处理,也不承担相关法律责任。如果您发现有涉嫌版权的内容,欢迎发送邮件至:czq8825#qq.com(发邮件时,请将#更换为@)进行举报,并提供相关证据,一经查实,本站将立刻删除涉嫌侵权内容。