服务器之家:专注于服务器技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - C# - C#环形队列的实现方法详解

C#环形队列的实现方法详解

2021-12-07 14:34独孤飞 C#

这篇文章先是简单的给大家介绍了什么是环形队列和环形队列的优点,然后通过实例代码给大家介绍C#如何实现环形队列,有需要的朋友们可以参考借鉴,下面来一起看看吧。

一、环形队列是什么

队列是一种常用的数据结构,这种结构保证了数据是按照“先进先出”的原则进行操作的,即最先进去的元素也是最先出来的元素.环形队列是一种特殊的队列结构,保证了元素也是先进先出的,但与一般队列的区别是,他们是环形的,即队列头部的上个元素是队列尾部,通常是容纳元素数固定的一个闭环。

二、环形队列的优点

 1.保证元素是先进先出的

        是由队列的性质保证的,在环形队列中通过对队列的顺序访问保证。

 2.元素空间可以重复利用

       因为一般的环形队列都是一个元素数固定的一个闭环,可以在环形队列初始化的时候分配好确定的内存空间,当进队或出队时只需要返回指定元素内存空间的地址即可,这些内存空间可以重复利用,避免频繁内存分配和释放的开销。

 3.为多线程数据通信提供了一种高效的机制。

       在最典型的生产者消费者模型中,如果引入环形队列,那么生成者只需要生成“东西”然后放到环形队列中即可,而消费者只需要从环形队列里取“东西”并且消费即可,没有任何锁或者等待,巧妙的高效实现了多线程数据通信。

三、c#环形队列的实现

看了一个数据结构的教程,是用c++写的,可自己c#还是一个菜鸟,更别说c++了,但还是大胆尝试用c#将其中的环形队列的实现写出来,先上代码:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
public class myqueue<t> : idisposable
 {
  private t[] queue;
  private int length;
  private int capacity;
  private int head = 0;
  private int tail = 0;
 
  public myqueue(int capacity) {
   this.capacity = capacity;
   this.head = 0;
   this.tail = 0;
   this.length = 0;
   this.queue = new t[capacity];
  }
 
  public void clear() {
   head = 0;
   tail = 0;
   length = 0;
  }
 
  public bool isempty() {
   return length == 0;
  }
 
  public bool isfull() {
   return length == capacity;
  }
 
  public int length() {
   return length;
  }
 
  public bool enqueue(t node) {
   if (!isfull()) {
    queue[tail] = node;
    tail = (++tail) % capacity;
    length++;
    return true;
   }
   return false;
  }
 
  public t dequeue() {
   t node = default(t);
   if (!isempty()) {
    node = queue[head];
    head = (++head) % capacity;
    length--;
   }
   return node;
  }
 
  public void traverse() {
   for (int i = head; i < length + head; i++) {
    console.writeline(queue[i % capacity]);
    console.writeline($"前面还有{i - head}个");
   }
  }
 
  public void dispose() {
   queue = null;
  }
 }

为了能够通用,所以用的是泛型来实现环形队列类。这里最重要的是进队(enqueue)和出队(dequeue)两个方法,进队或出队后头和尾的位置都要通过取模运算来获得,因为是环形队列嘛,你懂的。

1、简单类型队列

好了,测试下入队:

?
1
2
3
4
5
6
7
8
9
10
11
12
class program
 {
  static void main(string[] args) {
   myqueue<int> queue = new myqueue<int>(4);
   queue.enqueue(10);
   queue.enqueue(16);
   queue.enqueue(18);
   queue.enqueue(12);
   queue.traverse();
   console.read();
  }
 }

显示结果:

C#环形队列的实现方法详解

再测试下出队:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
class program
 {
  static void main(string[] args) {
   myqueue<int> queue = new myqueue<int>(4);
   queue.enqueue(10);
   queue.enqueue(16);
   queue.enqueue(18);
   queue.enqueue(12);
   queue.traverse();
 
   console.writeline("弹两个出去");
   queue.dequeue();
   queue.dequeue();
   console.writeline();
   queue.traverse();
   console.read();
  }
 }

运行结果:

C#环形队列的实现方法详解

2、复杂类型队列

之前也说了,这个队列类是用的泛型写的,对应于c++的模板了,那就意味着任何类型都可以使用这个队列类,来测试个自定义的类试试,如下先定义一个customer类:

?
1
2
3
4
5
6
7
8
9
10
11
12
public class customer
 {
  public string name { get; set; }
 
  public int age { get; set; }
 
  public void pringinfo() {
   console.writeline("姓名:" + name);
   console.writeline("年龄:" + age);
   console.writeline();
  }
 }

然后进行入队,如下:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
class program
 {
  static void main(string[] args) {
   myqueue<customer> queue = new myqueue<customer>(5);
   queue.enqueue(new customer() { name = "宋小二", age = 29 });
   queue.enqueue(new customer() { name = "陈小三", age = 28 });
   queue.enqueue(new customer() { name = "王小四", age = 26 });
   queue.enqueue(new customer() { name = "朱小五", age = 48 });
   for (int i = 0; i < queue.length(); i++) {
    queue[i].pringinfo();
   }
   console.read();
  }
 }

上面的代码 queue[i].pringinfo();是通过索引来实现,所以我们得在队列类中实现索引,添加如下代码到myqueue.cs类中,如下:

?
1
2
3
4
5
public t this[int index] {
 get {
  return queue[index];
 }
}

感觉用for循环来遍历还是不够好,想用foreach,那就给myqueue类加个遍历接口,如下:

C#环形队列的实现方法详解

然后实现这个接口,如下:

?
1
2
3
4
5
6
7
8
9
10
11
public ienumerator<t> getenumerator() {
   foreach(t node in queue) {
    if(node != null) {
     yield return node;
    }
   }
  }
 
  ienumerator ienumerable.getenumerator() {
   return getenumerator();
  }

这样遍历的地方就可以改成foreach了,如下:

C#环形队列的实现方法详解

执行结果:

C#环形队列的实现方法详解

总结:

编程的思想才是最重要的,无关语言。以上就是这篇文章的全部内容了,希望能对大家的学习或者工作带来一定的帮助,如果有疑问大家可以留言交流。

延伸 · 阅读

精彩推荐
  • C#SQLite在C#中的安装与操作技巧

    SQLite在C#中的安装与操作技巧

    SQLite,是一款轻型的数据库,用于本地的数据储存。其优点有很多,下面通过本文给大家介绍SQLite在C#中的安装与操作技巧,感兴趣的的朋友参考下吧...

    蓝曈魅11162022-01-20
  • C#深入理解C#的数组

    深入理解C#的数组

    本篇文章主要介绍了C#的数组,数组是一种数据结构,详细的介绍了数组的声明和访问等,有兴趣的可以了解一下。...

    佳园9492021-12-10
  • C#C#设计模式之Strategy策略模式解决007大破密码危机问题示例

    C#设计模式之Strategy策略模式解决007大破密码危机问题示例

    这篇文章主要介绍了C#设计模式之Strategy策略模式解决007大破密码危机问题,简单描述了策略模式的定义并结合加密解密算法实例分析了C#策略模式的具体使用...

    GhostRider10972022-01-21
  • C#三十分钟快速掌握C# 6.0知识点

    三十分钟快速掌握C# 6.0知识点

    这篇文章主要介绍了C# 6.0的相关知识点,文中介绍的非常详细,通过这篇文字可以让大家在三十分钟内快速的掌握C# 6.0,需要的朋友可以参考借鉴,下面来...

    雨夜潇湘8272021-12-28
  • C#VS2012 程序打包部署图文详解

    VS2012 程序打包部署图文详解

    VS2012虽然没有集成打包工具,但它为我们提供了下载的端口,需要我们手动安装一个插件InstallShield。网上有很多第三方的打包工具,但为什么偏要使用微软...

    张信秀7712021-12-15
  • C#C#微信公众号与订阅号接口开发示例代码

    C#微信公众号与订阅号接口开发示例代码

    这篇文章主要介绍了C#微信公众号与订阅号接口开发示例代码,结合实例形式简单分析了C#针对微信接口的调用与处理技巧,需要的朋友可以参考下...

    smartsmile20127762021-11-25
  • C#如何使用C#将Tensorflow训练的.pb文件用在生产环境详解

    如何使用C#将Tensorflow训练的.pb文件用在生产环境详解

    这篇文章主要给大家介绍了关于如何使用C#将Tensorflow训练的.pb文件用在生产环境的相关资料,文中通过示例代码介绍的非常详细,需要的朋友可以参考借鉴...

    bbird201811792022-03-05
  • C#利用C#实现网络爬虫

    利用C#实现网络爬虫

    这篇文章主要介绍了利用C#实现网络爬虫,完整的介绍了C#实现网络爬虫详细过程,感兴趣的小伙伴们可以参考一下...

    C#教程网11852021-11-16