zoukankan      html  css  js  c++  java
  • 自己写一个泛型集合类型,可实现添加和遍历


    在"C#中List<T>是怎么存放元素的"中,分析了List<T>的源码,了解了List<T>是如何存放元素的。这次,就自定义一个泛型集合类型,可实现添加元素,并支持遍历。

     

    该泛型集合类型一定需要一个添加元素的方法,在添加元素的时候需要考虑:当添加的元素超过当前数组的容量,就让数组扩容;为了支持循环遍历,该泛型集合类型必须提供一个迭代器(实现IEnumerator接口)。

        public class MyList<T>
    
        {
    
            T[] items = new T[5];
    
            private int count;
    
            public void Add(T item)
    
            {
    
                if(count == items.Length)
    
                   Array.Resize(ref  items, items.Length * 2);
    
                items[count++] = item;
    
            }
    
            public IEnumerator<T> GetEnumerator()
    
            {
    
                return new MyEnumeraor(this);
    
            }
    
            class MyEnumeraor : IEnumerator<T>
    
            {
    
                private int index = -1;
    
                private MyList<T> _myList;
    
                public MyEnumeraor(MyList<T> myList)
    
                {
    
                    _myList = myList;
    
                }
    
                public T Current
    
                {
    
                    get
    
                    {
    
                        if (index < 0 || index >= _myList.count)
    
                        {
    
                            return default(T);
    
                        }
    
                        return _myList.items[index];
    
                    }
    
                }
    
                public void Dispose()
    
                {
    
                    
    
                }
    
                object System.Collections.IEnumerator.Current
    
                {
    
                    get { return Current; }
    
                }
    
                public bool MoveNext()
    
                {
    
                    return ++index < _myList.count;
    
                }
    
                public void Reset()
    
                {
    
                    index = -1;
    
                }
    
            }
    
        }
    

    ○ 泛型集合类型维护着一个T类型的泛型数组
    ○ 私有字段count是用来计数的,每添加一个元素计数加1
    ○ 添加方法考虑了当count计数等于当前元素的长度,就让数组扩容为当前的2倍
    ○ 迭代器实现了IEnumerator<T>接口

     

    客户端调用。

        class Program
    
        {
    
            static void Main(string[] args)
    
            {
    
                MyList<int> list = new MyList<int>();
    
                list.Add(1);
    
                list.Add(2);
    
                foreach (int item in list)
    
                {
    
                    Console.WriteLine(item);
    
                }
    
                Console.ReadKey();
    
            }
    
        }
    

    另外,IEnumerable和IEnumerator的区别是什么呢?
    其实,真正执行迭代的是IEnumerator迭代器。IEnumerable接口就提供了一个方法,就是返回IEnumerator迭代器。

    public interface IEnumerable
    
    {
    
        IEnumerator GetEnumerator();
    
    }
    

      关于#1楼曾向秋的评论

    如果按照曾向秋的方式在客户端调用,输出的结果是0,不是1,显然不是我们所期望的。之所以不是1是因为:当调用list.GetEnumerator()的时候,就会调用MyEnumeraor类的构造函数,而我们没有在构造函数中让集合的指针自增1。

     

    public MyEnumeraor(MyList<T> myList)
    
    {
    
        _myList = myList;
    
    }

    现在,我们需要在MyEnumeraor类的构造函数,让集合的指针自增1,并且需要把index变量作为静态字段放在MyList类下。

     

        public class MyList<T> : IEnumerable<T>
    
        {
    
            T[] items = new T[5];
    
            private int count;
    
            private static int index = -1;
    
            public void Add(T item)
    
            {
    
                if(count == items.Length)
    
                   Array.Resize(ref  items, items.Length * 2);
    
                items[count++] = item;
    
            }
    
            public IEnumerator<T> GetEnumerator()
    
            {
    
                return new MyEnumeraor(this);
    
            }
    
            class MyEnumeraor : IEnumerator<T>
    
            {           
    
                private MyList<T> _myList;
    
                public MyEnumeraor(MyList<T> myList)
    
                {
    
                    index++;
    
                    _myList = myList;
    
                }
    
                public T Current
    
                {
    
                    get
    
                    {
    
                        if (index < 0 || index >= _myList.count)
    
                        {
    
                            return default(T);
    
                        }
    
                        return _myList.items[index];
    
                    }
    
                   
    
                }
    
                public void Dispose()
    
                {                
    
                }
    
                object IEnumerator.Current
    
                {
    
                    get { return Current; }
    
                }
    
                public bool MoveNext()
    
                {
    
                    return ++index < _myList.count;
    
                    
    
                }
    
                public void Reset()
    
                {
    
                    index = -1;
    
                }
    
            }
    
            IEnumerator IEnumerable.GetEnumerator()
    
            {
    
               return GetEnumerator();
    
            }
    
        }

    然后,客户端就可以想像曾向秋所说的那样调用了。

     

            static void Main(string[] args)
    
            {
    
                MyList<int> list = new MyList<int>();
    
                list.Add(1);
    
                Console.WriteLine(list.GetEnumerator().Current);
    
                list.Add(2);
    
                Console.WriteLine(list.GetEnumerator().Current);
    
                Console.ReadKey();
    
            }
  • 相关阅读:
    【报错问题】mysql无法使用别名查询
    【报错问题】java.lang.IllegalStateException: It is illegal to call this method if the current request is not in asynchronous mode
    【源码】-springboot 启动后立马执行的方式
    【LINUX】$搭配使用的含义
    【JAVA】javaMail附件名超过60显示错误
    【Gradle】简单入门
    慢SQL案例之一
    【Flink】一. 什么是Flink?
    【spring基础】环境的搭建与后台
    [org.apache.common][打算学习开源工具包]
  • 原文地址:https://www.cnblogs.com/darrenji/p/3978354.html
Copyright © 2011-2022 走看看