C#实现泛型动态循环数组队列的方法

作者:Mib_D 时间:2022-11-03 05:05:42 

任务

循环数组

实现目标:(1)创建一个新的数组数据结构;

(2)该数据结构为泛型;

(3)可以按照元素多少进行扩容缩容;

(4)进行添加删除操作的时间复杂度小于O(n);

优势:在取出放入的操作中消耗的资源更少

劣势:取出特定元素或特定下标元素平均消耗的资源为普通数组平均消耗资源的最大值

循环数组队列

实现目标:(1)根据循环数组构建出循环的队列数据结构

优势:节省资源,运行速度快;

劣势:不能灵活取出

重点:如何实现循环的计算下标语句。

循环下标语句

完整代码:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;

namespace DataStructrue
{
   /// <summary>
   /// 循环数组
   /// (1)添加功能
   /// (2)删除功能
   /// (3)查询(任何、首尾处)功能
   /// (4)修改(任何,首位处)功能
   /// </summary>
   /// <typeparam name="E"></typeparam>
   class Array2<E>
   {
       private E[] data;
       private int N;
       private int first;
       private int last;
       public Array2(int capacity)
       {
           data = new E[capacity];
           N = 0;
           first = 0;
           last = 0;
       }
       public Array2() : this(10) { }
       public int Count { get { return N; } }
       public bool IsEmpty { get { return N==0; } }
       public E GetFirst()
           return data[first];
       /// <summary>
       /// 添加一个元素
       /// </summary>
       /// <param name="e"></param>
       public void Add(E e)
           if (N==data.Length)
           {
               ResetCapacity(data.Length*2);
           }
           data[last] = e;
           last = (last + 1) % data.Length;
           N++;
       /// 移除早放进的一个元素
       /// <returns></returns>
       public E RemoveLast()
           if (N==0)
               throw new ArgumentException("队列已空");
           if (N<=(data.Length/4))
               ResetCapacity(data.Length / 2);
           E de = data[first];
           data[first] = default;
           first = (first + 1) % data.Length;
           N--;
           return de;
       /// 移除特定下标元素
       /// 消耗大,不建议使用
       /// <param name="index"></param>
       public E RemoveAt(int index)
           if (index > data.Length || index < 0 ||N==0)
               throw new ArgumentException("非法索引");
           if (first > last)
               if (index < first && index >= last)
               {
                   throw new ArgumentException("非法索引");
               }
           else if (last > first)
           E rd = data[index];
           for (int i = index+1; i !=last ; i=(i+1)%data.Length)
               data[i-1] = data[i];
           last--;
           return rd;
       /// 移除特定元素
       public E Remove(E e)
           for (int i = first; i !=last; i=(i+1)%data.Length)
               if (data[i].Equals(e))
                   return RemoveAt(i);
           return data[last];
       /// 对数组进行扩容操作
       /// <param name="newcapacity"></param>
       private void ResetCapacity(int newcapacity)
           E[] data2 = new E[newcapacity];
           for (int i = 0; i < N; i++)
               data2[i] = data[first];
               first = (first + 1) % data.Length;
               last = i+1;
           data = data2;
       public override string ToString()
           //实例化
           StringBuilder res = new();
           //重写格式1:输出数组元素个数以及长度
           //res.Append(string.Format("Array1:   count={0}    capacity={1}\n",N,data.Length));
           res.Append(string.Format("A2Queue: Count = {0} Capacity = {1}\n[", N, data.Length));
               res.Append(data[(first+i)%data.Length]);
               if (i!=N-1)
                   res.Append(',');
           res.Append(']'+"\n");
           //返回
           return res.ToString();
   }
}

补充:c#使用数组实现泛型队列Quque<T>,以循环的方式使用数组提高性能

队列简述

一种先进先出的数据结构

本文主旨

  • 提供一个确定容量的高性能队列的实现

  • 更进一步可以对队列做动态扩容,每次队列满了的时候增加队列容量

  • 队列也可以使用链表实现

实现代码

using System;

namespace DataStructure
{
   /// <summary>
   /// 用数组实现队列
   /// 用2个index标记开始合结束
   /// </summary>
   /// <typeparam name="T"></typeparam>
   public class ArrayQueue<T>
   {
       private int mCapicity;
       private int mStartIndex;
       private int mEndIndex;
       private int mCount;
       private T[] mArray;
       public ArrayQueue(int capicity)
       {
           mCapicity = capicity;
           mArray = new T[capicity];
       }
       public int Count
           get
           {
               return mCount;
           }
       public bool IsFull
               return mCount == mCapicity;
       public int Capicity
           get { return mCapicity; }
       public bool IsEmpty
               return mCount == 0;
       public void Clear()
           mStartIndex = 0;
           mEndIndex = 0;
           mCount = 0;
           mCapicity = 0;
           mArray = null;
       public void Enqueue(T e)
           //队列满了
           if (IsFull)
               throw new Exception("queue is full");
           mArray[mEndIndex] = e;
           mCount++;
           //计算下一个位置
           mEndIndex++;
           if (mEndIndex == mCapicity)
               mEndIndex = 0;
       public T Dequeue()
           //队列空
           if (IsEmpty)
               throw new Exception("queue is empty");
           var r = mArray[mStartIndex];
           //计算下一次取元素的index
           //取出元素后增加start
           mStartIndex++;
           //到达尾部,开始循环,下一次从头开始取
           if (mStartIndex == mCapicity)
               mStartIndex = 0;
           mCount--;
           return r;
   }
}

测试代码

namespace DataStructure
{
   public class ArrayQueueTest : BaseSolution
   {
       public void Test()
       {
           var queue = new ArrayQueue<int>(4);
           queue.Enqueue(1);
           queue.Enqueue(2);
           queue.Enqueue(3);
           queue.Enqueue(4);
           // println(queue.Capicity);
           // println(queue.Count);

println(queue.Dequeue());
           queue.Enqueue(5);
           while (!queue.IsEmpty)
           {
               println(queue.Dequeue());
           }
       }
   }
}

来源:https://www.cnblogs.com/yotomib/p/15854490.html

标签:C#,泛型,动态循环,数组队列
0
投稿

猜你喜欢

  • Android Studio 报错failed to create jvm error code -4的解决方法

    2023-01-22 03:13:49
  • winform拦截关闭按钮触发的事件示例

    2022-05-19 00:06:14
  • SpringBoot多线程进行异步请求的处理方式

    2021-11-10 10:48:30
  • 深入理解Android热修复技术原理之so库热修复技术

    2023-11-19 15:02:10
  • spring项目中切面及AOP的使用方法

    2021-12-01 21:11:29
  • Android onClick方法与setOnClickListener方法对比

    2022-02-09 22:21:40
  • c# 获取CookieContainer的所有cookies函数代码

    2023-06-17 23:11:30
  • Java解析使用JSON的多种方法

    2022-08-13 00:18:01
  • JavaWeb实现文件上传下载功能实例详解

    2023-05-08 19:43:51
  • Java数据结构及算法实例:冒泡排序 Bubble Sort

    2022-10-17 08:39:45
  • C#如何通过T4自动生成代码详解

    2021-12-21 15:43:46
  • 如何在C# 枚举中增加行为

    2022-10-28 06:11:34
  • Android编程获取网络时间实例分析

    2022-08-05 16:28:01
  • Java并发编程之ConcurrentLinkedQueue源码详解

    2023-01-22 16:19:51
  • Android实现简单的照相功能

    2023-07-01 08:46:16
  • Android中导航组件Navigation的实现原理

    2022-08-25 13:11:47
  • Android 使用SharePerference判断是否为第一次登陆的实现代码

    2021-07-31 18:51:32
  • springboot集成与使用Sentinel的方法

    2023-08-31 11:22:30
  • 初步编写IDEA\\AndroidStudio翻译插件的方法

    2022-12-14 19:30:30
  • Java实现删除排序链表中的重复元素的方法

    2022-11-28 08:27:22
  • asp之家 软件编程 m.aspxhome.com