1.定义

  

  栈:后进先出(LIFO-last in first out):最后插入的元素最先出来。

  队列:先进先出(FIFO-first in first out):最先插入的元素最先出来。

2.用数组实现栈和队列

实现栈:

  由于数组大小未知,如果每次插入元素都扩展一次数据(每次扩展都意味着构建一个新数组,然后把旧数组复制给新数组),那么性能消耗相当严重。

  这里使用贪心算法,数组每次被填满后,加入下一个元素时,把数组拓展成现有数组的两倍大小。

  每次移除元素时,检测数组空余空间有多少。当数组里的元素个数只有整个数组大小的四分之一时,数组减半。

  为什么不是当数组里的元素个数只有整个数组大小的二分之一时,数组减半?考虑以下情况:数组有4个元素,数组大小为4个元素空间。此时,加一个元素,数组拓展成8个空间;再减一个元素,数组缩小为4个空间;如此循环,性能消耗严重。

  具体代码(Java):

  

public ResizingArrayStackOfStrings()
{
s=new String[1];
int N = 0;
} pubilc void Push(String item)
{
//如果下一个加入元素超出数组容量,拓展数组
if(N == s.length) Resize(2 * s.length);
s[N++] = item;
} private void Resize(int capacity)
{
String[] copy = new String[capacity];
//将旧数组元素复制给新数组
for(int i=0; i<N; i++) copy[i] = s[i];
s = copy;
} public String Pop()
{
String item = s[--N];
s[N] = null;
//剩余元素只占数组四分之一空间时,数组减半
if(N>0 && N=s.length/4) Resize(s.length/2);
return item;
}

效果如下图:

实现队列

  与栈类似:

数组每次被填满后,加入下一个元素时,把数组拓展成现有数组的两倍大小。

  每次移除元素时,检测数组空余空间有多少。当数组里的元素个数只有整个数组大小的四分之一时,数组减半。

  不同之处在于:

由于是先进先出,移除是从队列的最前端开始的。所以当我们移除数个数据后,队列数据是存储在数组的中间部分的。令队列数据的尾端数据ID为Num,首端数据ID为HeadIndex,则Num - HeadIndex为队列数据元素个数。

当队列数据元素个数为整个数组空间的四分之一时,数组减半,且队列数据左移至数组最左端。即Num-=HeadIndex;HeadIndex=0;

  图中,HeadIndex=2;Num=5;

具体代码:

.h:

UCLASS()
class ALGORITHM_API AStackAndQueuesExerciseTwo : public AActor
{
GENERATED_BODY() public:
// Sets default values for this actor's properties
AStackAndQueuesExerciseTwo();
// Called every frame
virtual void Tick(float DeltaTime) override;
//输入
void Enqueue(int Input);
//重构数组(拓展或缩小)
void Resize(int Capacity);
//输出且移除
int Dequeue();
//队列里没元素了?
bool IsEmpty(); protected:
// Called when the game starts or when spawned
virtual void BeginPlay() override; public: private:
//记录数组中有多少个Int
int Num;
//队列数组
TArray<int> MyIntArray;
//记录下一个移除的数据ID
int HeadIndex;
}; .cpp: AStackAndQueuesExerciseTwo::AStackAndQueuesExerciseTwo()
{
// Set this actor to call Tick() every frame. You can turn this off to improve performance if you don't need it.
PrimaryActorTick.bCanEverTick = true;
//一开始数组没成员
Num = ;
HeadIndex = ;
//数组中有一个假元素
MyIntArray.Add();
} // Called when the game starts or when spawned
void AStackAndQueuesExerciseTwo::BeginPlay()
{
Super::BeginPlay();
//测试
Enqueue();
Enqueue();
Enqueue();
Enqueue();
Enqueue();
Dequeue();
Dequeue();
Dequeue();
//队列数组成员
for (int i = HeadIndex; i < Num; i++)
{
UKismetSystemLibrary::PrintString(this, "i: " + FString::FromInt(i) + " End: " + FString::FromInt(MyIntArray[i]));
}
//队列数组的容量
UKismetSystemLibrary::PrintString(this, "MyIntArray.Num(): " + FString::FromInt(MyIntArray.Num()));
} // Called every frame
void AStackAndQueuesExerciseTwo::Tick(float DeltaTime)
{
Super::Tick(DeltaTime); } void AStackAndQueuesExerciseTwo::Enqueue(int Input)
{
//如果队列数组已满,拓展数组
if (Num == MyIntArray.Num())
{
Resize( * MyIntArray.Num());
}
//拓展或者数组有空位时,添加元素
if (Num < MyIntArray.Num())
{
MyIntArray[Num] = Input;
}
Num++;
} void AStackAndQueuesExerciseTwo::Resize(const int Capacity)
{
//int a[] = new int[Capacity];
TArray<int> Copy;
//添加数个假元素填充数组
for (int i = ; i < Capacity; i++)
{
Copy.Add();
}
//将队列数组赋值给Copy数组,如果是缩小数组,则把队列数组左移,节省空间
for (int i = HeadIndex; i < Num; i++)
{
Copy[i - HeadIndex] = MyIntArray[i];
}
MyIntArray = Copy;
} int AStackAndQueuesExerciseTwo::Dequeue()
{
//判断数组是否为空
if (IsEmpty())
{
UKismetSystemLibrary::PrintString(this, "No Element Exist!!!");
return ;
}
else
{
UKismetSystemLibrary::PrintString(this, "Dequeue: " + FString::FromInt(MyIntArray[HeadIndex]));
}
HeadIndex++;
//如果移除元素后,所剩元素为数组空间的四分之一,则数组减半
if ((Num - HeadIndex) != && (Num - HeadIndex) == (MyIntArray.Num() / ))
{
Resize(MyIntArray.Num() / );
//移除空间后,队列数组左移,节省空间
Num -= HeadIndex;
HeadIndex = ;
return MyIntArray[HeadIndex];
}
else
{
return MyIntArray[HeadIndex - ];
} }
//如果下一个要移除的数据不存在,则为空数组
bool AStackAndQueuesExerciseTwo::IsEmpty()
{
return HeadIndex >= Num;
}

栈与队列(Stack and Queue)的更多相关文章

  1. Coursera Algorithms week2 栈和队列 练习测验: Queue with two stacks

    题目原文: Implement a queue with two stacks so that each queue operations takes a constant amortized num ...

  2. LeetCode 232题用栈实现队列(Implement Queue using Stacks) Java语言求解

    题目链接 https://leetcode-cn.com/problems/implement-queue-using-stacks/ 题目描述 使用栈实现队列的下列操作: push(x) -- 将一 ...

  3. C#LeetCode刷题之#232-用栈实现队列​​​​​​​​​​​​​​(Implement Queue using Stacks)

    问题 该文章的最新版本已迁移至个人博客[比特飞],单击链接 https://www.byteflying.com/archives/4108 访问. 使用栈实现队列的下列操作: push(x) -- ...

  4. LeetCode 232. 用栈实现队列(Implement Queue using Stacks) 4

    232. 用栈实现队列 232. Implement Queue using Stacks 题目描述 使用栈实现队列的下列操作: push(x) -- 将一个元素放入队列的尾部. pop() -- 从 ...

  5. Leedcode算法专题训练(栈和队列)

    1. 用栈实现队列 232. Implement Queue using Stacks (Easy) Leetcode / 力扣 class MyQueue { Stack<Integer> ...

  6. lintcode :implement queue by two stacks 用栈实现队列

    题目 用栈实现队列 正如标题所述,你需要使用两个栈来实现队列的一些操作. 队列应支持push(element),pop() 和 top(),其中pop是弹出队列中的第一个(最前面的)元素. pop和t ...

  7. [Swift]LeetCode232. 用栈实现队列 | Implement Queue using Stacks

    Implement the following operations of a queue using stacks. push(x) -- Push element x to the back of ...

  8. LeetCode 232:用栈实现队列 Implement Queue using Stacks

    题目: 使用栈实现队列的下列操作: push(x) -- 将一个元素放入队列的尾部. pop() -- 从队列首部移除元素. peek() -- 返回队列首部的元素. empty() -- 返回队列是 ...

  9. LeetCode#232-Implement Queue using Stacks-用栈实现队列

    一.题目 使用栈实现队列的下列操作: push(x) -- 将一个元素放入队列的尾部. pop() -- 从队列首部移除元素. peek() -- 返回队列首部的元素. empty() -- 返回队列 ...

随机推荐

  1. linux ssh root登陆出现错误:Permission denied, please try again

    密码已检测过多遍还是登录失败 经检查 vim /etc/ssh/sshd_config PermitRootLogin no 改成 PermitRootLogin yes 修改之后重启就可以了

  2. react router @4 和 vue路由 详解(五)react怎么通过路由传参

    完整版:https://www.cnblogs.com/yangyangxxb/p/10066650.html 7.react怎么通过路由传参? a.通配符传参(刷新页面数据不丢失) //在定义路由的 ...

  3. 修改Host,配置域名访问

    修改Host,配置域名访问   虽然我们已经能够通过localhost访问本地网站了,为了提高逼格,我们可以修改host文件,设置一个自己喜欢的域名指向本地网站,岂不是更高大上. 明确需求 通过配置, ...

  4. docker 将正在运行的容器打包为镜像

    将容器打包成镜像 docker commit -a "runoob.com" -m "my apache" 容器名称或id 打包的镜像名称:标签 OPTIONS ...

  5. 数据库-->表操作

    一.MySQL存储引擎 # InnoDB MySql 5.6 版本默认的存储引擎.InnoDB 是一个事务安全的存储引擎,它具备提交.回滚以及崩溃恢复的功能以保护用户数据.InnoDB 的行级别锁定以 ...

  6. C++输出数组名

    1.如果C++输出的数组是char类型的,那么输出的就是数组中的元素. 2.如果使用的是其他类型的数组作为输出的话,那么就是一个16进制的地址. 3.还是那句话,对数组的操作,很多时候都是指针的操作, ...

  7. Cracking The Coding Interview 1.6

    //原文: // // Given an image represented by an NxN matrix, where each pixel in the image is 4 bytes, w ...

  8. Linux学习 :移植linux-3.4.83到JZ2440开发板

    一.编译环境搭建: 1.linux源码下载:https://www.kernel.org/ (最新)  https://mirrors.edge.kernel.org/pub/linux/kernel ...

  9. 数控AGC实现(转)

    相关链接:    一种混合式高动态范围AGC算法与FPGA实现     http://www.sohu.com/a/221438387_781333 基于FPGA的快速自动增益控制系统设计      ...

  10. python笔记7-if中的is ;in ;not搭配用法

    names="111 222 333" print("111" in names)#返回的是True,用in返回的是布尔值in在里面 print("1 ...