C#泛型語(yǔ)法詳解
一、為什么要有泛型?
我們?cè)趯?xiě)一些方法時(shí)可能會(huì)方法名相同,參數(shù)類(lèi)型不同的方法,這種叫做重載。如果只是因?yàn)閰?shù)類(lèi)型不同里面做的業(yè)務(wù)邏輯都是相同的,那可能就是復(fù)制粘貼方法,改變參數(shù)類(lèi)型,例如一些排序算法,int、float、double等類(lèi)型的排序,參數(shù)數(shù)組存的數(shù)據(jù)類(lèi)型不一樣,還有像根據(jù)索引找到List集合中的對(duì)象??赡苓@個(gè)對(duì)象是Person、Dog等對(duì)象,這樣方法改變的只是參數(shù)類(lèi)型,那就是能不能寫(xiě)一個(gè)方法,傳遞不同的參數(shù)類(lèi)型呢?于是乎有了泛型。
二、什么是泛型?
泛型通過(guò)參數(shù)化類(lèi)型來(lái)實(shí)現(xiàn)在同一份代碼上操作多種數(shù)據(jù)類(lèi)型。例如使用泛型的類(lèi)型參數(shù)T,定義一個(gè)類(lèi)Stack<T>,可以用Stack<int>、Stack<string>或Stack<Person>實(shí)例化它,從而使類(lèi)Stack可以處理int、string、Person類(lèi)型數(shù)據(jù)。這樣可以避免運(yùn)行時(shí)類(lèi)型轉(zhuǎn)換或封箱操作的代價(jià)和風(fēng)險(xiǎn),類(lèi)似C++的模板。泛型提醒的是將具體的東西模糊化,這與后面的反射正好相反。
三、泛型demo
1.泛型類(lèi)
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace Generic
{
public class Stack<T>
{
private T[] s;
int pos;
public Stack(int size)
{
s = new T[size];
pos = 0;
}
public void Push(T val)
{
s[pos] = val;
pos++;
}
public T Pop()
{
pos--;
return s[pos];
}
public void display()
{
Console.WriteLine("Stack Push:");
foreach (T i in s)
{
Console.WriteLine(i);
}
}
}
}using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace Generic
{
class Program
{
static void Main(string[] args)
{
Stack<int> s1 = new Stack<int>(2);
s1.Push(1);
s1.Push(2);
s1.display();
Console.WriteLine("Stack Pop:");
Console.WriteLine(s1.Pop());
Console.WriteLine(s1.Pop());
Stack<string> s2 = new Stack<string>(2);
s2.Push(@"One");
s2.Push(@"Two");
s2.display();
Console.WriteLine("Stack Pop:");
Console.WriteLine(s2.Pop());
Console.WriteLine(s2.Pop());
Console.ReadLine();
}
}
}上面定義了一個(gè)泛型類(lèi),主要是維護(hù)一個(gè)棧,棧里存放T類(lèi)型的數(shù)據(jù),在demo中可以定義int、string類(lèi)型的棧,這樣就很方便,使用一套代碼可以維護(hù)多種數(shù)據(jù)類(lèi)型。如果沒(méi)有這個(gè)可能還要維護(hù)double、float等代碼。
2.泛型方法
上面是泛型類(lèi),主要是在類(lèi)層面進(jìn)行參數(shù)化,我們還可以在更小的層面,在函數(shù)上進(jìn)行泛型化。
我們可以在上面Mina類(lèi)中定義一個(gè)靜態(tài)的泛型方法,用來(lái)獲取找數(shù)值在數(shù)組中的位置。
public static int Find<T>(T[] valus, T val)
{
for (int i = 0; i < valus.Length; i++)
{
if (valus[i].Equals(val))
{
return i;
}
}
return -1;
}我們可以用上面的方法來(lái)查找int數(shù)組、float數(shù)組
int val = 4;
int pos = Find<int>(new int[] {1,2,3,4,5 },val);
Console.WriteLine(string.Format("int Pos:{0}",pos));
float val1 = 4;
pos = Find<float>(new float[] { 1, 2, 3, 4, 5 }, val1);
Console.WriteLine(string.Format("float Pos:{0}", pos));
Console.ReadLine();下面是兩個(gè)demo的輸出

四、約束
約束是指對(duì)泛型類(lèi)型參數(shù)施加限制,用于限制可以傳遞到該類(lèi)型參數(shù)的類(lèi)型種類(lèi)。如果使用某個(gè)約束不允許的類(lèi)型來(lái)實(shí)例化,則會(huì)產(chǎn)生編譯時(shí)錯(cuò)誤。約束使用where關(guān)鍵字指定。
約束有4種類(lèi)型:
- 1.基類(lèi)約束
指定編譯器泛型類(lèi)型參數(shù)必須派生自特定基類(lèi)
修飾符 class 類(lèi)名<類(lèi)型參數(shù)列表> where 類(lèi)型參數(shù):基類(lèi)名
{ 類(lèi)體}
- 2.接口約束
指定編譯器泛型類(lèi)型參數(shù)必須派生自特定接口
修飾符 class 類(lèi)名<類(lèi)型參數(shù)列表> where 類(lèi)型參數(shù):接口名
{ 類(lèi)體}
- 3.默認(rèn)構(gòu)造函數(shù)約束
指示編譯器泛型類(lèi)型參數(shù)公開(kāi)了默認(rèn)的公共構(gòu)造函數(shù)(不帶任何參數(shù)的公共構(gòu)造函數(shù))
修飾符 class 類(lèi)名<類(lèi)型參數(shù)列表> where 類(lèi)型參數(shù):new ()
{ 類(lèi)體}
- 4.引用/值類(lèi)型約束
指示編譯器泛型類(lèi)型參數(shù)必須是引用類(lèi)型或值類(lèi)型
修飾符 class 類(lèi)名<類(lèi)型參數(shù)列表> where 類(lèi)型參數(shù):struct(或class)
{ 類(lèi)體}
可以對(duì)同一類(lèi)型參數(shù)使用多個(gè)約束,并且約束自身可以也可以是泛型類(lèi)型,多個(gè)約束之間用逗號(hào)隔開(kāi)。
五、泛型委托
泛型委托主要是想講一下Action<T>和Func<TResult>兩個(gè)委托,因?yàn)檫@兩個(gè)在Linq中是經(jīng)常見(jiàn)到的。
- Action<T>只能委托必須是無(wú)返回值的方法
- Fun<TResult>只是委托必須有返回值的方法
不管是不是泛型委托,只要是委托委托那能用Lamdba表達(dá)式,因?yàn)椴还躄amdba表達(dá)式還是匿名函數(shù)其實(shí)都是將函數(shù)變量化。
下面簡(jiǎn)單的來(lái)做的demo說(shuō)下兩個(gè)的用法,這個(gè)會(huì)了基本linq會(huì)了一半了。
Action<string> action = s => {
Console.WriteLine(s);
};
action("cuiyanwei");
Func<int, int, int> func = (int a, int b)=>{
return a + b;
};
int result=func(1, 2);
Console.WriteLine("sum:{0}",result);
Console.ReadLine();上面其實(shí)都是將函數(shù)做為變量,這也是委托的思想。action是實(shí)例化了一個(gè)只有一個(gè)字符串參數(shù)沒(méi)有返回值得函數(shù)變量。func是實(shí)例化了一個(gè)有兩個(gè)int類(lèi)型的參數(shù)返回值為int的函數(shù)變量。下面來(lái)看下輸出結(jié)果:

我們可以看到通過(guò)Lamdba表達(dá)式和泛型的結(jié)合,算是又方便了開(kāi)發(fā)者們,更加方便實(shí)用。
到此這篇關(guān)于C#泛型語(yǔ)法的文章就介紹到這了。希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
C#使用HttpWebRequest與HttpWebResponse模擬用戶(hù)登錄
這篇文章主要為大家詳細(xì)介紹了C#使用HttpWebRequest與HttpWebResponse模擬用戶(hù)登錄,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-04-04
unity實(shí)現(xiàn)多點(diǎn)觸控代碼
這篇文章主要介紹了unity實(shí)現(xiàn)多點(diǎn)觸控代碼,我最近在學(xué)習(xí)Unity游戲引擎。先從Unity平面開(kāi)始,本章介紹Unity 平面上的多點(diǎn)觸摸。有需要的小伙伴參考下。2015-03-03
解析c#在未出現(xiàn)異常情況下查看當(dāng)前調(diào)用堆棧的解決方法
本篇文章是對(duì)c#在未出現(xiàn)異常情況下查看當(dāng)前調(diào)用堆棧的解決方法進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-05-05
C#多線(xiàn)程學(xué)習(xí)之(六)互斥對(duì)象用法實(shí)例
這篇文章主要介紹了C#多線(xiàn)程學(xué)習(xí)之互斥對(duì)象用法,實(shí)例分析了C#中互斥對(duì)象的使用技巧,非常具有實(shí)用價(jià)值,需要的朋友可以參考下2015-04-04

