前面楼主简单介绍了JavaScript数据结构栈的实现,http://www.cnblogs.com/qq503665965/p/6537894.html,本次将介绍队列的实现。
队列是一种特殊的线性表,特殊之处在于它只允许在表的前端(front)进行删除操作,而在表的后端(rear)进行插入操作,和栈一样,队列是一种操作受限制的线性表。进行插入操作的端称为队尾,进行删除操作的端称为队头。
队列的两种主要操作是:向队列中插入新元素和删除队列中的元素。插入操作也叫做入队,删除操作也叫做出队。入队操作在队尾插入新元素,出队操作删除队头的元素。下图演示了这两个操作。

队列的另外一项重要操作是读取队头的元素。这个操作叫做 peek() 。该操作返回队头元素,但不把它从队列中删除。除了读取队头元素,我们还想知道队列中存储了多少元素,可以使用 length 属性满足该需求;要想清空队列中的所有元素,可以使用 clear() 方法来实现。下表定义了队列的一些主要方法:
| dataStorage | Array | 存储数据的底层数据结构 |
| enqueue | int | 入队 |
| dequeue | fucntion | 出队 |
| front | fucntion | 返回队首元素 |
| back | fucntion | 返回队尾元素 |
| toString | fucntion | 显示队列内的所有元素 |
| empty | function | 判断队列是否为空 |
队列的实现
1 function Queue(array) {
2 if (array instanceof Array) {
3 this.dataStorage = array;
4 } else {
5 throw "'"+array + "' is not an Array";
6 }
7 }
8
9 Queue.prototype = {
10 enqueue: function (element) {//向队列中添加元素
11 this.dataStorage.push(element);
12 },
13 dequeue: function () {//删除队首元素
14 return this.dataStorage.shift();
15 },
16 front: function () {//返回队首元素
17 return this.dataStorage[0];
18 },
19 back: function () {//返回队尾元素
20 return this.dataStorage[this.dataStorage.length - 1];
21 },
22 toString: function () {//输出队列中所有元素
23 var result = "";
24 for (var i = 0; i < this.dataStorage.length; i++) {
25 result += this.dataStorage[i] + " ";
26 }
27 return result;
28 },
29 empty: function () {//判定队列是否为空
30 if (this.dataStorage.length == 0) {
31 return true;
32 }
33 return false;
34 }
35 }
测试:
1 var queue = new Queue(['a','b']); 2 console.log(queue.toString()); 3 //queue.dequeue();//a出队 4 //console.log(queue.toString()); 5 console.log(queue.front());//队头元素 6 console.log(queue.back());//队尾元素 7 queue.dequeue(); 8 queue.dequeue(); 9 console.log(queue.empty());
队列的应用——基数排序
原理:对于 0~99 的数字,基数排序将数据集扫描两次。第一次按个位上的数字进行排序,第二次按十位上的数字进行排序。每个数字根据对应位上的数值被分在不同的盒子里。
假设有这样一个数组:{4, 73, 51, 76, 70, 19, 63, 64, 43, 12},经过基数排序第一次扫描之后,数字被分配到如下盒子中:
1 Box 0: 70 2 Box 1: 51 3 Box 2: 12 4 Box 3: 73,63,43 5 Box 4: 4,64 6 Box 5: 76 7 Box 6: 8 Box 7: 9 Box 8: 10 Box 9: 19
根据盒子的顺序,对数字进行第一次排序的结果如下:
70,51,12,73,63,43,4,64,76,19
然后根据十位上的数值再将上次排序的结果分配到不同的盒子中:
1 Box 0: 4 2 Box 1: 12,19 3 Box 2: 4 Box 3: 5 Box 4: 43 6 Box 5: 51 7 Box 6: 63,64 8 Box 7: 70,73,76 9 Box 8: 10 Box 9:
最后,将盒子中的数字取出,组成一个新的列表,该列表即为排好序的数字:
4,12,19,43,51,63,64,70,73,76
使用队列代表盒子,可以实现这个算法。我们需要九个队列,每个对应一个数字。将所有队列保存在一个数组中,使用取余和除法操作决定个位和十位。算法的剩余部分将数字加入相应的队列,根据个位数值对其重新排序,然后再根据十位上的数值进行排序,结果即为排好序的数字。
下面是根据相应位(个位或十位)上的数值,将数字分配到相应队列的方法:
1 function allocation(nums,queues,n,digit) {
2 for (var i = 0; i < n; ++i) {
3 if (digit == 1) {
4 queues[nums[i] % 10].enqueue(nums[i]);
5 }
6 else {
7 queues[Math.floor(nums[i] / 10)].enqueue(nums[i]);
8 }
9 }
10 }
下面是从队列中收集数字的方法:
1 function collection(queues, nums) {
2 var i = 0;
3 for (var digit = 0; digit < 10; ++digit) {
4 while (!queues[digit].empty()) {
5 nums[i++] = queues[digit].dequeue();
6 }
7 }
8 }
测试程序:
1 var queues = [];
2 for (var i = 0; i < 10; ++i) {
3 queues[i] = new Queue([]);
4 }
5 var nums = [];
6 for (var i = 0; i < 10; ++i) {//目标排序数组
7 nums[i] = Math.floor(Math.floor(Math.random() * 101));
8 }
9 console.log("排序前:");
10 showArray(nums);
11 allocation(nums, queues, 10, 1);
12 collection(queues, nums);
13 allocation(nums,queues,10,10);
14 collection(queues, nums);
15 console.log("排序后");
16 showArray(nums);
17
18 function showArray(array) {
19 console.log(array);
20 }
结果:

下篇楼主将带来,链表的实现。
本文的示例代码地址:https://github.com/LJunChina/JavaScript
