溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點(diǎn)擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

Linq中如何實(shí)現(xiàn)Linq Func<T>

發(fā)布時間:2021-12-02 09:23:15 來源:億速云 閱讀:134 作者:小新 欄目:編程語言

小編給大家分享一下Linq中如何實(shí)現(xiàn)Linq Func<T>,相信大部分人都還不怎么了解,因此分享這篇文章給大家參考一下,希望大家閱讀完這篇文章后大有收獲,下面讓我們一起去了解一下吧!

在Linq中,任何接收λ表達(dá)式(委托類型)的方法都可以轉(zhuǎn)換為接收相同委托類型的Expression<T>的方法,并且不需要更改客戶機(jī)代碼。例如:

privatestaticvoid DoSomething(Predicate<Mock> predicate)

可以替換為:

privatestaticvoid DoSomething(  Expression<Predicate<Mock><Mock>> predicate)

在上述兩種情況下,調(diào)用代碼可以是相同的λ表達(dá)式:

DoSomething(x => x.Value > 25);

這里發(fā)生的情況是,編譯器不會將指針傳入到第二個方法簽名的匿名委托中,而是生成以表達(dá)式樹的形式構(gòu)建AST(抽象語法樹)的IL代碼。如果您打開Reflector(我的類型反射類的名字也由此而來,它是任何高級開發(fā)人員都應(yīng)該經(jīng)常使用的最偉大的工具)并取消對DoSomething的方法調(diào)用,就可以看到:

ParameterExpression expression1 =  Expression.Parameter(typeof(Mock), "x");  Program.DoSomething(  Expression.Lambda<Predicate<Mock>>(  Expression.GT(Expression.Field(  expression1, fieldof(Mock.Value)),  Expression.Constant(0x19, typeof(int))),  newParameterExpression[]{expression1 })  );

這里您可以看到編譯器如何使用Expression類上的靜態(tài)方法構(gòu)建整個表達(dá)式(我對API的詳細(xì)看法另外單獨(dú)討論)。當(dāng)然,在方法實(shí)現(xiàn)中,您可以檢查相同的樹并執(zhí)行任何想執(zhí)行的操作。***的Linq CTP包含一個非常酷的可視化工具,在運(yùn)行時到達(dá)您的方法主體時可以用來查看表達(dá)式樹中的情況。到現(xiàn)在為止,您應(yīng)該明白了我正在實(shí)現(xiàn)一個強(qiáng)類型反射:我接收一個表達(dá)式樹,并在其中搜索方法調(diào)用節(jié)點(diǎn)(或者,對于屬性和字段來說是成員訪問)。下面是Method<>方法的實(shí)現(xiàn):

publicstaticMethodInfo Method<TDeclaringType>(  Expression<Operation> method)  {  return GetMethodInfo(method);  }     privatestaticMethodInfo GetMethodInfo(Expression method)  {  LambdaExpression lambda = method asLambdaExpression;  if (lambda == null)  thrownewArgumentNullException("method");   MethodCallExpression methodExpr = null;     // 我們的Operation<T>返回一個對象,故首先可以聲名一  // 個類型轉(zhuǎn)換(如果方法無返回對象)或直接方法調(diào)用。  if (lambda.Body.NodeType == ExpressionType.Cast)  {  // 類型轉(zhuǎn)換是一個一元操作,而操作數(shù)是一個方法調(diào)用表達(dá)式。  methodExpr = ((UnaryExpression)lambda.Body).  Operand asMethodCallExpression;  }  elseif (lambda.Body.NodeType == ExpressionType.MethodCall ||  lambda.Body.NodeType == ExpressionType.MethodCallVirtual)  {  methodExpr = lambda.Body asMethodCallExpression;  }  if (methodExpr == null)  thrownewArgumentException("method");  return methodExpr.Method;  }

我創(chuàng)建的就是Operation委托類型。不能使用Linq Func<T>(以及T、Arg0……),因為它們返回的是布爾值。我需要更靈活的對象,簡單來說就是返回對象的對象,以及接收一些固定參數(shù)類型(例如Func<T>)的委托“重載”。因此我得到如下內(nèi)容:

publicdelegateobjectOperation();  publicdelegateobjectOperation<T>(T declaringType);  publicdelegateobjectOperation(T declaringType, A0 arg0);  ...

注意,API的用戶從來都不會知道這些委托類型的對象的存在,就像查詢操作符的用戶從不知道Func<T>的存在一樣。我希望將來這些委托能夠消失,而代之以更好的東西(可能是publicdelegateobject Operation < params T> ;))。此外,注意我是如何將新的參數(shù)類型的參數(shù)添加到T“后面”的,T是重載的通用轉(zhuǎn)換,與Linq Func<T>中的功能正好相反。

以上是“Linq中如何實(shí)現(xiàn)Linq Func<T>”這篇文章的所有內(nèi)容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內(nèi)容對大家有所幫助,如果還想學(xué)習(xí)更多知識,歡迎關(guān)注億速云行業(yè)資訊頻道!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報,并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI