.net泛型通用函數的特殊問題的解決方法
自從2.0版本的net framework推出之后泛型(Generic)得到了廣泛好評。它不必像object類型一樣性能上因為“拆箱”或者“裝箱”得到損失,同時在編譯語法檢測階段就可以實時檢測出傳入或者傳出的類型是否符合特定條件。
但“金無赤足,人無完人”——在我們享受這些幸福編程的同時,泛型自身類型的不確定也帶來了一個顯著的問題——無法進行運算符重載。譬如現在我要寫一個函數(一個通用的選擇排序算法,使用泛型T),該怎么辦呢?如果你簡單使用這樣的代碼(C#如下):
//從小到大,改進型選擇排序算法
public static void Sort<T>(T[] array)
{
bool flag = false; //標記是否已經排序
for(int i=0;i<array.Length-1;++i)
{
flag = false; //每次假定都已經排序,無須再排序
for(int j=i+1;i<array.Length;++j)
{
if(array[i]>array[j])
{
int temp = array[i];
array[i]=array[j];
array[j]=templ
flag = true; //已經排序
}
}
if(!flag)
{
break;
}
}
}
編譯之后很快發(fā)現提示“運算符‘>'無法作用于T”一類的提示。
為什么呢?我們知道,凡是可以進行大于、小于比較的類型肯定都定義了運算符重載。一般類必須為此定義方可進行比較,不然大于號或者小于號(或者其它運算符)無法知道如何比較而發(fā)生錯誤。那么泛型因為事先都不知道什么類型?編譯器檢查器自然無法推斷你運行時動態(tài)傳入的這個類型一定保證是實現了運算符重載,嚴格語法檢查情況下就自然報錯。
怎么辦呢?強制規(guī)定泛型T必須實現比較器(強制T必須實現IComparable,或者類似接口)。
public static void Sort<T>(T[] array)where T:IComparable
{
bool flag = false; //標記是否已經排序
for(int i=0;i<array.Length-1;++i)
{
flag = false; //每次假定都已經排序,無須再排序
for(int j=i+1;i<array.Length;++j)
{
if(array[i].Compare(array[j])>0)
{
int temp = array[i];
array[i]=array[j];
array[j]=templ
flag = true; //已經排序
}
}
if(!flag)
{
break;
}
}
}
一旦對泛型進行約束,那么泛型必然是實現該接口的類,必然擁有此方法(Compare方法返回結果int類型,如果大于0表示前面一個數字大于后面一個)。
當然,微軟類庫中有一個Comparer靜態(tài)類,已經實現了此接口可以直接進行比較(http://msdn.microsoft.com/zh-cn/library/system.collections.comparer.comparer.aspx),因此我們也可以選擇直接使用這個靜態(tài)類中的Compare方法得到結果。
【例2】實現一個通用的“+”——即如果傳入的字符串,則自動按照字符串進行字符拼接;如果傳入的是其它基本類型(int,double等),則返回相加結果。
微軟沒有為“+”預定義接口,因此無法直接使用接口的方式來做(當然你自己強制定義一個,也可以如法炮制)。我們現在換一個方法——使用表達式樹(C#代碼如下):
public static T Add<T>(T a, T b)
{
Expression left = Expression.Constant(a);
Expression right = Expression.Constant(b);
Type t = typeof(T);
Expression value;
if (t == typeof(string))
{
value = Expression.Constant(a.ToString()+b.ToString());
}
else
{
value = Expression.Add(left, right);
}
Expression<Func<T>> addExp = Expression.Lambda<Func<T>>(value);
Func<T> addFunc = addExp.Compile();
return addFunc();
}
動態(tài)判斷T是string還是其它基本類型,然后調用不同的方法組合成為表達式樹,動態(tài)編譯成為一個Func表達式,返回結果即可。
相關文章
Asp.net中DataTable導出到Excel的方法介紹
本篇文章介紹了,Asp.net中DataTable導出到Excel的方法。需要的朋友參考下2013-05-05Windows Server 2012 R2 或 2016無法安裝.Net 3.5.1
這篇文章主要為大家詳細介紹了Windows Server 2012 R2 或 2016 無法安裝 .Net 3.5.1,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-02-02asp.net線程批量導入數據時通過ajax獲取執(zhí)行狀態(tài)
asp.net線程批量導入數據是大家日常工作中常遇到的一個要求,但批量添加時間一般較長,如果能返回執(zhí)行的狀態(tài)就好,那么下面這篇文章主要給大家介紹了asp.net線程批量導入數據時通過ajax獲取執(zhí)行狀態(tài)的方法,有需要的朋友可以參考下。2016-12-12asp.net動態(tài)產生checkbox(數據源為DB或內存集合)
動態(tài)產生一組checkbox(數據源為DB或內存集合)且post提交時后臺能及時獲取等等,打算使用repeater+input(checkbox)+input(hidden)來實現2013-10-10.NET性能調優(yōu)之一:ANTS Performance Profiler的使用介紹
本系列文章主要會介紹一些.NET性能調優(yōu)的工具、Web性能優(yōu)化的規(guī)則(如YSlow)及方法等等內容。成文前最不希望看到的就是園子里不間斷的“哪個語言好,哪個語言性能高”的爭論,不多說,真正的明白人都應該知道這樣的爭論有沒有意義,希望我們能從實際性能優(yōu)化的角度去討論問題2013-01-01使用Visual Studio創(chuàng)建ASP.NET Web API項目
這篇文章介紹了使用Visual Studio創(chuàng)建ASP.NET Web API項目的方法,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2022-03-03