栏目分类:
子分类:
返回
名师互学网用户登录
快速导航关闭
当前搜索
当前分类
子分类
实用工具
热门搜索
名师互学网 > IT > 软件开发 > 后端开发 > C/C++/C# > C#教程

详解c# 泛型类的功能

C#教程 更新时间: 发布时间: IT归档 最新发布 模块sitemap 名妆网 法律咨询 聚返吧 英语巴士网 伯小乐 网商动力

详解c# 泛型类的功能

  在泛型类中,由于不知道泛型参数T是什么类型,可能是引用类型,也可能是值类型,因此不能将null等赋予泛型类型。如何对泛型对象赋初值、如何保证泛型的正确性等,以使用泛型文档管理器为例:

  文档管理器用于从队列中读写文档。首先创建一个泛型管理器Adddocument()方法添加一个文档到队列中,IsdocumentAvailabe只读属性指示队列中是否还有文档。

public class documentManager
{
  private readonly Queue documentQueue = new Queue();

  public void Adddocument(T doc)
  {
    lock (this)
    {
      documentQueue.Enqueue(doc);
    }
  }

  public bool IsdocumentAvailable
  {
    get { return documentQueue.Count > 0; }
  }
}

1、默认值

  给documentManager类添加一个Getdocument()方法,该方法以返回队列中的一个文档。如果队列中存在文档,则返回一个文档;如果队列中已没有文档,则返回默认值。但是,对于泛型T,不能将null赋予T的对象,因为无法确定它是引用类型还是值类型。在C#中,为我们提供了一个default关键字,泛型T的对象赋予默认值,如:引用类型为null、值类型int等为0……

public T Getdocument()
{
  T doc = default(T);
  lock (this)
  {
    if (documentQueue.Count > 0)
    {
      doc = documentQueue.Dequeue();
    }
  }
  return doc;
}

2、约束

  如果泛型类需要调用泛型类型中的方法,那么必须对泛型添加约束。否则,不能确保声明的泛型类型实现了对应的类型,具有相关方法。创建文档类document,其实现了接口Idocument:

public interface Idocument
{
  string Title { get; set; }
  string Content { get; set; }
}

public class document : Idocument
{
  public document()
  {
  }

  public document(string title, string content)
  {
    this.Title = title;
    this.Content = content;
  }

  public string Title { get; set; }
  public string Content { get; set; }
}

  给泛型文档管理器documentManager添加方法DisplayAlldocuments(),使得队列中所有文档的标题能展示出。在展示文档标题前,将类型T强制转换为IDocumnet接口,以显示标题:

public void DisplayAlldocuments()
{
  foreach (T doc in documentQueue)
  {
    Console.WriteLine((doc as Idocument).Title);//强制转换
  }
}

  但是,如果类型T没有实现接口Idocument,在对类型进行强制转换时就会出现一个异常。如果对方法添加rty……catch处理,将非常损耗性能。同样的,即使类型实现了接口Idocument,在进行转换时也会出现性能的损耗。

  那么,如果能对泛型Tdocument进行约束,使得泛型类型必须实现接口Idocument,则不会出现对类型进行强制转换时的异常。甚至不需要强制转换,性能也将得到优化。因此,前面的泛型文档管理器改写为(前面的T,改写为Tdocument,以此暗示是文档类型):

public class documentManager
  where Tdocument : Idocument
{
   //……   
}

  对于实现了约束的泛型文档管理器,可以处理任何实现了Idocument接口的类。其DisplayAlldocuments()方法改写为:

public void DisplayAlldocuments()
{
  foreach (Tdocument doc in documentQueue)
  {
    Console.WriteLine(doc.Title);
  }
}

  在其他地方调用时,可以用document类型实例化泛型类型documentManager。因为document实现了接口Idocument:

static void Main()
{
  var dm = new documentManager();
  dm.Adddocument(new document("Title A", "Sample A"));
  dm.Adddocument(new document("Title B", "Sample B"));

  dm.DisplayAlldocuments();

  if (dm.IsdocumentAvailable)
  {
    document d = dm.Getdocument();
    Console.WriteLine(d.Content);
  }
}

  泛型类型支持的几种约束:struct(结构约束,类型T必须是值类型)、class(类约束,类型T必须是引用类型)、IFoo(类型T必须实现接口IFoo)、new()(构造函数约束,类型T必须有一个无参构造函数)、TOther(类型T派生自TOther,也称“裸类型约束”)。

  泛型约束中:

  • 只能为无参构造函数定义构造约束,不能为有任何参数的构造函数定义构造函数约束。
  • 泛型可以有多个约束。如:public class documentManager where Tdocument : Idocument,new()。
  • where不能定义必须由泛型类型实现的运算符

3、继承

  泛型类也可以实现继承,如Queue里,继承实现了接口IEnumerable接口。泛型类型可以实现泛型接口,也可以派生自一个类。泛型类型可以派生自泛型基类:

class base
{
  //...............
}

class Derived:base
{
  //...............
}

  派生类可以是泛型类,也可以是非泛型类型:

abstract class Calc
{
  public abstract T Add(T x, T y);
  public abstract T Sub(T x, T y);
}

class IncCalc: Calc
{
  public override int Add(int x, int y)
  {
    return x + y;
  }
  public override int Sub(int x, int y)
  {
    return x - y;
  }
}
class DoubleCalc : Calc
{
  public override double Add(double x, double y)
  {
    return x + y;
  }
  public override double Sub(double x, double y)
  {
    return x - y;
  }
}

4、静态成员

  泛型类的静态成员只能在一个实例中共享:

class StaticDemo
{
  public static string Type;
}

static void Main()
{
  StaticDemo.Type = "int类型";
  StaticDemo.Type = "Object类型";
  Console.WriteLine(StaticDemo.Type);//输出:int类型
}

  实际上,每当用一个类型去代替泛型中的T时,都是在创造一个实例类型。因此,泛型类型中的静态字段,会在不同的类型替代泛型T的实例中重新生成。这样设计也有好处,可以为程序提供一个“泛型缓存”的概念,使用泛型的静态成员,使它存放在缓存中,方便调用。

以上就是详解c# 泛型类的功能的详细内容,更多关于c# 泛型类的资料请关注考高分网其它相关文章!

转载请注明:文章转载自 www.mshxw.com
我们一直用心在做
关于我们 文章归档 网站地图 联系我们

版权所有 (c)2021-2022 MSHXW.COM

ICP备案号:晋ICP备2021003244-6号