当前位置: 编程技术>.net/c#/asp.net
c# 泛型类型参数与约束的深入分析
来源: 互联网 发布时间:2014-10-21
本文导语: 泛型类型参数简介在定义泛型类型和泛型方法时,常用到泛型类型参数,泛型类型参数是在实例化泛型时指定类型的占位符。泛型类型参数放在“”内。泛型类型参数命名建议:(1)当泛型类型参数为单个字母时,建议用T表...
泛型类型参数简介
在定义泛型类型和泛型方法时,常用到泛型类型参数,泛型类型参数是在实例化泛型时指定类型的占位符。泛型类型参数放在“”内。
泛型类型参数命名建议:
(1)当泛型类型参数为单个字母时,建议用T表示。
(2)当泛型类型参数用单词定义时,建议在单词前加T。
private void PromptName(T t) {}
private void PromptName(Tuser user){}
泛型类型参数约束
在定义泛型类时,可以对在实例化泛型类时用于类型参数的类型种类施加限制。如果实例化泛型类时使用某个约束所不允许的类型来实例化类,则会产生编译时错误。
泛型约束分类:
(1)类型参数约束为结构(struct)。
public class ShowObjectType where T : struct
{
public void ShowValue(T t)
{
Console.WriteLine(t.GetType());
}
}
class GenericConstraint
{
static void Main()
{
ShowObjectType showInt = new ShowObjectType();
showInt.ShowValue(5);
showInt.ShowValue(5);//从参数可以推导出类型参数类型,则可以省略类型参数类型
//因为约束为值类型,下面代码不能通过编译
ShowObjectType showString = new ShowObjectType();
showString.ShowValue("5");
Console.Read();
}
}
(2)类型参数约束为类(class)。
在应用 where T : class 约束时,避免对类型参数使用 == 和 != 运算符,因为这些运算符仅测试类型为引用类型,而不测试值相等性。
class GenericConstraint
{
static void Main()
{
List list = new List();
AddClass(list, "hello generic");
Console.Read();
}
private static void AddClass(List list, T t) where T : class
{
list.Add(t);
}
}
(3)类型参数约束为具体类。
约束为具体类时,可利用类型参数调用具体类的属性和方法。
class GenericConstraint
{
static void Main()
{
Person person = new Person { ID = 1, Name = "David" };
PromptName(person);
Console.Read();
}
//此约束T为Person对象或者继承Person对象
private static void PromptName(T t) where T : Person
{
//此处可使用Person的Name属性
if (t.Name == "David")
{
Console.WriteLine("Person name is David");
}
string name = t.GetName();
Console.WriteLine("Person name is {0}", name);
}
}
public class Person
{
private int id;
public int ID
{
get { return id; }
set { id = value; }
}
private string name;
public string Name
{
get { return name; }
set { name = value; }
}
public string GetName()
{
return Name;
}
}
(4)约束多个参数。
class Base { }
class Test
where U : struct
where T : Base, new() { }
(5)未绑定类型参数。
没有约束的类型参数,称为未绑定的类型参数。
class List{}
在定义泛型类型和泛型方法时,常用到泛型类型参数,泛型类型参数是在实例化泛型时指定类型的占位符。泛型类型参数放在“”内。
泛型类型参数命名建议:
(1)当泛型类型参数为单个字母时,建议用T表示。
(2)当泛型类型参数用单词定义时,建议在单词前加T。
代码如下:
private void PromptName(T t) {}
private void PromptName(Tuser user){}
泛型类型参数约束
在定义泛型类时,可以对在实例化泛型类时用于类型参数的类型种类施加限制。如果实例化泛型类时使用某个约束所不允许的类型来实例化类,则会产生编译时错误。
泛型约束分类:
约束
说明
T:结构
类型参数必须是值类型。 可以指定除 Nullable 以外的任何值类型。
T:类
类型参数必须是引用类型;这一点也适用于任何类、接口、委托或数组类型。
T:new()
类型参数必须具有无参数的公共构造函数。 当与其他约束一起使用时,new() 约束必须最后指定。
T:
类型参数必须是指定的基类或派生自指定的基类。
T:
类型参数必须是指定的接口或实现指定的接口。 可以指定多个接口约束。 约束接口也可以是泛型的。
T:U
为 T 提供的类型参数必须是为 U 提供的参数或派生自为 U 提供的参数。
(1)类型参数约束为结构(struct)。
代码如下:
public class ShowObjectType where T : struct
{
public void ShowValue(T t)
{
Console.WriteLine(t.GetType());
}
}
class GenericConstraint
{
static void Main()
{
ShowObjectType showInt = new ShowObjectType();
showInt.ShowValue(5);
showInt.ShowValue(5);//从参数可以推导出类型参数类型,则可以省略类型参数类型
//因为约束为值类型,下面代码不能通过编译
ShowObjectType showString = new ShowObjectType();
showString.ShowValue("5");
Console.Read();
}
}
(2)类型参数约束为类(class)。
在应用 where T : class 约束时,避免对类型参数使用 == 和 != 运算符,因为这些运算符仅测试类型为引用类型,而不测试值相等性。
代码如下:
class GenericConstraint
{
static void Main()
{
List list = new List();
AddClass(list, "hello generic");
Console.Read();
}
private static void AddClass(List list, T t) where T : class
{
list.Add(t);
}
}
(3)类型参数约束为具体类。
约束为具体类时,可利用类型参数调用具体类的属性和方法。
代码如下:
class GenericConstraint
{
static void Main()
{
Person person = new Person { ID = 1, Name = "David" };
PromptName(person);
Console.Read();
}
//此约束T为Person对象或者继承Person对象
private static void PromptName(T t) where T : Person
{
//此处可使用Person的Name属性
if (t.Name == "David")
{
Console.WriteLine("Person name is David");
}
string name = t.GetName();
Console.WriteLine("Person name is {0}", name);
}
}
public class Person
{
private int id;
public int ID
{
get { return id; }
set { id = value; }
}
private string name;
public string Name
{
get { return name; }
set { name = value; }
}
public string GetName()
{
return Name;
}
}
(4)约束多个参数。
代码如下:
class Base { }
class Test
where U : struct
where T : Base, new() { }
(5)未绑定类型参数。
没有约束的类型参数,称为未绑定的类型参数。
代码如下:
class List{}