.NET基础之自定义泛型分析_实用技巧

本文实例分析了.NET基础之自定义泛型。分享给大家供大家参考。具体分析如下:

在.NET中泛型使用非常频繁,在控制台应用程序中,默认的引入了System.Collection.Generics名称空间,其中就提供了我们经常使用的泛型:List<T>和Dictionary<T>,相信用过它们的都知道它们的强大。还有一种我们经常使用的简单的泛型:System.Nullable<T>,即可空类型。我们可以:
 
System.Nullable<int> nullableInt;
声明一个可空的int类型,由于C#语法对这个做了简化通常我们都不这样写,而是这样写:
 
int? nullableInt
下面重点介绍一下如何自定义泛型。
 
定义泛型类
 
创建泛型类是需要在类定义中用尖括号语法:

复制代码 代码如下:

class MyGenericClass<T>
{
    ...
}

T可以是任意的标示符,只要遵守命名规则即可。

可以把类型用在类成员的返回类型,方法参数类型等,例如:

复制代码 代码如下:

class MyGenericClass<T1, T2, T3>
{
    private T1 t1Object;
 
    public MyGenericClass(T1 item)
    {
        t1Object = item;
    }
 
    public T1 T1Object
    {
        get
        {
            return t1Object;
        }
    }
}

注意如果不能假定提供了什么类型。下面的代码不能执行:

复制代码 代码如下:

class MyGenericClass<T1, T2, T3>
{
    private T1 t1Object;
 
    public MyGenericClass()
    {
        t1Object = new T1();
    }
}

因为我们不知道T1是否有公有的默认构造函数。

default关键字
 
如果我们定义了一个泛型的字段,我们想在构造函数中初始化它,但是我们不知道它的引用类型还是值类型,那么default就派上用处了:

复制代码 代码如下:

public MyGenericClass()
{
    t1Object = default(T1);
}

如果是值类型就赋值0,引用类型就赋值null。

约束类型
 
在定义泛型的时候我们可以对类型进行约束,通过where关键字实现:

复制代码 代码如下:

class MyGenericClass<T1> where T : constraint1,constraint
{
    ...
}

constraint定义了约束,多个约束用逗号隔开,如果有多个类型:

复制代码 代码如下:

class MyGenericClass<T1, T2> where T1 : constraint1 where T2 : constraint
{
    ...
}

下面给出一些可用的约束
 
                                        约束                                                                  说明
 
                                where T:struct                                     使用结构约束,类型T必须是值类型
 
                                where T:calss                                       类约束指定,类型T必须是引用类型
 
                                where T:interface                                  指定类型T必须实现是接口或者实现了接口
 
                                where T:base-class                               指定类型T必须是基类或者派生于基类
 
                               where T:new()                                       指定类型T必须有一个默认构造函数
 
 
 
下面结合以上知识给个实例:(PS不要看到代码多 其实很简单的 耐心看下去)
 
先定义四个类Animal、Cow 、Chicken和SuperCow

复制代码 代码如下:

#region Animal 虚基类 有一个name属性 Feed方法和一个虚方法MakeANoise
//虚基类 有一个name属性 Feed方法和一个虚方法MakeANoise
public abstract class Animal
{
        protected string name;
 
        public string Name
        {
            get
            {
                return name;
            }
            set
            {
                name = value;
            }
        }
 
        public Animal()
        {
            name = "The animal with no name";
        }
 
        public Animal(string newName)
        {
            name = newName;
        }
 
        public void Feed()
        {
            Console.WriteLine("{0} has been fed.", name);
        }
 
        public abstract void MakeANoise();
}
#endregion

//Cow Animal的子类,实现虚方法
public class Cow:Animal
{
        public Cow(string name) :
            base(name)
        {
        }
        public override void MakeANoise()
        {
            Console.WriteLine("{0} says 'moo!'", name);
        }
}

//Chicken类,Animal子类
public class Chicken:Animal
{
        public Chicken(string name)
            : base(name)
        { }
        public override void MakeANoise()
        {
            Console.WriteLine("{0} says 'cluck'", name);
        }
}

//Cow的子类,有一个自己的方法Fly
class SuperCow : Cow
{
        public SuperCow(string name) : base(name)
        {
        }
 
        public void Fly()
        {
            Console.WriteLine("{0} is flying!", name);
        }
 
        public override void MakeANoise()
        {
            Console.WriteLine("{0} says 'I am supercow!'", name);
        }
}

类准备好了之后,我们可以开始定义我们的泛型了:

复制代码 代码如下:

//继承了迭代器接口,这样方便使用Foreach 约束它的类型为Animal及其子类
public class Farm<T>:IEnumerable<T> where T : Animal
{
        private List<T> animals = new List<T>();
 
        public List<T> Animals
        {
            get
            {
                return animals;   
            }
        }
        //迭代器
        public IEnumerator<T> GetEnumerator()
        {
            return animals.GetEnumerator();
        }
 
        IEnumerator IEnumerable.GetEnumerator()
        {
            return animals.GetEnumerator();
        }
 
        //执行所有animal的MakeANoise()
        public void MakeNoises()
        {
            foreach (T animal in animals)
            {
                animal.MakeANoise();
            }
        }
        //执行所有animal的Feed()
        public void FeedTheAnimals()
        {
            foreach (T animal in animals)
            {
                animal.Feed();
            }
        }
        //获得animals中的cow
        public Farm<Cow> GetCows()
        {
            Farm<Cow> cowFarm = new Farm<Cow>();
            foreach (T animal in animals)
            {
                if (animal is Cow)
                {
                    cowFarm.Animals.Add(animal as Cow);
                }
            }
            return cowFarm;
        }
}

泛型定义好了,我们用写代码来调用它:

复制代码 代码如下:

class Program
{
        static void Main(string[] args)
        {
            Farm<Animal> farm = new Farm<Animal>();
            farm.Animals.Add(new Cow("Jack"));
            farm.Animals.Add(new Chicken("Vera"));
            farm.Animals.Add(new Chicken("Sally"));
            farm.Animals.Add(new SuperCow("Kevin"));
            farm.MakeNoises();
 
            Farm<Cow> dairyFarm = farm.GetCows();
            dairyFarm.FeedTheAnimals();
 
            foreach (Cow cow in dairyFarm)
            {
                if (cow is SuperCow)
                {
                    (cow as SuperCow).Fly();
                }
            }
            Console.ReadKey();
        }
}

希望本文所述对大家的.net程序设计有所帮助。

时间: 2024-10-30 09:15:16

.NET基础之自定义泛型分析_实用技巧的相关文章

.Net与JS时间日期格式的转换问题对比分析_实用技巧

本文实例分析了.Net与JS时间日期格式的转换问题.分享给大家供大家参考,具体如下: Js中的1415349957524整数 ,其实代表的是1970.1.1 00:00:00至现今某个时间点的时间间隔毫秒数.而在.Net中,我们可以用Ticks属性得到0001.1.1 00:00:00至现今某个时间点的时间间隔毫秒数. 具体代码实现如下所示: JS: //获取1970.1.1 00:00:00至现在的毫秒数 var milDate = +(new Date); .Net: //获取1970.1.

.NET开发基础:从简单的例子理解泛型 分享_实用技巧

从简单的例子理解泛型话说有家影视公司选拔偶像派男主角,导演说了,男演员,身高是王道.于是有下面代码:  复制代码 代码如下: //男演员实体类public class Boy{    //姓名    private string mName;    //身高    private int mHeight;    public string Name {        get { return this.mName; }    }    public int Height {        get

list泛型自定义排序示例_实用技巧

复制代码 代码如下: static void Main(string[] args){     Employee employee = new Employee();    //设置初始值    List<Employee> employeeList = new List<Employee>();    employeeList.Add(new Employee() { EmpId = "001", EmpName = "Tony" }); 

ASP.NET MVC小结之基础篇(二)_实用技巧

整理除了这个笔记,共享一下子,基本MVC的所有东西都介绍了,但是都是很基础的东西.本来打算一篇发表完的,但是发现东西有点多,所以分成了两篇文章,这是最后一篇了! 1.ASP.NET MVC请求过程 1 2.Controller (1) 控制器在ASP.NET MVC中扮演着处理客户端请求的角色 1)必须实现System.Web.Mvc.IController接口 ->通常直接继承System.Web.MVC.Controller类 2)必须要以Controller结尾 3)通过不同的Action

ASP.NET中保护自定义的服务器控件_实用技巧

自定义服务器控件是扩展 ASP.NET Web 服务器控件的功能的一种方式.下文提供了针对自定义服务器控件的用户和开发人员的基本安全准则.有关创建自定义服务器控件的更多信息,请参见开发自定义 ASP.NET 服务器控件. IDE(如 Microsoft Visual Studio 2005)简化了自定义控件的使用及开发.但是,无论使用哪一 IDE,下面列出的安全准则均适用. 有关 ASP.NET Web 应用程序安全性的常规信息,请参见 ASP.NET Web 应用程序安全性. 针对自定义服务器

在.NET2.0中使用自定义事务操作_实用技巧

.net 2.0 framework 中新增了 System.Transactions 命名空间,其中提供的一系列接口和类使得在.net 2.0 中使用事务比起从前要方便了许多.有关在 .net 2.0 下操作数据库事务的文章已经有了很多,这里只提一下如何设计自定义事务操作. 一.事务使用基础 先看一段使用事务的代码: 1using (TransactionScope ts= new TransactionScope())2{3 //自定义操作4 ts.Complete();5} 这里使用 us

ASP.NET MVC小结之基础篇(一)_实用技巧

前言:前几天要准备一个演讲,所以准备了MVC的一些基本的东西,以前也使用过MVC,但是只是使用,而不是去了解,所以趁着这个机会好好的把别人的MVC视频看了一下(是一个微软的MVP会员发布的视频,相信有些人都看过),整理除了这个笔记,共享一下子,基本MVC的所有东西都介绍了,但是都是很基础的东西.本来打算一篇发表完的,但是发现东西有点多,所以分成了两篇文章! 什么是ASP.NET MVC (1) ASP.NET MVC是微软官方提供的MVC模式编写ASP.NET Web应用程序的一个框架 (2)M

ASP.NET服务器控件的生命周期分析_实用技巧

本文实例分析了ASP.NET服务器控件的生命周期.分享给大家供大家参考.具体如下: (1)初始化----在此阶段中,主要完成两项工作:一.初始化在传入Web请求生命周期内所需的设置:二.跟踪视图状态.首先,页面框架通过默认方式引发Init事件,并调用OnInit()方法,控件开发人员可以重写该方法为控件提供初始化逻辑.此后,页面框架将调用TrackViewState方法来跟踪视图状态.需要注意的是:多数情况下,Control基类提供的TrackViewState方法实现已经足够了.只有在控件定义

.net Cookies安全性实践分析_实用技巧

 跨站脚本攻击一直是Web上常见的手段之一,攻击一般是劫持用户会话,拿到私有的钥匙.如何劫持这个会话呢? 一.Sniffer(这里采用的软件是Sniffer,java版的那个) Sniffer监听会话是不可护的,就和KOF里拉尔夫的宇宙幻影一样,只要打中,一定是致命的.        上图,就是一组侦听数据,当然这里侦听的是我本机的数据.实际上是可以侦听局域网,甚至广域网上的数据.得到一组16进制的值.可以用以下代码获得其值: 复制代码 代码如下: string bytestr = ""