1.进栈:将元素从栈顶插入栈中
2.出栈:将栈顶的元素弹出栈
3.栈空:判断栈里是否存在元素
4.栈顶:返回栈顶的元素
栈解决表达式求值的问题
更多的应用
·栈的主要特点就是“先进后出”因此又称为先进后出表 栈的图示:栈的示意图
这里我们将直接跳过栈的顺序存储结构,主要记录栈的链式存储结构;
//typedef Elemtype 要定义的类型;定义Elemtype的类型减少修改时的代码量 这里是帮助理解 结构体中的 Elemtype
typedef struct stack
{
Elemtype data; //定义栈内元素类型
struct stack *next; //指针域
}LinStack; //链栈结点类型
因为栈是在栈顶插入元素,并且最后输出元素的顺序和输入的顺序正好相反 所以我们在对栈进行插入操作时只能够使用头插法来push我们待插入的元素
void InitStack(LinStack* &s)
{
s = new LinStack; //(LinkStack*)malloc(sizeof(LinStack));
s -> next = NULL; //将指针指向空,表示此时站内没有元素
}
我们这里统一使用 new来代替malloc分配内存
void DestroyStack(LinStack* &s)
{
LinStack *pre = s, *p = s -> next; //pre指向头结点,p指向首结点
while( p != NULL )
{
free (pre); //释放pre的空间
pre = p; //p和pre指针一起后移
p = pre -> next;
}
free (pre); //删除尾结点
}
bool StackEmpty(LinStack* s)
{
return s -> next == NULL;
}
定义成bool变量 如果指向NULL则表明栈是空的返回 true 否则返回 false 表示栈中存在元素
void Push(LinStack* &s,Elemtype e)
{
LinStack *p; //新建结点p
s = new LinStack;
p -> data = e; //将元素e存入栈中
p -> next = s -> next;
s -> next = p;
}
void Pop(LinStack* &s,Elemtype e)
{
LinStack *p;
if(s -> next == NULL)
return false; //如果栈顶没有元素那么返回false 表示栈空无法弹出
p = s -> next;
e = p -> data;
s -> next = p -> next;
delete p;
return true;
}
如果我们在删除的时候不需要栈顶的元素 那么可以这样写
void Pop(LinStack* &s,Elemtype e)
{
LinStack *p;
if(s -> next == NULL)
return false; //如果栈顶没有元素那么返回false 表示栈空无法弹出
p = s -> next;
s -> next = p -> next;
delete p;
return true;
}
- 取栈顶元素
bool GetTop(LinStack* s,Elemtype e)
{
if(s -> next == NULL)
return false;
e = s -> next -> data; //由于设置了头结点,所以我们要返回首结点的值
return true;
以上就是 C语言实现栈的操作的代码了;
既然标签加了C++;那么我就写一下C++中STL的stack来实现一下栈的基本操作吧
STL中< stack >实现栈的操作
//头文件
include<iostream>
include<stack>
using namespace std;
int main()
{
satck<int> s; //定义一个 元素为int的栈s
s.pop(); //弹出栈顶元素
s.push(); //向栈顶插入元素
s.top(); //返回栈顶元素
s.size(); //返回栈中的元素个数
s.empty(); //判断栈是否为空
s.clear(); //清空栈内的元素
}
除此之外:栈还可以帮助我们解决很多的问题 如 表达式求值 和 括号匹配等等经典问题
给定一个表达式,其中运算符仅包含 +,-,*,/(加 减 乘 整除),可能包含括号,请你求出表达式的最终值。
注意:
共一行,为给定表达式。
共一行,为表达式的结果。
表达式的长度不超过105
( 2 + 2 ) * ( 1 + 1 )
8
代码如下
#include
#include
#include
using namespace std;
stack<int> num;
stack<char> op;
void eval() // 从num弹出两个数,从op弹出一个操作符,运算结束再压入num
{
int b = num.top(); num.pop();
int a = num.top(); num.pop();
char c = op.top(); op.pop();
int x;
if (c == '+') x = a + b;
else if (c == '-') x = a - b;
else if (c == '*') x = a * b;
else x = a / b;
num.push(x);
}
int main()
{
// 初始化:优先级,string读取
map<char, int> pr = { {'+', 1}, {'-', 1}, {'*', 2}, {'/', 2}};
string str;
cin >> str;
for (int i = 0; i < str.size(); i ++ )
{
char c = str[i];
if (isdigit(c)) // 是数字的时候,取出对应数字压入栈中
{
int x = 0, j = i;
while (j < str.size() && isdigit(str[j])) x = x * 10 + str[j ++ ] - '0';
i = j - 1;
num.push(x);
}
else if (c == '(') op.push(c); // 左括号压入栈中
else if (c == ')') // 右括号依次弹出并运算直到左括号,再弹出
{
while (op.top() != '(') eval();
op.pop();
}
else // 一般运算符:判栈顶如果优先级>=当前符号,依次弹出并运算,再压入
{
while (op.size() && op.top() != '(' && pr[op.top()] >= pr[c]) eval();
op.push(c);
}
}
while (op.size()) eval(); // 最后一定要依次弹出并运算
cout << num.top();
return 0;
感谢你的阅读