数据结构浅析:队列

数据结构浅析:队列

图片来源:marketwatch.com
图片来源:marketwatch.com

黑色星期五快到了,新版的 Microsoft Surface Studio 已经上架(我是 Windows 的忠实粉丝)。我们一起来讨论每一个人最喜欢的购物经历:排队等待。以及一个比较老的数据结构,队列。

队列

队列就是一条线

添加(enqueues)总是添加到线的尾部

移除(dequeues)总是从线的头部开始移除

队列遵循先进先出(FIFO)模式。

使用范例

  • 解决同时有多个用户请求服务,比如有3个人几乎同时购买最后一张机票
  • 广度优先搜索算法中数据的排列

为了解决第一种使用场景,我们帮助微软创建一个队列数据结构来管理所有的新版 Surface Studio 订购请求。

在开始之前,快速的说明一下 JavaScript 的数组。和栈一样,JavaScript 的数组本身就有队列的功能。

怎么用 JavaScript 的数组来表现队列

Enqueue:向数组的尾部添加:

<pre>

Array.push(someVal)

</pre>

Dequeue:移除数组的第一个元素并将它返回:

<pre>

Array.shift()

</pre>

如果由于你感到叛逆的原因,你也可以添加到数组的头部,然后从尾部开始移除。

Enqueue 向数组的头部添加数据:

<pre>

Array.unshift(someVal)

</pre>

Dequeue 从数组的尾部开始移除数据:

<pre>

Array.pop()

</pre>

也就是说,为了理解透彻,你需要使用 JavaScript 对象重新实现一个队列。

所以你需要为微软做的第一件事情就是创建一个真实的队列,用于存储在他们网站上点击了订购按钮的每一个顾客。

class Queue{
  constructor(){
    this._storage = {};
    this._start = -1; //replicating 0 index used for arrays
    this._end = -1; //replicating 0 index used for arrays
  }
  
  size(){
   return this._end - this._start;
  }
}
let appleQueue = new Queue();

变量名称前面的下划线仅仅是用于说明这是一个私有变量,不可直接访问。

栈数据结构不一样(它的添加和移除发生在同一端),队列的本质要求我们同时管理其两端。因此,你需要创建 start 变量用于跟踪队列的头,而 end 变量用于跟踪队列的尾。

最后,获取队列大小最简单的方式(不用创建一个不必要的计数器变量)就是跟踪 start 和 end 之间的差。

首先,你应该提供一种方式来将点击了订购按钮的人员添加到队列。你可以通过 enqueue 方法实现:

class Queue{
  constructor(){
    this._storage = {};
    this._start = -1; //replicating 0 index used for arrays
    this._end = -1; //replicating 0 index used for arrays
  }
  
  enqueue(val){
    this._storage[++this._end] = val; 
     
    //++this._end just means increment the end variable first
    //It's equivalent to
    //this._end++   //->
    //this._storage[this._end] = val;
  }
  
  size(){
   return this._end - this._start;
  }
}
let microsoftQueue = new Queue();
microsoftQueue.enqueue("{user: ILoveWindows@gmail.com}")
microsoftQueue.enqueue("{user: cortanaIsMyBestFriend@hotmail.com}")
microsoftQueue.enqueue("{user: InternetExplorer8Fan@outlook.com}")
microsoftQueue.enqueue("{user: IThrowApplesOutMyWindow@yahoo.com}")

很棒!现在你的 microsoftQueue 中存储的内容看起来应该是这样:

{
  0: "{email: ILoveWindows@gmail.com}"
  1: "{email: cortanaIsMyBestFriend@hotmail.com}"
  2: "{email: InternetExplorer8Fan@outlook.com}"
  3: "{email: IThrowApplesOutMyWindow@yahoo.com}"
}

简单的说明一下上面 users ({user: ...})的表现方式。

当用户在客户端点击订购按钮的时候,他们就会将所有相关的信息发送到处理请求的服务端。当系统之间交换数据时(比如客户端和服务端),最常用的方式就是通过 AjaxJSONJavaScript Object Notation)形式发送。

这和 JavaScript 对象很相似,因为它仅仅是一个字符串版本的键值对。如果你对 JavaScript 不熟悉,它和字典或者哈希表(我们在后续文章中介绍)很相似。有关 JSON 的更多信息,Andreas Grech 在 Stack Overflow 上有一篇很棒的文章可以参考。

现在回到我们讨论的队列。

得益于你创建的队列。微软现在可以按照交易的先后顺序高效地跟踪所有已经购买 Surface Studio 的顾客。为了保证所有的顾客是按照正确的顺序获得服务,你需要创建一个准确的 dequeue 方法来跟踪所有顾客的顺序,并且在提供服务之后将他们从队列中移除。

class Queue{
  constructor(){
    this._storage = {};
    this._start = -1; //replicating 0 index used for arrays
    this._end = -1; //replicating 0 index used for arrays
  }
  
  enqueue(val){
    this._storage[++this._end] = val; 
  }
  dequeue(){
    if(this._end > this._start){ //check if there are values
      let nextUp = this._storage[++this._start];
      delete this._storage[this._start];
      return nextUp;
    }
  }  
  
  size(){
   return this._end - this._start;
  }
}
let microsoftQueue = new Queue();
microsoftQueue.enqueue("{user: ILoveWindows@gmail.com}")
microsoftQueue.enqueue("{user: cortanaIsMyBestFriend@hotmail.com}")
microsoftQueue.enqueue("{user: InternetExplorer8Fan@outlook.com}")
microsoftQueue.enqueue("{user: IThrowApplesOutMyWindow@yahoo.com}")
//Function to send everyone their Surface Studio!
let sendSurface = recepient => {
   sendTo(recepient);
}
//When your server is ready to handle this queue, execute this:
while(microsoftQueue.size() > 0){
  sendSurface(microsoftQueue.dequeue());
}

好了,现在 microsoftQueue 中等待的每一位都得到了他们的新 Surface Studio。

还有一些快速优化可以使代码工作更合理。

  • 1、一旦队列中每一个人都已经获得了服务,你可以重置 start 和 end。你的队列不太可能达到 JavaScript 的最大数,但是为了安全起见还是需要重置。
  • 2、由于JavaScript 中 0 被当做 “false” 处理(更多信息可以参考这里),你可以将 “end > start 检查” 用 size 方法替换
dequeue(){
    if(this.size()){ //0 is a falsey value...coerced to return false
      let nextUp = this._storage[++this._start];
      delete this._storage[this._start];
      if(!this.size()){ //Recheck after incrementing (!0 == true)
        this._start = -1;
        this._end = -1; 
      }
      
      return nextUp;
    }
}

就这么多,你已经完成了一个基本的队列实现!

队列方法的时间复杂度分析

再来看一下代码

class Queue{
  constructor(){
    this._storage = {};
    this._start = -1; //replicating 0 index used for arrays
    this._end = -1; //replicating 0 index used for arrays
  }
  
  enqueue(val){
    this._storage[++this._end] = val; 
  }
  dequeue(){
    if(this.size()){ /
      let nextUp = this._storage[++this._start];
      delete this._storage[this._start];
      if(!this.size()){ 
        this._start = -1;
        this._end = -1; 
      }
      
      return nextUp;
    }
  }
  
  size(){
   return this._end - this._start;
  }
}

栈分析时使用的逻辑在这里同样适用:

Enqueue(添加)是O(1)。因为你在总是知道队列的尾部在哪里,你不必在遍历所有元素之后再添加。

Dequeue(删除)是O(1)。同样因为你总是知道头部的位置,所以移除元素时也不需要遍历。

SizeO(1)。因为有 start 和 end 两个变量,所以队列的大小总是已知的。

这里有一点非常重要的需要说明:队列并非意味着无限大小,尽管我们实现的 queue 类 和 JavaScript 的数组在系统内存耗尽前允许你持续添加元素。

一种优化方式是通过使用限制空间的数组来创建循环队列。Damian Gordon 在 Youtube 上提供了非常有用的视频讲解。这对于我们在后续文章中分析哈希表时也非常得有用!

总结

队列:

1、遵循先进先出(FIFO)模式

2、有一个 start 和 end 属性,分别用于跟踪队列的头部和尾部

3、有 enqueue(add) 和dequeue(remove)方法,用于管理队列内容

4、有 size 属性用于跟踪队列的大小

挑战

尝试仅仅使用栈来重新实现队列,小提示,只需要两个栈就可。

本文译自:A Gentle Introduction to Data Structures: How Queues Work

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 211,042评论 6 490
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 89,996评论 2 384
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 156,674评论 0 345
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,340评论 1 283
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,404评论 5 384
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,749评论 1 289
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,902评论 3 405
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,662评论 0 266
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,110评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,451评论 2 325
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,577评论 1 340
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,258评论 4 328
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,848评论 3 312
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,726评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,952评论 1 264
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,271评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,452评论 2 348

推荐阅读更多精彩内容