数据结构—栈(Stack)
- 栈的定义--Stack
栈是只允许在末端进行插入和删除的线性表。栈具有后进先出的特性(LIFO ,Last In Fast Out)。

学过数据结构的人都知道:栈可以用两种方式来实现,一种方法是用数组实现栈,这种栈成为静态栈;另外一种方法是用链表实现栈,这种栈叫做动态栈。
- 栈提供如下操作
s.empty() 如果栈为空返回true,否则返回false
s.size() 返回栈中元素的个数
s.pop() 删除栈顶元素但不返回其值
s.top() 返回栈顶的元素,但不删除该元素
s.push() 在栈顶压入新元素
- 栈的实现
下面是用C++实现的一个栈结构的源码(顺序表)
#pragma once
#include<iostream>
#include<assert.h>
using namespace std;
template<typename T>
class Stack
{
public:
Stack()
:array(NULL)
, size()
, capacity()
{}
~Stack()
{
if (array != NULL)
{
delete[] array;
}
capacity = ;
size = ;
}
Stack(const Stack<T>& s)
{
array = new T[s.size];
memcpy(array, s.array, sizeof(T)*s.size);
size =s.size;
capacity = s.capacity;
}
Stack<T>& operator=( const Stack<T>& s)
{
capacity = s.capacity;
size = s.size;
this->array =s.array;
return *this;
}
void Print()
{
for (int index = ; index < size; index++)
{
cout << array[index] << " ";
}
cout << endl;
}
void Push(const T& x)
{
_CheckCapacity(); array[size++] = x;
} void Pop()
{
assert(size > );
--size;
} size_t Size()
{
return size;
} bool Empty()
{
return size == ;
} const T& Top()
{
assert(size > ); return array[size - ];
}
protected:
void _CheckCapacity()
{
if (size >= capacity)
{
T *temp = new T[capacity * + ];
for (int index = ; index < size; index++)
{
temp[index] =array[index];
}
delete[] array;
array = temp;
capacity = capacity * + ;
} } protected:
T * array;
size_t size;
size_t capacity;
};
void fun()
{
Stack<int> s;
s.Push();
s.Push();
s.Push();
s.Push();
s.Print();
cout << s.Top() << " ";
s.Pop();
}
void main()
{
fun();
system("pause");
}
- 栈的应用
1.算术表达式求值。波兰表达式(后缀表达式) 
实现:
#include <iostream>
#include <assert.h>
#include"stack"
using namespace std; enum Type
{
OP_NUM,
OP_SYMBOL
}; enum OP_SMB
{
ADD,
SUB,
MUL,
DIV,
}; struct Cell
{
Type _type;
int _value;
}; Cell RNPArray[] =
{
{ OP_NUM, },
{ OP_NUM, },
{ OP_NUM, },
{ OP_SYMBOL, ADD },
{ OP_SYMBOL, MUL },
{ OP_NUM, },
{ OP_SYMBOL, SUB },
{ OP_NUM, },
{ OP_NUM, },
{ OP_SYMBOL, DIV },
{ OP_SYMBOL, ADD },
}; int CountRNP(Cell* a, size_t size)
{
stack<int> s;
for (int i = ; i < size; ++i)
{
if (a[i]._type == OP_NUM)
{
s.push(a[i]._value);
}
else if (a[i]._type == OP_SYMBOL)
{
int right = s.top();
s.pop();
int left = s.top();
s.pop(); switch (a[i]._value)
{
case ADD:
s.push(left + right);
break;
case SUB:
s.push(left - right);
break;
case MUL:
s.push(left * right);
break;
case DIV:
s.push(left / right);
break;
}
}
}
return s.top();
}
void test()
{ int ret=CountRNP(RNPArray, sizeof(RNPArray) / sizeof(RNPArray[]));
cout << ret;
}
int main()
{
test();
return ;
}
2.迷宫问题

#define _CRT_SECURE_NO_WARNINGS 1
#include<iostream>
#include<stack>
#include<assert.h>
using namespace std;
//从文件获取初始迷宫地图
void GetMazeMap(int *maze, int rows, int cols)
{
assert(maze);
FILE *fOut = fopen("MazeMap.txt", "r");
assert(fOut);
for (int i = ; i < rows; i++)
{
for (int j = ; j < cols;)
{
char ch = fgetc(fOut);
if (ch == '' || ch == '')
{
maze[i*cols + j] = ch - '';
j++;
}
}
}
fclose(fOut);
}
//打印迷宫地图
void PrintMazeMap(int*maze, int rows, int cols)
{
assert(maze);
for (int i = ; i < rows; i++)
{
for (int j = ; j < cols; j++)
{
cout << maze[i*cols + j] << " ";
}
cout << endl;
}
cout << endl;
}
struct Point
{
size_t row;
size_t col;
};
//检查该点是否为可通行
bool CheckIsAccess(int *maze, int rows, int cols,Point cur)
{
assert(maze);
if (cur.row < rows&&cur.col < cols&& (maze[cur.row*cols + cur.col] == ))
{
return true;
}
return false;
}
//获取走出迷宫路径
stack<Point> GetMazePath(int* maze, int rows, int cols, Point entry)
{
assert(maze);
stack<Point> path;
Point cur = entry;
path.push(entry);
maze[cur.row*cols + cur.col] = ; while (!path.empty())
{
Point cur = path.top();
Point next = cur;
if (next.row ==rows-)
{
return path;
}
//试探 上
next = cur;
next.row--;
if (CheckIsAccess(maze, , , next))
{
path.push(next);
maze[next.row*cols + next.col] = ;
continue;
}
//下
next = cur;
next.row++;
if (CheckIsAccess(maze, , , next))
{
path.push(next);
maze[next.row*cols + next.col] = ;
continue;
}
//左
next = cur;
next.col--;
if (CheckIsAccess(maze, , , next))
{
path.push(next);
maze[next.row*cols + next.col] = ;
continue;
}
//右
next = cur;
next.col++;
if (CheckIsAccess(maze, , , next))
{
path.push(next);
maze[next.row*cols + next.col] = ;
continue;
}
}
cout << "Not have exit" << endl;
return path;
}
void TestMaze()
{
int maze[][] = {}; GetMazeMap((int *)maze, , );
PrintMazeMap((int*)maze, , );
Point entry = { , };
GetMazePath((int*)maze, , , entry);
PrintMazeMap((int*)maze, , );
}
int main()
{
TestMaze();
//system("pause");
getchar();
return ;
}
数据结构—栈(Stack)的更多相关文章
- C# 数据结构 栈 Stack
栈和队列是非常重要的两种数据结构,栈和队列也是线性结构,线性表.栈和队列这三种数据结构的数据元素和元素的逻辑关系也相同 差别在于:线性表的操作不受限制,栈和队列操作受限制(遵循一定的原则),因此栈和队 ...
- java数据结构 栈stack
栈(Stack) 栈(Stack)实现了一个后进先出(LIFO)的数据结构. 你可以把栈理解为对象的垂直分布的栈,当你添加一个新元素时,就将新元素放在其他元素的顶部. 当你从栈中取元素的时候,就从栈顶 ...
- 模板 - 数据结构 - 栈/Stack
普通的栈大家都会写,STL的栈据说默认实现方式是deque,没关系反正deque跑得飞快. 这里收录的是一些奇怪的栈,当然双栈实现的队列收录在队列里面. 对顶栈 众所周知,栈可以维护一系列前缀和,包括 ...
- 数据结构----栈stack
栈的概念与数据结构 栈(有时称为“后进先出栈”)是一个元素的有序集合,其中添加移除新元素总发生在同一端.这一端通常称为“顶部”.与顶部对应的端称为“底部”.栈的底部很重要,因为在栈中靠近底部的元素是存 ...
- [C++][数据结构]栈(stack)的实现
对于栈的定义,前人之述备矣. 我实现的是一个stack<value>容器类,支持push,pop,top,size,empty,clear和copy construction操作. 主要的 ...
- python基本数据结构栈stack和队列queue
1,栈,后进先出,多用于反转 Python里面实现栈,就是把list包装成一个类,再添加一些方法作为栈的基本操作. 栈的实现: class Stack(object): #初始化栈为空列表 def _ ...
- 【Java数据结构学习笔记之二】Java数据结构与算法之栈(Stack)实现
本篇是java数据结构与算法的第2篇,从本篇开始我们将来了解栈的设计与实现,以下是本篇的相关知识点: 栈的抽象数据类型 顺序栈的设计与实现 链式栈的设计与实现 栈的应用 栈的抽象数据类型 栈是 ...
- Python与数据结构[1] -> 栈/Stack[0] -> 链表栈与数组栈的 Python 实现
栈 / Stack 目录 链表栈 数组栈 栈是一种基本的线性数据结构(先入后出FILO),在 C 语言中有链表和数组两种实现方式,下面用 Python 对这两种栈进行实现. 1 链表栈 链表栈是以单链 ...
- java数据结构——栈(Stack)
学习数据结构与算法是枯燥的,但只有坚持不懈的积累,才会有硕果累累的明天. /** * 继续学习Java数据结构 ————栈 * 栈的实现其实还是使用数组,只不过我们不能直接访问数组下标,而是通过一个指 ...
随机推荐
- Linux系统文件和目录的属性及权限
1 文件属性概述 Linux系统中的文件或目录的属性主要包括:索引节点(inode).文件类型.权限属性.硬链接数.所归属的用户和用户组.最近修改时间等内容(文件名严格来说不属于文件的属性): 下面是 ...
- Linux环境下tomcat如何热部署
1.修改tomcat配置文件 1.1第一步修改tomcat-users.xml <role rolename="manager-gui" /> <role rol ...
- 揭开js之constructor属性的神秘面纱
揭开 constructor 在 Javascript 语言中,constructor 属性是专门为 function 而设计的,它存在于每一个 function 的prototype 属性中.这个 ...
- jquery添加html代码的几种方法
经常用jq来DOM添加html代码 就总结了jq里面最常用的动态添加html代码的方法 append在元素内部的尾部加上元素 prepend在元素内部的前部加上元素 after在元素外部的尾部加上元素 ...
- git克隆出错 github clone Permission denied (publickey) fatal Could not read from remote repo
原文网址:http://blog.csdn.net/feeling450/article/details/53067563 github clone "Permission denied ( ...
- linux实现DNS轮询实现负载平衡
DNS 轮询机制会受到多方面的影响,如:A记录的TTL时间长短的影响:别的 DNS 服务器 Cache 的影响:windows 客户端也有一个DNS Cache.这些都会影响 DNS 轮询的效果.因此 ...
- 一笔画问题 南阳acm42(貌似没用到什么算法)
一笔画问题 时间限制:3000 ms | 内存限制:65535 KB 难度:4 描述 zyc从小就比较喜欢玩一些小游戏,其中就包括画一笔画,他想请你帮他写一个程序,判断一个图是否能够用一笔画下 ...
- C++ 指针初始化要注意的地方
1. 声明多个指针的时候: int* P1,P2; 如上所示,声明的是创建一个指针P1和一个int型的变量P2.而不是声明的两个指针. 对每个指针变量名,都需要使用一个*. 在C++中,int* 是一 ...
- delphi 数据库中Connection与Query连接数量问题思考
今天闲着没事,测试了一下Connection连接MSSQL,可以承受多少连接. 1.看看ADOConnection的连接数:写了一个代码,动态创建,测试了10000个连接,花了大约5~10分钟创 ...
- 获取单片机唯一id(stm32获取单片机唯一id)
stm32唯一id: 不同型号的stm32单片机,id不在同一地址上!具体地址可以通过用户手册中的Device electronic signature>Unique device ID reg ...