ASP.NET泛型二之泛型的使用方法
".NET泛型"系列:
ASP.NET泛型一之泛型簡(jiǎn)介與基本語(yǔ)法
ASP.NET泛型三之使用協(xié)變和逆變實(shí)現(xiàn)類型轉(zhuǎn)換
ASP.NET泛型四之使用Lazy<T>實(shí)現(xiàn)延遲加載
在" ASP.NET泛型一之泛型簡(jiǎn)介與基本語(yǔ)法"中,了解了泛型的基本概念,本篇偏重于泛型的使用。主要包括:
泛型方法重載需要注意的問(wèn)題
public class MyArray<T>
{
public T myData;
public MyArray()
{
myData = default(T);
}
public void ShowInfo()
{
Console.WriteLine(myData.ToString());
}
public void ShowInfo(string str)
{
Console.WriteLine(str);
}
public void ShowInfo<T>(T data)
{
Console.WriteLine(data.ToString());
}
}以上,說(shuō)明:泛型方法可以作為方法的重載。
可以這樣調(diào)用。
MyArray<Student> myArray = new MyArray<Student>();
myArray.ShowInfo<CollegeStudent>(new CollegeStudent());
myArray.ShowInfo<string>("HelloWorld");但還有一種情況是:兩個(gè)語(yǔ)義不明的重載方法,在編譯的時(shí)候是通過(guò)的,但在調(diào)用的時(shí)候就不會(huì)通過(guò)。比如以下在編譯時(shí)沒(méi)問(wèn)題:
public class MyArray<T>
{
public void ShowInfo<TA, TB>(TA a, TB b){};
public void ShowInfo<TB, TA>(TA a, TB b){};
}如果這樣調(diào)用,就有問(wèn)題:
MyArray<Student> myArray = new MyArray<Student>(); myArray.showInfo<Student, Student>(new Student(), new Student());
所以,對(duì)于泛型重載方法,需要注意語(yǔ)義不明的情況。
泛型的類型推斷
編譯器可以根據(jù)方法參數(shù)的類型來(lái)推斷使用哪個(gè)重載方法,優(yōu)先調(diào)用一般重載方法,然后再調(diào)用泛型重載方法。
myArray.ShowInfo("hello"); //會(huì)調(diào)用 ShowInfo(string str)重載方法
myArray.ShowInfo(new CollegeStudent());//會(huì)調(diào)用ShowInfo<T>(T data)重載方法。泛型方法也可以有約束
我們知道泛型類可以有約束,泛型方法也一樣。
public void ShowInfo<T>(T data) where TData : Student
{
Console.WriteLine(data.ToString());
}泛型接口
.NET集合類就提供了多個(gè)泛型接口,比如:IList<T>, ICollection<T>, IComparable<>, IComparer<T>, IEnumerable<T>, IEnumerator<T>, IDictionary<TKey,TValue>,等等。
自定義類的時(shí)候,有時(shí)候需要讓自定義類實(shí)現(xiàn)一個(gè)指定了具體類型的泛型接口:
class MyClass<T> : IComparable<Int32>, IComparable<String>
泛型委托
public class Generic Delegate
{
//聲明泛型委托
public delegate string MyGenericDelegate<T>(T t);
public static string GetPoint(Point p)
{
return stirng.Format("地址是{0},{1}", p.X, p.Y);
}
public static string GetMsg(string str)
{
return str;
}
}
public static void Main()
{
MyGenericDelegate<string> myStrDel = new MyGenericDelegate<string>(GetMsg);
Console.WriteLine(myStrDel("hello"));
MyGenericDelegate<Point> myPointDel = new MyGenericDelegate<Point>(GetPoint);
Console.WriteLine(myPointDel(new Point(100, 200)));
}使用EventHandler<TEventArgs>事件泛型
它的完整定義是:
public delegate void EventHandler<TEventArgs>(object sender, TEventArgs e) where TEventArgs: EventArgs
假設(shè)有一個(gè)MessageReceiver類,當(dāng)建立連接時(shí)觸發(fā)OnConnected事件,在接收到信息是觸發(fā)OnMessageReceived事件。
在創(chuàng)建MessageReceiver類之前,我們先要自定義一個(gè)派生于EventArgs,且和MessageReceiver相關(guān)的類。
public sealed class MessageReceivedEventArgs : EventArgs
{
public string Message {get;set;}
public MessageReceivedEventArgs(string msg)
{
this.Message = msg;
}
}MessageReceiver類主要包含2個(gè)事件,一個(gè)是OnConnected事件,另一個(gè)是OnMessageReceived事件。
public class MessageReceiver
{
public event EventHandler OnConnected;
public event EventHandler<MessageReceivedEventArgs> OnMessageReceived;
...
public void DoSth()
{
if(OnMessageReceived != null)
{
OnMessageReceived(this, new MessageReceivedEventArgs(msg));
}
}
}以上,通過(guò)if(OnMessageReceived != null)這個(gè)判斷,能保證:當(dāng)沒(méi)有訂閱者注冊(cè)事件的時(shí)候,這個(gè)事件不被觸發(fā)。但在多線程場(chǎng)景中,這樣做也不是最合理的:
假設(shè)線程A作為訂閱者注冊(cè)事件,正準(zhǔn)備觸發(fā)事件的時(shí)候,線程B也作為訂閱者剛好在此刻注銷了事件,即OnMessageReceived變成了null,這就牽累到線程A也無(wú)法觸發(fā)事件。
解決辦法是把事件變量賦值給一個(gè)局部變量:
public class MessageReceiver
{
public event EventHandler OnConnected;
public event EventHandler<MessageReceivedEventArgs> OnMessageReceived;
...
public void DoSth()
{
var handler = OnMessageReceived;
if(handler != null)
{
handler(this, new MessageReceivedEventArgs(msg));
}
}
}這樣,當(dāng)線程A作為訂閱者注冊(cè)并準(zhǔn)備觸發(fā)事件的時(shí)候,及時(shí)線程B在此刻注銷注冊(cè),讓OnMessageReceived為null,由于已經(jīng)把OnMessageReceived賦值給了局部變量handler,線程A依然能觸發(fā)事件。
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,謝謝大家對(duì)腳本之家的支持。如果你想了解更多相關(guān)內(nèi)容請(qǐng)查看下面相關(guān)鏈接
相關(guān)文章
asp.net ext treepanel 動(dòng)態(tài)加載XML的實(shí)現(xiàn)方法
當(dāng)你在asp.net下面 使用Ext TreePanel直接加載服務(wù)器上XML文件會(huì)出現(xiàn)樹(shù)不能顯示,樹(shù)據(jù)不能正確加載的問(wèn)題。2008-10-10
ASP.NET設(shè)計(jì)網(wǎng)絡(luò)硬盤之刪除文件夾實(shí)現(xiàn)代碼
對(duì)于不再需要的文件/文件夾用戶有權(quán)限做刪除處理。這一節(jié)就介紹如何實(shí)現(xiàn)這一功能2012-10-10
禁用aspx頁(yè)面的客戶端緩存(防止頁(yè)面被修改)
默認(rèn)情況下,IE打開(kāi)一個(gè)網(wǎng)頁(yè),會(huì)在本地進(jìn)行緩存,在某些時(shí)候也會(huì)帶來(lái)了弊端,比如修改信息的頁(yè)面等等因?yàn)閁RL并沒(méi)有改變,所以IE會(huì)讀取本地緩存,這種情況特別容易出現(xiàn)在彈出對(duì)話框或窗口進(jìn)行修改的方式感興趣的朋友可以了解下,或許對(duì)你有所幫助2013-02-02
datagrid和repeader控件中替換標(biāo)識(shí)值的方法
本節(jié)主要介紹了datagrid和repeader控件中替換標(biāo)識(shí)值的方法,需要的朋友可以參考下2014-08-08
ASP.NET MVC中設(shè)置跨域訪問(wèn)問(wèn)題
這篇文章主要介紹了ASP.NET MVC中設(shè)置跨域訪問(wèn)問(wèn)題,需要的朋友可以參考下2018-06-06
ASP.NET Core應(yīng)用錯(cuò)誤處理之三種呈現(xiàn)錯(cuò)誤頁(yè)面的方式
這篇文章主要給大家介紹了關(guān)于ASP.NET Core應(yīng)用錯(cuò)誤處理之三種呈現(xiàn)錯(cuò)誤頁(yè)面的方式的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-01-01
vs2010制作簡(jiǎn)單的asp.net網(wǎng)站
這篇文章主要介紹了vs2010制作簡(jiǎn)單的asp.net網(wǎng)站,只要十步哦,感興趣的小伙伴們可以參考一下2015-09-09
Asp.net內(nèi)置對(duì)象之Request對(duì)象(概述及應(yīng)用)
Request對(duì)象主要用于獲取來(lái)自客戶端的數(shù)據(jù),如用戶填入表單的數(shù)據(jù)、保存在客戶端的Cookie等,本文將圍繞Request對(duì)象,講解其的主要作用:讀取窗體變量、讀取查詢字符串變量、取得Web服務(wù)器端的系統(tǒng)信息。取得客戶端瀏覽器信息等等,感興趣的朋友可以了解下2013-02-02

