使用Expression做Linq的參數化排序
Linq非常的好用,減少大量的資料庫操作手序,使用具名的類別,減少了在程式中寫SQL寫錯字的可能性,問題來了,如果我想用QueryString中的參數,作為排序的依據,但是因為是具名的類別,不能指定字串,剛開始我是用switch一個一個指定,但欄位一多就覺得這方法很笨,在搜尋更好的方法中發現使用System.Linq.Expressions.Expression可以決解這個問題。 如果各位有仔細看,會發現System.Linq.Queryable下的Method參數都有Expression,如本篇要用的OrderBy。
| 1 2 3 4 | publicstaticIOrderedQueryable<TSource> OrderBy<TSource, TKey>(thisIQueryable<TSource> source, Expression<Func<TSource, TKey>> keySelector);publicstaticIOrderedQueryable<TSource> OrderBy<TSource, TKey>(thisIQueryable<TSource> source, Expression<Func<TSource, TKey>> keySelector, IComparer<TKey> comparer);publicstaticIOrderedQueryable<TSource> OrderByDescending<TSource, TKey>(thisIQueryable<TSource> source, Expression<Func<TSource, TKey>> keySelector);publicstaticIOrderedQueryable<TSource> OrderByDescending<TSource, TKey>(thisIQueryable<TSource> source, Expression<Func<TSource, TKey>> keySelector, IComparer<TKey> comparer); | 
那什麼是Expression呢?
簡單說是動態產生Delegate,真的執行的時候才Compile,你會說在Visaul Studio明明就Compile啦,怎麼又到執行才Compile?
實際上在Visaul Studio Compile時,會先編成Expression,等到執行時再將Expression Compile成Delegate,如下面範例。
| 1 2 3 4 5 6 | //在Visual Studio這樣寫的東西。.OrderBy(x=>x.Name);//Compile後其實是編成Expression(從.Net Reflector中取得後,有修改成易讀格式)。Parameter p = Expression.Parameter(typeof(type), "x");// 參數XExpression.Lambda<Func<type, string>>(Expression.Property(p, "Name"), p)); // x.Name//並不是所有的Lambda都會編成Expression,而是只有參數是Expression才會編成Expression,其他的還是直接編成Method。 | 
EXPRESSION TREES, TAKE TWO – INTRODUCING SYSTEM.LINQ.EXPRESSIONS V4.0
Building LINQ Queries at Runtime in C#
所以也可以自己產生Expression做OrderBy的參數,範例如下:
| 1 2 3 4 5 6 7 8 9 10 11 12 13 | if(!string.IsNullOrEmpty(this.Request.QueryString["Order"])){   // 產生Expression   var param = Expression.Parameter(typeof(Project), "x");   var orderExpression = Expression.Lambda<Func<Project, object>>(Expression.Property(param, this.Request.QueryString["Order"]), param);   if(this.Request.QueryString["OrderDirection"] == "Desc")   {       query = query.OrderByDescending(orderExpression);   }   else   {       query = query.OrderBy(orderExpression);   }} | 
但上面的範例遇到遇到nullable的型別會掛到所以又寫了幾個Extension來使用
| 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 | publicstaticclassIQueryableExtension{    privatestaticMethodInfo orderbyInfo = null;    privatestaticMethodInfo orderbyDecInfo = null;    publicstaticIQueryable<T> OrderBy<T>(thisIQueryable<T> query, stringproperty) where T : class    {           Type entityType = typeof(T);        Type entityPropertyType = entityType.GetProperty(property).PropertyType;        var orderPara = Expression.Parameter(entityType, "o");        var orderExpr = Expression.Lambda(Expression.Property(orderPara, property), orderPara);        if(orderbyInfo==null)        {            //因為呼叫OrderBy需要知道型別,不知道的情況下無法直接呼叫,所以用反射的方式呼叫            //泛型的GetMethod很難,所以用GetMethods在用Linq取出Method,找到後快取。            orderbyInfo = typeof(Queryable).GetMethods().Single(x => x.Name == "OrderBy"&& x.GetParameters().Length == 2);        }        //因為是泛型Mehtod要呼叫MakeGenericMethod決定泛型型別        returnorderbyInfo.MakeGenericMethod(newType[] { entityType, entityPropertyType }).Invoke(null, newobject[] { query, orderExpr }) asIQueryable<T>;    }    publicstaticIQueryable<T> OrderByDescending<T>(thisIQueryable<T> query, stringproperty)    {        Type entityType = typeof(T);        Type entityPropertyType = entityType.GetProperty(property).PropertyType;        var orderPara = Expression.Parameter(entityType, "o");        var orderExpr = Expression.Lambda(Expression.Property(orderPara, property), orderPara);        if(orderbyDecInfo == null)        {            orderbyDecInfo = typeof(Queryable).GetMethods().Single(x => x.Name == "OrderByDescending"&& x.GetParameters().Length == 2);        }        returnorderbyDecInfo.MakeGenericMethod(newType[] { entityType, entityPropertyType }).Invoke(null, newobject[] { query, orderExpr }) asIQueryable<T>;    }} | 
使用Expression做Linq的參數化排序的更多相关文章
- 02.C#可空類型、默認參數、LINQ(一章1.3-1.4)
		利用上班時間發個隨筆,不知領導會不會看到,可能會有同事看到也說不定啊:) 關于可空類型,在C#1中沒有這個概念,在C#3中引入的.那比如我們要實現一個表示人的類,人有名字和年齡兩個屬性,如何表示一個沒 ... 
- TestNG的參数化測试、共享线程池配置、參数默认值配置
		在使用TestNG进行測试时,常常会使用到一些參数化配置,比方数据库.连接池.线程池数. 使用TestNG的參数@Parameter注解进行自己主动化读取 原创文章,版权全部.同意转载,标明出处:ht ... 
- 使用 new Q_max_capacity 參數,同樣 loading 下,粗估耗電量(UI 上的 %)。
		Precondition : 除了 Q_max 外,其它參數皆同. old Q_max_capacity : 1500 mAh new Q_max_capacity : 2200 mAh 有一個 lo ... 
- QTP自带订票实现循环执行,參数化和将异常提示信息输出
		做这个样例主要是为了积累一些较基础的知识,以便日后可參考学习 这个样例是一个订票的C/Sclient程序 一.业务需求: 1.实现异常和正常数据登录窗体,系统对数据进行校验 2.登录成功后.进行订票业 ... 
- Linux下安裝Oracle database內核參數設置
		參考:1529864.1 ************************************************** RAM ... 
- 同一個Loader對象傳入不同參數時,从数据库中查询的結果每次都一樣
		發現問題: LoaderManager().initLoader()方法調用時會根據第一個參數ID去判斷是否已經存在一個Loader加載器,如果存在則複 用,不存在則建一個新的加載器.由於我第一次已經 ... 
- shell傳遞參數
		Shell 传递参数 我们可以在执行 Shell 脚本时,向脚本传递参数,脚本内获取参数的格式为:$n.n 代表一个数字,1 为执行脚本的第一个参数,2 为执行脚本的第二个参数,以此类推…… 比如我們 ... 
- 設定 gpio 為 讀取用途,需注意的參數
		Schematic 解說 上面的 線路圖, R1 R2 只能有一個被接上, R3 R4 只能有一個被接上, 是使用 gpio 讀取 電壓 判斷為0 或是 1 這時的 gpio 設定,其中一個參數需設為 ... 
- 在 kernel 下打出 有帶參數的log。 怪異現象與解決方式。
		code battery_log(BAT_LOG_CRTI, "youchihwang abc10010 xxxaaa8-2\r\n"); battery_log(BAT_LOG_ ... 
随机推荐
- 详解mysql int类型的长度值问题
			我的朋友海滨问我mysql在建表的时候int类型后的长度代表什么? 是该列允许存储值的最大宽度吗? 为什么我设置成int(1), 也一样能存10,100,1000呢. 当时我虽然知道int(1),这个 ... 
- 重新编译安装gcc-4.1.2(gcc版本降级)之TFS安装
			wget http://gcc.parentingamerica.com/releases/gcc-4.1.2/gcc-4.1.2.tar.gz tar -zxfv gcc-4.1.2.tar.gz ... 
- IIS安装时,添加/编辑应用程序扩展名映射 确定按钮不可用。
			原因是:执行文件的路径太长,需要激活按钮. 方法一:选择较短路径的执行文件,先激活按钮. 方法二:点击该路径,就可以激活确认按钮了. 
- iOS 中添加lib型target库的依赖问题
			今天在编码时遇到一个问题,总提示我找不到系统库文件. 我的项目结构类似下图 在TestLib中有引用CoreLocation库的类.但是CoreLocation库需要加在PhotoInfoDemo对象 ... 
- codeforces  485A.Factory  解题报告
			题目链接:http://codeforces.com/problemset/problem/485/A 题目意思:给出 a 和 m,a 表示第一日的details,要求该日结束时要多生产 a mod ... 
- codeforces  475A.Bayan Bus  解题报告
			题目链接:http://codeforces.com/problemset/problem/475/A 题目意思:输入一个整数 k(0 ≤ k ≤ 34),表示participants的人数,需要在一 ... 
- sublime text3侧边栏主题不生效问题解决
			sublime text3主题插件: Seti_UI 插件安装: 在线安装:需要FQ window: ctrl+shift+p 找install package:之后搜索 Seti_UI 安装完成后需 ... 
- 如何手动修改XP系统属性中的技术支持信息
			\windows\system32目录下有个oeminof.ini,里面是OEM显示的文字信息,把相应项目修改即可,OEM图片使用的是本目录下的OEMlogo.bmp(图片:创建一个图形文件,像素尺寸 ... 
- eclipse 和 android studio 快捷键对比
			操作 studio eclipse debug/run 计算变量值 alt+F8 ctrl+shift+I 跳到下一步 F8 F6 跳到下一个断点 shift+F8 F8 进入到代码 F7 F5 ... 
- ****CSS各种居中方法
			水平居中的text-align:center 和 margin:0 auto 这两种方法都是用来水平居中的,前者是针对父元素进行设置而后者则是对子元素.他们起作用的首要条件是子元素必须没有被flo ... 
