C#中的列表结合List与ArrayList介绍

来源:互联网 发布:华力微电子 知乎 编辑:程序博客网 时间:2024/05/21 12:46
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Collections;
  4. using System.Linq;
  5. using System.Text;
  6. using System.Collections.ObjectModel;
  7. namespace 集合
  8. {
  9.     class Program
  10.     {
  11.         static void Main(string[] args)
  12.         {
  13.             //using System.Collections.Generic; 命名空间中的List<T>
  14.             //using System.Collections; 命名空间中的ArrayList 
  15.             //都实现了列表集合,一个是泛形集合,一个是非泛型的
  16.             //下面我们将Person对象加到集合中
  17.             Person p1 = new Person( "aladdin" , 20 );
  18.             Person p2 = new Person("zhao", 10);
  19.             Person p3 = new Person("jacky", 40);
  20.             //如果不制定list的容器大小,默认是0,只要有元素加入是,会自动扩展到4,如果第5个元素加入时,就变成了8,第9个加入,就成16
  21.             //可以看出,总是成倍的增长,扩展时要重新开辟内存,这样会影响效率,如果事先知道元素个数,或者可能个数,最好给个尽量大的权衡值
  22.             //我们加入3个元素,设容器大小为4.注:设为4不是指只能放4个元素,如果超出,一样也会成倍扩展,这样做只是为了尽量扩展带来的开销
  23.             List<Person> list = new List<Person>(4);
  24.           
  25.             list.Add(p1);
  26.             list.Add(p2);
  27.             list.Add(p3);
  28.             //本方法是清除多于的没有用的内存空间,例:如果开辟大小为100,而我们只用了4个,其余的放着,是不是很浪费 
  29.             //本方法调用时会检查元素个数是不是占到了容器大小的90%以上,如果是,则不进行回收.
  30.             list.TrimExcess();
  31.             //ArrayList方法与List<>用法一样,不同的是,它是对象集合,参数是Object这样会有装箱拆箱的可能,尽量用List<>
  32.             //本处不再做演示
  33.             // 1 初始化集合器
  34.             // C#3.0开始,提供了初始化功能,但是并没有反应到IL代码中,在IL中,一样也是把个转化成ADD方法来调用
  35.             List<int> l2 = new List<int>() { 1 ,2 ,3 ,4 ,5 };
  36.             // 2 添加元素 AddRange() 本方法可以一次性添加一批对象
  37.             List<Person> lists = new List<Person>(10);
  38.             //参数是一个必须可能跌代的对象,也可是数组 
  39.             list.AddRange( new Person[] { new Person( "aladdin" ,20) , new Person("zhao",6)});
  40.             
  41.             //构造传入批量参数 ,与AddRange效果一样
  42.             List<Person> mylist = new List<Person>(new Person[] { new Person( "aladdin" ,20) , new Person("zhao",6)});
  43.             // 3 插入元素
  44.             // 使用Insert()方法,可以在指定位置插入元素
  45.             // 例 我们在1位置插入 则最后变成了 aladdin jacky zhao..插入意思就是,这个位我占了,以前占这位的和他之后的,通通往后移一位
  46.             mylist.Insert( 1 , new Person( "jacky" , 88 ));
  47.             foreach (Person p in mylist)
  48.             {
  49.                 Console.WriteLine( p.name );
  50.             }
  51.             // 4 访问元素
  52.             // ArrayList 与 List<T>都是提供了索引器来访问的
  53.             Console.WriteLine( "----------------访问元素------------------------");
  54.             for (int i = 0; i < mylist.Count; i++)
  55.             {
  56.                 Console.WriteLine(mylist[i].name);
  57.             }
  58.             //还可以使用foreach跌代器来实现,些处不再举例
  59.             //使用Foreach方法
  60.             //public delegate void Action<T>(T obj);例用委托做为参数 
  61.             //些处我们用呀妈Day表达式实现
  62.             Console.WriteLine( "-----------------用ForEach方法输出------------------------");
  63.             mylist.ForEach( param => Console.WriteLine( param.name) ) ;
  64.             // 5删除元素
  65.             //删除元素可以使用RemoveAt()直接传入索引器值
  66.             //将第一个元素直接删除
  67.             mylist.RemoveAt(0);
  68.             //也可以将要删除的元素传给Remove方法
  69.             List<Person> lists2 = new List<Person>(10);
  70.             Person per1 = new Person( "aladdin" , 100 );
  71.             Person per2 = new Person("zhao", 100);
  72.             Person per3 = new Person("jacky", 100);
  73.             lists2.Add(per1);
  74.             lists2.Add(per2);
  75.             lists2.Add(per3);
  76.             lists2.Remove(per3);
  77.             Console.WriteLine( "-------删除后的元素---------");
  78.             foreach (Person per in lists2)
  79.             {
  80.                 Console.WriteLine( per.name );
  81.             }
  82.             //从结果可以看出 名称为Jacky的元素被删除了
  83.             //下面说一下Remove方法的删除过程 
  84.             // 用IndexOf方法确定出对象的索引,然后按索引删除
  85.             // 在IndexOf方法内,首先检查元素是不是实现了IEquatable接口,如果是,就调用这个接口中的Equals方法
  86.             // 如果没有实现,则调用Object中的Equals方法比较元素(也就是址址比较)
  87.             // 以上我们删除per3,很显明显一个地址,所以被删除了 
  88.             // 下面我们改装了Person ,实现了IEquatable<Person>,在比较方法中,始终返回false , 则per3会比较失败,不会被删除
  89.             // 结果3个都在
  90.             // 如果要删除对象,最好使用索引直接删除,因为Remove方法经历了一系列过程后,最后才按索引删除!
  91.             // RemoveRange()删除一个范围
  92.             // 第一个参数 开始位置 第二个 个数
  93.             //lists2.RemoveRange( 1 , 2 );
  94.             //Console.WriteLine( "批量删除后----------------");
  95.             //foreach (Person per in lists2)
  96.             //{
  97.             //    Console.WriteLine(per.name);
  98.             //}
  99.             // 6 搜索
  100.             // 搜索有很多种方式,可以使用IndexOf LastIndexOf FindIndex FindLasIndex Find FindLas ,如果只是查看元素存不,可以使用Exists()方法
  101.             // IndexOf() 方法 需要将一个对象做参数, 如果打到,就返回本元素在集合中的索引,如果找不到就返回-1,IndexOf还可以使用IEquatable接口来比较元素
  102.             List<Person> ls3 = new List<Person>(10);
  103.             Person person1 = new Person("aladdin", 100);
  104.             Person person2 = new Person("zhao", 100);
  105.             Person person3 = new Person("jacky", 100);
  106.             ls3.Add(person1);
  107.             ls3.Add(person2);
  108.             ls3.Add(person3);
  109.             // 为了使用默认的地址比较,我们把Person的接口暂时去掉
  110.             int index = ls3.IndexOf(person3);
  111.             Console.WriteLine( "per3 的索引:" + index); //2
  112.             // 还可以指定搜索范围 从第3个开始,范围长度是1
  113.             int index2 = ls3.IndexOf(person3,2,1);
  114.             Console.WriteLine(index2);
  115.             //IEquatable比较方法前面已经写过,不再举例
  116.             // FindIndex()方法是用来搜索带有一定特性的元素
  117.             // 例用委托做参数  public delegate bool Predicate<T>(T obj);
  118.             int index3 = ls3.FindIndex(param => param.name.Equals("jacky"));
  119.             Console.WriteLine( index3 );// 2
  120.             // FindLastIndex是从后面查第一个出现的元素,因为我们这里没有重复元素,所以体现不出他只查找一个,就停下来的效果
  121.             int index4 = ls3.FindLastIndex(p => p.name.Equals("aladdin"));
  122.             Console.WriteLine(index4);
  123.             // Find方法与FindIndex方法用法一样,不同的是,它返回的是元素本身
  124.             Person ppp = ls3.Find( p => p.name.Equals("jacky")) ;
  125.             Console.WriteLine(ppp);
  126.             // 如果要查找所有的匹配元素,而不是找到第一个就停下来,就使用FindAll方法
  127.             // 我们查找所有年纪等于100的对象,3个都符合
  128.             List<Person> newList = ls3.FindAll(p => p.age == 100);
  129.             Console.WriteLine( "----------查找所有---------");
  130.             foreach (Person p in newList)
  131.             {
  132.                 Console.WriteLine( p.name );
  133.             }
  134.             // 7 排序
  135.             // List可以例用Sort方法排序,实现算法是快速排序
  136.             // 本方法有好几个重载
  137.           
  138.             //public void Sort(); //只对元素实现了IComparable才能使用这个方法 ,如果实现了则,可以直接调用一次sort之后,就排好序了
  139.             //public void Sort(Comparison<T> comparison); //我们的Person并没有实现那个接口,所以要用泛型委托当参数的方法
  140.             //public void Sort(IComparer<T> comparer); //泛型接口当参数 public delegate int Comparison<T>(T x, T y);
  141.             //public void Sort(int index, int count, IComparer<T> comparer); //可以指定范围
  142.             List<Person> ls4 = new List<Person>(10);
  143.             Person person4 = new Person("aladdin", 100);
  144.             Person person5 = new Person("zhao", 33);
  145.             Person person6 = new Person("jacky", 44);
  146.             ls4.Add(person4);
  147.             ls4.Add(person5);
  148.             ls4.Add(person6);
  149.             ls4.Sort(MyComparFunc);
  150.             Console.WriteLine( "-------------排序后的-------------");
  151.             foreach (Person p in ls4)
  152.             {
  153.                 Console.WriteLine( p.name + p.age );
  154.             }
  155.             Console.WriteLine( "--------颠倒循序------------------");
  156.             ls4.Reverse();
  157.             foreach (Person p in ls4)
  158.             {
  159.                 Console.WriteLine(p.name + p.age);
  160.             }
  161.             // 8 类型转换  可以将集合中的元素转换成任意类型的元素,比如,我们要将集合中的Person转换成为Racer对象Racer只包含名字,没有年纪
  162.             // public List<TOutput> ConvertAll<TOutput>(Converter<T, TOutput> converter);
  163.             // public delegate TOutput Converter<TInput, TOutput>(TInput input);  委托参数
  164.             List<Racer> ls5 = ls4.ConvertAll<Racer>((input) => new Racer( input.name)) ;
  165.             Console.WriteLine( "-----------转换后的玩意--------");
  166.             foreach (Racer r in ls5)
  167.             {
  168.                 Console.WriteLine( r.name );
  169.             }
  170.             // 9 只读集合
  171.             // 在创建完集合以后,肯定是可读写的,如果不是,他就不能再添加新元素了,但是,如果是认为填充完毕,不要再做修改.
  172.             // 可以使用只读集合,使用AsReadOnly方法() 返回ReadOnlyCollection<T>类型,它与List<>操作是一样的,但是一但有修改集合的操作,就会刨出异常
  173.             // 他屏蔽了通常的ADD等方法
  174.             ReadOnlyCollection<Racer> persss =  ls5.AsReadOnly();
  175.             Console.WriteLine("输出只读集合");
  176.             foreach (Racer r in persss)
  177.             {
  178.                 Console.WriteLine( r.name );
  179.             }
  180.             Console.ReadLine();
  181.         }
  182.         //为了比较,写的委托实现方法,这个用呀妈Day表达式有不太好写
  183.         public static  int MyComparFunc(Person p1, Person p2)
  184.         {
  185.             if (p1.age == p2.age)
  186.             {
  187.                 return 0;
  188.             }
  189.             else if (p1.age > p2.age)
  190.             {
  191.                 return 1;
  192.             }
  193.             else
  194.             {
  195.                 return -1;
  196.             }
  197.         }
  198.     }
  199.     class Person//:IEquatable<Person>
  200.     {
  201.         public string name;
  202.         public int age;
  203.         public Person( string name , int age )
  204.         {
  205.             this.name = name;
  206.             this.age = age;
  207.         }
  208.         ////始终给一个False值
  209.         //public bool Equals(Person other)
  210.         //{
  211.         //    return false;
  212.         //}
  213.     }
  214.     class Racer
  215.     {
  216.         public string name;
  217.         public Racer(string name)
  218.         {
  219.             this.name = name;
  220.         }
  221.     }
  222. }