应无所住,而生其心
排名
1
文章
860
粉丝
112
评论
163
net core webapi post传递参数
庸人 : 确实坑哈,我也是下班好了好几次,发现后台传递对象是可以的,但...
百度编辑器自定义模板
庸人 : 我建议换个编辑器,因为现在百度富文本已经停止维护了,用tinymec...
ICP备案 :渝ICP备18016597号-1
网站信息:2018-2025TNBLOG.NET
技术交流:群号656732739
联系我们:contact@tnblog.net
公网安备:50010702506256
欢迎加群交流技术

c#使用泛型实现一个自己的list

7673人阅读 2019/9/27 10:54 总访问:5182377 评论:0 收藏:0 手机
分类: .NET

实现一个自己list,实现微软自带list的常用功能,就是可以直接把list对象名换成自己的使用


例如:这里把MyList换成List效果一样

  1. static void Main(string[] args)
  2. {
  3.     MyList<int> intlist = new MyList<int>();
  4.     for (int i = 0; i < 8; i++)
  5.     {
  6.         intlist.Add(i);
  7.     }
  8.     //删除
  9.     intlist.Remove(0);   
  10.     Console.WriteLine("条数:" + intlist.Count);
  11.     Console.WriteLine("容量:" + intlist.Capacity);
  12.     //遍历
  13.     for (int i = 0; i < intlist.Count; i++)
  14.     {
  15.       Console.WriteLine("item:" + intlist[i]);
  16.     }
  17.     //foreach遍历
  18.     foreach (int item in intlist)
  19.     {
  20.       Console.WriteLine(item);
  21.     }
  22. }

效果

添加功能:

添加功能其实很简单,因为list里边是维护的一个数组,数组的长度是固定的,我们只需要在装满的时候进行扩容就行了,测试了一下微软自带list扩容规则是上一次的2倍,代码实现如下:

  1. public void Add(T t)
  2. {
  3.     //当装不下的时候扩容
  4.     int length = array.Length;
  5.     if (index == length)
  6.     {
  7.         //创建一个新数组
  8.         T[] temp = new T[length * 2];
  9.         Array.Copy(array, temp, length);
  10.         array = temp;
  11.     }
  12.     array[index] = t;
  13.     index++;
  14. }

这里扩容的代码放到index++下面也可以,不影响功能,只是如果放到下面index++之后,他装满会马上扩容,如果是放到上面刚好装满的情况不会扩容,要下一次来存储的时候发现存不下了才会扩容,这样效率会快一点,说的高大上一点就是延迟扩容,按需扩容


获取条数与容量:

这里我们就先弄成只读就好,外面无法修改,无法破坏我们的逻辑,这点在自己写框架什么的很非常重要哇。面向对象的三大特征的之一的封装

  1.     //list条数
  2.     public int Count
  3.     {
  4.         get
  5.         {
  6.             return index;
  7.         }
  8.     }
  9.     //list容量
  10.     public int Capacity
  11.     {
  12.         get
  13.         {
  14.             return array.Length;
  15.         }
  16.     }

删除功能:

因为list里边是维护的一个数组,删除功能也是对数组的一个操作

要做删除当然第一步就是找到需要删除元素的位置


  • 方法1:创建一个新的数组,然后把除了删除的元素复制到新数组去,也就是跳过需要删除的元素即可



方法1:分成删除前,删除后

  1. //先把删除前面的装进来
  2. for (int i = 0; i < index; i++)
  3. {
  4.     newArray[i] = array[i];
  5. }
  6. //然后把需要的跳过,把后面的数据装过来,这里循环长度使用实际存储来计算,不要使用数组的长度来,因为很有可能没有装满
  7. for (int i = index; i < this.index - 1; i++)
  8. {
  9.     newArray[i] = array[i + 1];
  10. }
  11. array = newArray;


方法2:不用分删除前,删除后

  1. public void Remove(T t)
  2. {
  3.     //需要删除的位置
  4.     int poi = -1;
  5.     for (int i = 0; i < Count; i++)
  6.     {
  7.         if (t.Equals(array[i]))
  8.         {
  9.             poi = i;
  10.             break;
  11.         }
  12.     }
  13.     if (poi == -1)
  14.     {
  15.         return;
  16.     }
  17.     //创建一个新数组
  18.     T[] temp = new T[array.Length];
  19.     for (int i = 0, j = 0; i < array.Length - 1; i++, j++)
  20.     {
  21.         //这个刚好是需要被删除
  22.         /*可以这样理解,当i和poi没有相等的时候,i=j所以newArray[i] = array[j]就是直接相等,没有错位移动格子
  23.          当i和poi相等的时候j就比i多1就相当于移动一格了。所以这个i和j是否相等就相当于方法1中分成的删除前和删除后的操作了
  24.         */
  25.         if (i == poi)
  26.         {
  27.             j++;
  28.         }
  29.         temp[i] = array[j];
  30.     }
  31.     array = temp;
  32.     index--;
  33. }
  • 方法3:其实没有必要创建一个新的数组,我们可以在当前这个数组上进行位置的改动,删除位置前不需要动后面的位置依次向前移动即可


 代码的话很简单一个Array.Copy即可搞定

  1. public void Remove(T t)
  2. {
  3.     //找到需要删除的位置
  4.     var poi = FindIndex(t);
  5.     //参数解释:源数组,源数组开始位置 , 目标数组,目标数组开始位置,循环赋值次数
  6.     Array.Copy(array, poi + 1array, poi, index - poi - 1);
  7.     //数组长度减一,因为删除了长度会减少一
  8.     index--;
  9. }

开始可以不使用Array.Copy可以自己使用循环来写,好理解一点

  1. //循环的长度是数组的实际存储数量而不是总数,如果没有存储满,就会移动很多次,这样会浪费效率
  2. for (int i = index; i < this.index - 1; i++)
  3. {
  4.     array[i] = array[i + 1];
  5. }


这里发现了一个有趣的问题,就是我们依次向前移动后,最后一个位置的数组是否存在的问题,

比如list集合里边存储有1,2,3,4我们把3删除掉,4就会向3的位置移动变成了:1,2,4 

但是以前那个4会不会存在,会不会集合里边变成了1,2,4,4

我们正常的遍历肯定是没有问题的:

但是就是想测试一下后面那个元素,我们就用for循环,长度就是要加一个1,因为其实用户已经删除了一个

果然像我们推理的那样,然后我们有理由怀疑list本身会不会也是这样的,测试一下你就知道其实当你传递下标和list长度相同的时候就会报索引超出了长度,所以其实对于删除来说后面元素是什么值对使用根本不会有影响。

我们在自己的list实现一个取值下标超出问题。

  1. public T this[int _index]
  2. {
  3.     get
  4.     {
  5.         if (_index >= index)
  6.         {
  7.             throw new Exception("索引超出了长度");
  8.         }
  9.         return array[_index];
  10.     }
  11.     set
  12.     {
  13.         array[index] = value;
  14.         index++;
  15.     }
  16. }

但是呢,虽然移动后的数组对正常使用没有问题,但是还是想知道一下微软本身list里边维护的数组是否对这样的情况进行处理。正常情况下我们无法访问list中维护的私有数组,但是我们可以通过反射来。

  1. static void Main(string[] args)
  2. {
  3.     List<int> intlist = new List<int>();
  4.     for (int i = 1; i < 5; i++)
  5.     {
  6.         intlist.Add(i);
  7.     }
  8.     intlist.Remove(3);
  9.     //反射获取list中私有字段
  10.     var result = intlist.GetType().GetFields(BindingFlags.NonPublic | BindingFlags.Instance);
  11.     int[] findArray = null;
  12.     foreach (var item in result)
  13.     {
  14.         //找到list中维护的数组,不要问我怎么找到的,遍历看名字就知道了
  15.         if (item.Name == "_items")
  16.         {
  17.             findArray = (int[])item.GetValue(intlist);
  18.         }
  19.     }
  20.     foreach (int item in findArray)
  21.     {
  22.         Console.WriteLine(item);
  23.     }
  24.     Console.ReadLine();
  25. }

输出如下:

微软果然严谨这种问题都处理了,微软估计是赋予了这个泛型的默认值,ok跟着大佬的步伐我们也去这样处理一下

  1. public void Remove(T t)
  2. {
  3.     var poi = FindIndex(t);
  4.     Array.Copy(array, poi + 1array, poi, index - poi - 1);
  5.     //处理移动最后一个的元素为该类型的默认值
  6.     array[index-1] = default(T);
  7.     index--;
  8. }

就是这样一句:

array[index-1] = default(T);


ok我们把list换成我们自己的mylist试试,因为我自己写的list,里边维护的数组叫array所以也要修改一下

ok搞定研究点这种问题还是有点意思



实现删除一个范围RemoveRange:

代码如下,其实只要把删除功能做好,这个还是比较简单的

  1. /// <summary>
  2. /// 从开始位置删除一个范围
  3. /// </summary>
  4. /// <param name="begin">开始位置</param>
  5. /// <param name="length">长度</param>
  6. public void RemoveRange(int begin, int length)
  7. {
  8.     //删除0个无意义
  9.     if (length == 0)
  10.         return;
  11.     //循环次数应该等于Count - length,因为删除一个的时候是Count - 1,后面这个1就是删除长度,那删除Length个自然就是Count - length了
  12.     for (int i = begin; i < Count - length; i++)
  13.     {
  14.         //删除需要跳格应该等于i+length,如果是删除一个就应该只是跳一格,也就是 array[i] = array[i + 1];
  15.         array[i] = array[i + length];
  16.     }
  17.     //删除的长度要减去删除的长度,如果是删除一个就是index = index-1即可
  18.     index = index - length;
  19. }

批量删除功能RemoveAll:

当list集合中有重复的时候进行删除,其实也很简单,真正的核心是根据下标删除,这些删除只是调用核心的删除方法而已

  1. public void RemoveAll(T t)
  2. {
  3.     for (int i = 0; i < Count; i++)
  4.     {
  5.         if (array[i].Equals(t))
  6.         {
  7.             RemoveAt(i);
  8.         }
  9.     }
  10. }


资源下载地址:https://download.tnblog.net/resource/index/8965edcfbdd2437293fbc9d546b2c00c


欢迎加群讨论技术,1群:677373950(满了,可以加,但通过不了),2群:656732739。有需要软件开发,或者学习软件技术的朋友可以和我联系~(Q:815170684)

评价

css弹性盒子,flex布局

css弹性盒子由于版本不同浏览器问题造成了一些不同的写法display:flexbox;在google浏览器中如果使用下面的写法就不行displa...

css图片和文字对齐问题

文字和图片写到一排经常会出现对不齐的问题 这样感觉图片会上来一点没有和文字对齐,如下图 但是如果修改下html结...

GitHub 上传项目

补充简化方法:登录git创建项目--&gt;拉取刚刚创建的项目--&gt;复制需要的代码进去--&gt;上传提交即可先拉取项目在上传代码...

NET core 使用 EF code First

下面这些内容很老了看这篇:https://www.tnblog.net/aojiancc2/article/details/5365 项目使用多层,把数据库访问...

Windows平台分布式架构实践 - 负载均衡

原文地址: https://www.cnblogs.com/atree/p/windows_loadbalancer.html 概述  最近.NET的世界开始闹腾了,微软官方终...

css实现简单矩形对话框

在前端做项目时,我们可能会遇到写对话框的需求,这次做视频会议页面就遇到了,记录下日后有个参照。//网页部分 &lt;divcla...

cAPS.NET 保存base64位格式的图片

publicvoidUpload() { //取出图片对应的base64位字符 stringimgBase=Request[&quot;imgBase&quot;]; //c#里边的base6...

使用OLEDB读取不同版本Excel的连接字符串设置

使用OleBD读取excel的时候,excel不同的版本,连接字符串的写法也会不一样。///&lt;summary&gt; ///读取excel ///&lt;/su...

vs2017 对 cOM 组件的调用返回了错误 HRESULT E_FAIL

vs2017添加引用报错 对 COM 组件的调用返回了错误 HRESULT E_FAIL 1.以管理员身份打开vs2017开发人员命令指示符 2...

分布式服务架构与微服务架构概念的区别与联系

分布式:分散压力。微服务:分散能力。当下理解分布式:不同模块部署在不同服务器上作用:分布式解决网站高并发带来问题集...

分布式-微服务-集群的区别

1.分布式将一个大的系统划分为多个业务模块,业务模块分别部署到不同的机器上,各个业务模块之间通过接口进行数据交互。区...

NPOI操作excel 2007/2010版本

HSSFWorkbook:是操作Excel2003以前(包括2003)的版本,扩展名是.xlsXSSFWorkbook:是操作Excel2007的版本,扩展名是.xlsx先...

这样学英语三个月超过你过去学三年

本文作者三年间从四级勉强及格到高级口译笔试210,口试232。找工作面试时给其口试的老外考官听了一分钟就说你的英语不用考...

EasyUI弹窗批量修改combogrid下拉框的值

JS方法//点击弹出批量修改框 UpdateLot:function(){ varrow=$(&quot;#dg&quot;).datagrid(&quot;getChecked&quot;); if(...

js与controller中分割字符串的方法

js: varstr=OpenRule; varstrs=newArray(); strs=str.split(&quot;,&quot;); for(vari=0;i&lt;strs.length;i++){ $(&q...

如何修改cSS中存在的element.style内联样式

改腾讯地图的时候调整了下样式,发现样式一直存在问题,修改style里面的值,一点用都没有,html中这个值还找不到是在哪里出...