《数据结构与算法分析》学习笔记(四)——栈ADT
一、栈ADT是what?
1、定义
栈,是限制插入和删除都只能在一个位置上进行的表。
2、图示
3、栈的基本功能
(1)是否为空
(2)进栈
(3)出栈
(4)清空
(5)取栈顶
二、栈的链表实现
#ifndef Exercise_Stack_h
#define Exercise_Stack_h typedef struct Node *PrtToNode;
typedef PrtToNode Stack;
typedef int ElementType; struct Node
{
ElementType Element;
PrtToNode Next;
}; bool IsEmpty(Stack S); Stack CreateStack(); void MakeEmpty(Stack S); void Push(ElementType X,Stack S); void Pop(Stack S); ElementType Top(Stack S); #endif bool IsEmpty(Stack S)
{
return S->Next==nullptr;
} Stack CreateStack()
{
Stack S; S = new struct Node;
if(S==nullptr)
{
std::cout<<"Create Error!"<<std::endl;
return nullptr;
} S->Next=nullptr; return S;
} void MakeEmpty(Stack S)
{
if(S==nullptr)
{
std::cout<<"not initStack Error!"<<std::endl;
return ;
} while (!IsEmpty(S))
{
Pop(S);
} } void Push(ElementType X,Stack S)
{
PrtToNode Temp; Temp = new struct Node;
if(Temp==nullptr)
{
std::cout<<"init Node Error!"<<std::endl;
return;
} Temp->Element=X;
Temp->Next=S;
S->Next=Temp;
} void Pop(Stack S)
{
if(IsEmpty(S))
{
std::cout<<"Stack Empty!"<<std::endl;
return;
}
auto temp = S->Next;
S->Next=S->Next->Next;
delete temp;
} ElementType Top(Stack S)
{
if(IsEmpty(S))
{
std::cout<<"Stack Empty!"<<std::endl;
return ;
}
return S->Next->Element;
}
PS:优点:不用担心栈溢出的现象
缺点:开辟内存和释放内存的时候貌似开销比较昂贵。
三、栈的数组实现
typedef int ElementType;
const int MaxCapacity = ; struct StackRecord
{
int Capacity;
int TopOfStack;
ElementType *Array;
}; typedef struct StackRecord *Stack; bool IsFull(Stack S); bool IsEmpty(Stack S); Stack CreateStack(int MaxElements); void DisposeStack(Stack S); void MakeEmpty(Stack S); void Push(ElementType X,Stack S); void Pop(Stack S); ElementType Top(Stack S); ElementType TopandPop(Stack S); Stack CreateStack(int MaxElements)
{
Stack S; S = new struct StackRecord;
if(S==nullptr)
{
std::cout<<"Out of space!"<<std::endl;
return nullptr;
} S->Array = new ElementType(MaxElements);
if(S->Array==nullptr)
{
std::cout<<"Out of space!"<<std::endl;
return nullptr;
} S->Capacity=MaxElements;
S->TopOfStack=-; MakeEmpty(S); return S; } void DisposeStack(Stack S)
{
if(S!=nullptr)
{
delete S->Array;
delete S;
}
} bool IsEmpty(Stack S)
{
return S->TopOfStack==-;
} void MakeEmpty(Stack S)
{
S->TopOfStack=-;
} void Push(ElementType X,Stack S)
{
if(IsFull(S))
{
std::cout<<"Full Stack"<<std::endl;
return;
}
S->Array[++S->Array[S->TopOfStack]]=X; } ElementType Top(Stack S)
{
if(!IsEmpty(S))
{
return S->Array[S->TopOfStack];
} std::cout<<"Empty Error"<<std::endl;
return ;
} void Pop(Stack S)
{
if(IsEmpty(S))
{
std::cout<<"Empty Error"<<std::endl;
}
else{
S->TopOfStack--;
}
} ElementType TopandPop(Stack S)
{
if(!IsEmpty(S))
{
return S->Array[S->TopOfStack--];
}
std::cout<<"Empty Error"<<std::endl;
return ; } bool IsFull(Stack S)
{
return S->Capacity==MaxCapacity;
}
优点:把不断开空间的时间省下来了;
缺点:数组商都是有限的;
四、应用
(1)括号的匹配
void test(string s)
{
auto stack = CreateStack(); for(auto c: s)
{
if(c=='['||c=='{'||c=='(')
{
Push(c, stack);
}
else
{
if(c==']'||c=='}'||c==')')
{
if(IsEmpty(stack))
{
cout<<"Error!"<<endl;
}
else
{
if(c==']')
{
auto temp = TopandPop(stack);
if(temp!='[')
{
cout<<"Error!"<<endl;
}
}
else if(c=='}')
{
auto temp = TopandPop(stack);
if(temp!='{')
{
cout<<"Error!"<<endl;
}
}
else if(c==')')
{
auto temp = TopandPop(stack);
if(temp!='(')
{
cout<<"Error!"<<endl;
}
}
else
{
cout<<"Scanf Error!"<<endl;
} }
}
else
{
cout<<"Scanf Error!"<<endl;
}
} } if(stack->TopOfStack==-)
{
cout<<"RIGHT"<<endl;
}
else
{
cout<<"Error!"<<endl;
} }
(2)后缀表达式的计算。
后缀表达式的优点就是完全不需要考虑什么优先级之类的,直接从左边算到右边就可以了,简直方便快捷,在这里我就默默的实现下,小白勿喷
int calculate(string S) //10以内的整数运算
{ auto stack = CreateStack(); for(auto c:S)
{
switch (c)
{
case '+' :
{
auto t1=TopandPop(stack);
auto t2=TopandPop(stack);
double t3=t1+t2; Push(t3, stack);
} break;
case '-':
{
auto t1=TopandPop(stack);
auto t2=TopandPop(stack);
double t3=t2-t1; Push(t3, stack);
}
break;
case '*':
{
auto t1=TopandPop(stack);
auto t2=TopandPop(stack);
double t3=t2*t1; Push(t3, stack);
}
break;
case '/':
{
auto t1=TopandPop(stack);
auto t2=TopandPop(stack);
double t3=t2/t1; Push(t3, stack);
}
break;
default:
{
Push(c-'', stack);
}
break;
}
} return Top(stack); }
3、中缀表达式转化成后缀表达式
#include<iostream>
#include<cstdio>
#include<cstring>
#include<string>
#include<vector>
#include<stack>
#include<algorithm>
using namespace std; int pre(char a) //操作符优先级比较
{
if(a == '=' || a == '(')
return ;
else if(a == '+' || a == '-')
return ;
else if(a == '*' || a == '/')
return ;
else
{
cout<<"Error!"<<endl;
return ;
}
} int main()
{
string str;
stack<char> ope; //操作符
vector<char> ans;//后缀表达式
vector<char>::iterator start, end;
getchar(); //清除输入垃圾 while(!ope.empty()) //初始化
ope.pop();
ans.clear();
ope.push('='); //结束标志
cin>>str;
auto len = str.length();
for(int i = ; i < len; ++i)
{
if(str[i] >= '' && str[i] <= '') //操作数直接存入ans
ans.push_back(str[i]);
else if(str[i] == '(') //左括号入栈
ope.push(str[i]);
else if(str[i] == ')') //右括号,将匹配的左括号内容存入ans,左括号出栈
{
while (ope.top() != '(')
{
ans.push_back(ope.top());
ope.pop();
}
ope.pop(); //左括号出栈
}
else if(pre(str[i]) > pre(ope.top())) //优先级大于栈顶元素则入栈
ope.push(str[i]);
else //小于栈顶元素
{
while(pre(str[i]) <= pre(ope.top()))
{
ans.push_back(ope.top());
ope.pop();
}
ope.push(str[i]);
}
}
while(ope.top() != '=') //其余操作符存入后缀表达式中
{
ans.push_back(ope.top());
ope.pop();
}
for(start = ans.begin(), end = ans.end(); start < end; ++start)
printf("%c", *start);
printf("\n"); return ;
}
《数据结构与算法分析》学习笔记(四)——栈ADT的更多相关文章
- java之jvm学习笔记四(安全管理器)
java之jvm学习笔记四(安全管理器) 前面已经简述了java的安全模型的两个组成部分(类装载器,class文件校验器),接下来学习的是java安全模型的另外一个重要组成部分安全管理器. 安全管理器 ...
- <数据结构与算法分析>读书笔记--最大子序列和问题的求解
现在我们将要叙述四个算法来求解早先提出的最大子序列和问题. 第一个算法,它只是穷举式地尝试所有的可能.for循环中的循环变量反映了Java中数组从0开始而不是从1开始这样一个事实.还有,本算法并不计算 ...
- <数据结构与算法分析>读书笔记--运行时间计算
有几种方法估计一个程序的运行时间.前面的表是凭经验得到的(可以参考:<数据结构与算法分析>读书笔记--要分析的问题) 如果认为两个程序花费大致相同的时间,要确定哪个程序更快的最好方法很可能 ...
- <数据结构与算法分析>读书笔记--利用Java5泛型实现泛型构件
一.简单的泛型类和接口 当指定一个泛型类时,类的声明则包括一个或多个类型参数,这些参数被放入在类名后面的一对尖括号内. 示例一: package cn.generic.example; public ...
- 官网实例详解-目录和实例简介-keras学习笔记四
官网实例详解-目录和实例简介-keras学习笔记四 2018-06-11 10:36:18 wyx100 阅读数 4193更多 分类专栏: 人工智能 python 深度学习 keras 版权声明: ...
- C#可扩展编程之MEF学习笔记(四):见证奇迹的时刻
前面三篇讲了MEF的基础和基本到导入导出方法,下面就是见证MEF真正魅力所在的时刻.如果没有看过前面的文章,请到我的博客首页查看. 前面我们都是在一个项目中写了一个类来测试的,但实际开发中,我们往往要 ...
- IOS学习笔记(四)之UITextField和UITextView控件学习
IOS学习笔记(四)之UITextField和UITextView控件学习(博客地址:http://blog.csdn.net/developer_jiangqq) Author:hmjiangqq ...
- Learning ROS for Robotics Programming Second Edition学习笔记(四) indigo devices
中文译著已经出版,详情请参考:http://blog.csdn.net/ZhangRelay/article/category/6506865 Learning ROS for Robotics Pr ...
- Typescript 学习笔记四:回忆ES5 中的类
中文网:https://www.tslang.cn/ 官网:http://www.typescriptlang.org/ 目录: Typescript 学习笔记一:介绍.安装.编译 Typescrip ...
- ES6学习笔记<四> default、rest、Multi-line Strings
default 参数默认值 在实际开发 有时需要给一些参数默认值. 在ES6之前一般都这么处理参数默认值 function add(val_1,val_2){ val_1 = val_1 || 10; ...
随机推荐
- java分布式通信系统(J2EE分布式服务器架构)
一.序言 近几个月一直从事一个分布式异步通信系统,今天就整理并blog一下. 这是一个全国性的通信平台,对性能,海量数据,容错性以及扩展性有非常高的要求,所以在系统的架构上就不能简单的采用集中式.简单 ...
- cocos基础教程(2)Window环境下搭建(补充)
一.环境搭建 1.JDK.Eclipse与SDK 我用的JDK是1.7 Eclipse用的是Luna版的 这些之前都已经设好了,相关下载自己网上找吧 2. 下载最新的Cocos2d-x,我下的是3.5 ...
- 新浪微博的XSS漏洞攻击过程详解
今天晚上(2011年6月28日),新浪微博出现了一次比较大的XSS攻击事件.大量用户自动发送诸如:“郭美美事件的一些未注意到的细节”,“建 党大业中穿帮的地方”,“让女人心动的100句诗歌”,“3D肉 ...
- 淘宝(阿里百川)手机客户端开发日记第六篇 Service详解(一)
public abstract class Service; [API文档关于Service类的介绍] A Service is an application component representi ...
- [Effective JavaScript 笔记]第40条:避免继承标准类
ECMAScript标准库里配备了许多重要的类,如Array,function,以及Date等.扩展这些类生成子类可以方便完成很多工作,但它们的定义具有很多特殊的行为,所以很难写出行为正确的类. Ar ...
- 如何改变服务器的本地域名来访问本地服务器 而不用localhost或者127.0.0.1来访问
1. vim nginx.conf 如下: server { listen 80; server_name pma; ...
- HDOJ 1312 DFS&BFS
Red and Black Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) To ...
- C/C++程序终止时执行的函数——atexit()函数详解
很多时候我们需要在程序退出的时候做一些诸如释放资源的操作,但程序退出的方式有很多种,比如main()函数运行结束.在程序的某个地方用exit()结束程序.用户通过Ctrl+C或Ctrl+break操作 ...
- 使用pymongo需要手动关闭MongoDB Connection吗?
答:Disconnecting will close all underlying sockets in the connection pool. If this instance is used a ...
- Plus One Linked List
Given a non-negative number represented as a singly linked list of digits, plus one to the number. T ...