一、栈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的更多相关文章

  1. java之jvm学习笔记四(安全管理器)

    java之jvm学习笔记四(安全管理器) 前面已经简述了java的安全模型的两个组成部分(类装载器,class文件校验器),接下来学习的是java安全模型的另外一个重要组成部分安全管理器. 安全管理器 ...

  2. <数据结构与算法分析>读书笔记--最大子序列和问题的求解

    现在我们将要叙述四个算法来求解早先提出的最大子序列和问题. 第一个算法,它只是穷举式地尝试所有的可能.for循环中的循环变量反映了Java中数组从0开始而不是从1开始这样一个事实.还有,本算法并不计算 ...

  3. <数据结构与算法分析>读书笔记--运行时间计算

    有几种方法估计一个程序的运行时间.前面的表是凭经验得到的(可以参考:<数据结构与算法分析>读书笔记--要分析的问题) 如果认为两个程序花费大致相同的时间,要确定哪个程序更快的最好方法很可能 ...

  4. <数据结构与算法分析>读书笔记--利用Java5泛型实现泛型构件

    一.简单的泛型类和接口 当指定一个泛型类时,类的声明则包括一个或多个类型参数,这些参数被放入在类名后面的一对尖括号内. 示例一: package cn.generic.example; public ...

  5. 官网实例详解-目录和实例简介-keras学习笔记四

    官网实例详解-目录和实例简介-keras学习笔记四 2018-06-11 10:36:18 wyx100 阅读数 4193更多 分类专栏: 人工智能 python 深度学习 keras   版权声明: ...

  6. C#可扩展编程之MEF学习笔记(四):见证奇迹的时刻

    前面三篇讲了MEF的基础和基本到导入导出方法,下面就是见证MEF真正魅力所在的时刻.如果没有看过前面的文章,请到我的博客首页查看. 前面我们都是在一个项目中写了一个类来测试的,但实际开发中,我们往往要 ...

  7. IOS学习笔记(四)之UITextField和UITextView控件学习

    IOS学习笔记(四)之UITextField和UITextView控件学习(博客地址:http://blog.csdn.net/developer_jiangqq) Author:hmjiangqq ...

  8. Learning ROS for Robotics Programming Second Edition学习笔记(四) indigo devices

    中文译著已经出版,详情请参考:http://blog.csdn.net/ZhangRelay/article/category/6506865 Learning ROS for Robotics Pr ...

  9. Typescript 学习笔记四:回忆ES5 中的类

    中文网:https://www.tslang.cn/ 官网:http://www.typescriptlang.org/ 目录: Typescript 学习笔记一:介绍.安装.编译 Typescrip ...

  10. ES6学习笔记<四> default、rest、Multi-line Strings

    default 参数默认值 在实际开发 有时需要给一些参数默认值. 在ES6之前一般都这么处理参数默认值 function add(val_1,val_2){ val_1 = val_1 || 10; ...

随机推荐

  1. WPF 检测计算机网络连接情况

    ; ; ; ; [DllImport("wininet.dll")] private extern static bool InternetGetConnectedState(ou ...

  2. NoClassDefFoundError:aspectj/weaver/reflect/ReflectionWorld$Reflection

    Exception in thread "main" org.springframework.beans.factory.BeanDefinitionStoreException: ...

  3. sturct stat 结构体中 st_mode 的含义

    工作中遇到 else if( (s_buf.st_mode&S_IFMT) == S_IFDIR) return 2; else if( !(s_buf.st_mode&S_IFREG ...

  4. C语言中的struct和typedef struct<转载>

    原文:http://www.nowamagic.net/librarys/veda/detail/1785 typedef为C语言的关键字,作用是为一种数据类型定义一个新名字.这里的数据类型包括内部数 ...

  5. Python webpy微信公众号开发之 回复图文消息

    新建图文回复模板reply_pictext.xml: $def with (toUser,fromUser,createTime,title1,description1,picurl1,url1)&l ...

  6. Centos6.5更新e1000网卡驱动

    导读 在工作过程中经常遇到linux的操作系统网络不正常的情况,以前没有注意到,今天查看系统日志发现原来是网络驱动的问题.索性直接更新系统,更新网卡 问题:linux系统经常出现断网的情况,重启之后系 ...

  7. ris'In App Purchase总结

    原地址:http://www.cocoachina.com/bbs/read.php?tid=38555&page=1 In App Purchase属于iPhone SDK3.0的新特性,用 ...

  8. unity缓存和浏览器缓存

    原地址:http://www.cnblogs.com/hisiqi/p/3203553.html <蒸汽之城>游戏中,为什么会黑屏?或者无法正常进入游戏? 我们在进行多项测试中发现少数用户 ...

  9. Cocos2d坐标系转换

    Cocos2d-x坐标系和OpenGL坐标系相同,都是起源于笛卡尔坐标系(高中数学里面那种). 笛卡尔坐标系 笛卡尔坐标系中定义右手系原点在左下角,x向右,y向上,z向外,OpenGL坐标系为笛卡尔右 ...

  10. The CompilerVersion constant identifies the internal version number of the Delphi compiler.

    http://delphi.wikia.com/wiki/CompilerVersion_Constant The CompilerVersion constant identifies the in ...