[AaronYang]C#人爱学不学[4]

本文章不适合入门,只适合有一定基础的人看。我更相信知识细节见高低,我是从4.0开始学的,终于有时间系统的学习C#5.0,是5.0中的知识,会特殊标记下。但写的内容也可能含有其他版本framework的知识,也是为了方便自己更好的记忆C#知识。文章内容都是自己总结,无抄袭,如果你觉得文章档次太低,请大牛绕道 --Aaronyang的博客(www.ayjs.net)

1. 泛型-是C#的泛型

 1.1 性能方面比非泛型好点,比如拆箱装箱的问题。个人感觉代码可读性更好吧。还有就是 写代码可能可以写出很精彩的代码。命名用T开头,加有意义的单词,比如 Converter<TInput,TOut>,XX<TKey,TValue>

 1.2 题目:不百度,请自己至少列举5个 例如List<T>使用泛型的C#中的常用对象

 1.3 自己写个链式,并使用泛型知识的demo

       1.定义链式节点, 前一个节点,后一个节点,再加上自己这个节点就OK了 

   //定义链式节点, 前一个节点,后一个节点,再加上自己这个节点就OK了
    public class MyLinkedNode<T> { 

        public MyLinkedNode(T value){
            this.Value = value;
        }
        public T Value{get;private set;}

        /// <summary>
        /// 前一个节点对象
        /// </summary>
        public MyLinkedNode<T> Prev { get; internal set; }
        /// <summary>
        /// 后一个节点对象
        /// </summary>
        public MyLinkedNode<T> Next { get; internal set; }

    }

     2.接下来,在封装一个对节点的操作的操作类,正好复习 迭代器yield和理解IEnumerable<T>这个接口,时间有限,这里我只先实现AddLast和GetEnumerator

 //接下来,因为链式结构,只适合从尾部和首部增加元素,中间不方便增加元素。所以对节点的操作的封装类,一般 首部增加一个节点AddFirst(),尾部增加一个节点AddEnd()
    //移除一个节点RemoveFirst(),RemoveEnd(),这里我只先实现AddEnd和GetEnumerator
    public class MyLinkedList<T> : IEnumerable<T>
    {

        /// <summary>
        /// 默认 IEnumerable定义的,必须实现
        /// </summary>
        /// <returns></returns>
        public IEnumerator<T> GetEnumerator()
        {
            //使用yield,把MyLinkedNode<T>返回到IEnumerator中去
            MyLinkedNode<T> cur = First;
            while (cur!=null)    //如果下一个节点不为空
            {
                yield return cur.Value;
                cur = cur.Next;//把当前值设置成下一个节点的值
            }
        }

        System.Collections.IEnumerator System.Collections.IEnumerable.GetEnumerator()
        {
            return GetEnumerator();
        }

        /// <summary>
        /// 添加一个节点,默认都是从尾部增加的
        /// </summary>
        /// <param name="myLinkedNode">MyLinkedNode中类型的值</param>
        /// <returns></returns>
        public MyLinkedNode<T> AddEnd(T myLinkedNode)
        {
            MyLinkedNode<T> cur = new MyLinkedNode<T>(myLinkedNode);
            if (First == null)
            {  
                //如果该集合一个节点也没有,那么第一个和最后一个节点就等于当前节点
                First = End = cur;
            }
            else { 
             //因为尾部新节点增加,原来上次的尾部的节点就变成了该节点的前一个节点了,他自己变成了最后一个节点
                MyLinkedNode<T> prevEnd = End;
                //更新新的最后一个元素
                End.Prev = prevEnd;
                End.Next = cur;
                End = cur;
            }
            return cur;
        }

        //在链式结构里面都有 第一个节点和最后一个节点的特殊节点
        /// <summary>
        /// MyLinkedList集合中第一个元素
        /// </summary>
        public MyLinkedNode<T> First { get; private set; }
        /// <summary>
        /// MyLinkedList集合中最后一个元素
        /// </summary>
        public MyLinkedNode<T> End { get; private set; }

        /// <summary>
        /// 在开始的地方增加一个节点
        /// </summary>
        /// <param name="myLinkedNode">MyLinkedNode值</param>
        /// <returns></returns>
        public MyLinkedNode<T> AddFirst(T myLinkedNode)
        {
            throw new NotImplementedException();//自己实现
        }
        /// <summary>
        /// 移除一个节点
        /// </summary>
        /// <param name="myLinkedNode">MyLinkedNode对象</param>
        /// <returns></returns>
        public MyLinkedNode<T> RemoveEnd(T myLinkedNode)
        {
            throw new NotImplementedException();//自己实现
        }

    }

       3.使用这个数据结构的集合

 class Program
    {
        static void Main(string[] args)
        {
            MyLinkedList<int> m = new MyLinkedList<int>();
            m.AddEnd(1);
            m.AddEnd(10);
            m.AddEnd(100);
            foreach (var item in m)
            {
                Console.WriteLine(item+",");
            }
            Console.ReadLine();
        }
    }

效果

 整个过程来说,感觉还是挺有意义的,特别当泛型的概念融入其中,你的代码可能更精彩。

 1.4 一些数据结构的类型,只是加深印象: Queue<T>,Dictionary<TKey, TValue>, ILookup(TKey, TElement) 等

 1.5 泛型默认值初始化,举个例子      public T GetT(){ T t=default(T);  ...     }

 1.6 泛型约束与继承:

       public abstract class Class1<T> :TEnumerable<T>

       public class DBManager<TDb> where TDb:ICommonDb,new()

       Aaronyang拓展: where T:struct / class  / 接口  / new() 构造函数约束,指定类型T必须有一个默认构造函数 /  其他泛型,例如  where T1:T2

 *1.7 静态成员:AaronYang讲解:只要记得 T 不一样,里面的静态成员的值是不共享的,也不会受影响。   专业术语:泛型类的静态成员只能在类的一个实例中共享

      自己写了一个例子,一看就懂了

 class Program
    {
        static void Main(string[] args)
        {
            OwnStaticGeneric1<string>.obj = 4;
            OwnStaticGeneric1<int>.obj = 5;
            OwnStaticGeneric1<string>.obj = 6;
            OwnStaticGeneric1<int>.obj = 7;
            Console.WriteLine(OwnStaticGeneric1<string>.obj);//   =>6
            Console.WriteLine(OwnStaticGeneric1<int>.obj); // =>7


            OwnStaticGeneric2<string>.obj = "4";
            OwnStaticGeneric2<int>.obj = 5;
            OwnStaticGeneric2<string>.obj = "6";
            OwnStaticGeneric2<int>.obj = 7;
            Console.WriteLine(OwnStaticGeneric1<string>.obj);//   =>6
            Console.WriteLine(OwnStaticGeneric1<int>.obj); // =>7


            Console.ReadLine();
        }
    }

    public class OwnStaticGeneric1<T> {
        public static int obj;
    }

    public class OwnStaticGeneric2<T>
    {
        public static T obj;
    }

 1.8 高级知识: 泛型接口,感觉让你考架构师的样子

    1.8.1  拓展一下 可能会使用的 跟 ref和out差不多性质的  in(in修饰的参数,在方法体内的过程不会改写in的参数的值)关键字用法。如果不太懂,可以百度,也可以看我下面的例子,但提前,你最好懂out,ref的基础用法。

              有些人设计接口直接    IInterface<T1,T2>,其实也还有很奇妙的其他写法,用 in或者out或者ref 修饰泛型的场景

          讲解: 一个SubClass类实现了ITestIn接口,out定义输出类型,in定义输入类型,必须是把值输入,所以莫名的 ITestIn<string,object> 一下子就懂了。还不懂的话,建议你百度

          留个题目:如何 让用户写代码可以写出如下的效果,Student的值是不允许改变的,请设计接口

          public class Student:ICustomComparable<Student>{

                public int CompareTo(Student stu){

                    return ... ;

                }

          }

          参考部分答案(答案字体被我设置成白色了,查看的,自己选择后面的空白部分): public interface ICustomComparable<in T>{     int    CompareTo(T stu);   }

            

     1.8.2  其实上面的 in或者out修饰泛型,涉及到了泛型知识中的 协变(泛型参数被out修饰)与抗变(泛型参数被in修饰)的知识,不要太在意,会用就好!!!!!!oh! shit,抗英(国)

 

1.9 Nullable<T>      T必须是值类型,定义可为空,有兴趣的可以看看 public struct Nullable<T> where T:struct的实现

      等同写法:       Nullable<int>    a   等同于  int? a

      ?? 的用法: 例如 int y=x ?? 0;   如果x为null,则等于0,否则就是原值。

1.10 当然泛型也可以像用在类上那样用于方法上,这个知识太简单,不讲了

1.11 泛型委托,Lambda表达式最多,典型的有比如Func  Action等,这些在LINQ里再讲

1.12 作为一年以上的开发人员,都知道的我都跳过了,可能存在疑问的地方保留了。

 

 

 

======安徽六安=========www.ayjs.net==========aaronyang========杨洋==================

 

posted @ 2014-12-29 00:44  AYUI框架  阅读(1868)  评论(5编辑  收藏  举报