#返回上一级

@Author: 张海拔

@Update: 2014-01-12

@Link: http://www.cnblogs.com/zhanghaiba/p/3516660.html

 /*
*Author: ZhangHaiba
*Date: 2014-1-12
*File: dc_linux.c
*
*a demo shows how to use a stack to implement dc which as a built-in tool in Linux
*but, this demo only support int number and int operation + - * /
*/ #include <stdio.h>
#include <stdlib.h>
#include <ctype.h>
#define ST_LEN 1024 //the depth of stack
#define BUF_SIZE 32
#define NUM '0' //public from stack.h
void push(int);
int pop();
int top();
int is_empty();
int is_full();
void clear();
//private form stack.c
int stack[ST_LEN];
int sp = ; //point to next empty space //public form main.h
int token();
//public form main.c
char buf[BUF_SIZE];
int cnt = ; int main(void)
{
int c;
int op2, op1; while ((c = token()) != EOF) {
switch(c) {
case NUM:
push(atoi(buf));
break;
case '+':
if (size() >= ) {
op2 = pop(), op1 = pop();
push(op1 + op2);
} else
printf("dc: stack empty\n");
break;
case '-':
if (size() >= ) {
op2 = pop(), op1 = pop();
push(op1 - op2);
} else
printf("dc: stack empty\n");
break;
case '*':
if (size() >= ) {
op2 = pop(), op1 = pop();
push(op1 * op2);
} else
printf("dc: stack empty\n");
break;
case '/':
if (size() >= ) {
op2 = pop(), op1 = pop();
push(op1 / op2);
} else
printf("dc: stack empty\n");
break;
case 'p':
printf(is_empty() ? "dc: stack empty\n" : "%d\n", top());
default:
break;
}
}
return ;
} int token()
{
int c = getchar();
if (isdigit(c)) {
buf[cnt++] = c;
while ((c = getchar()) != EOF) {
if (isdigit(c))
buf[cnt++] = c;
else {
buf[cnt] = '\0';
cnt = ;
ungetc(c, stdin);
return NUM;
}
}
} else
return c;
} void push(int item)
{
if (!is_full())
stack[sp++] = item;
} int pop()
{
if (!is_empty())
return stack[--sp];
} int top()
{
if (!is_empty())
return stack[sp-];
} int is_full()
{
return sp >= ST_LEN;
} int is_empty()
{
return sp <= ;
} int size()
{
return sp;
} void clear()
{
sp = ;
}

Linux下有一个很强大计算器bc,而逆波兰计算器dc就是bc的后台。

逆波兰表达式(后缀表达式)不需要括号也不需要定义优先级就可以运算,dc就是用来处理这种“干净”的输入。

学习了栈这个数据结构,对于逆波兰表达式的计算是很容易的。

规则是:遇到操作数则压栈,遇到操作符则弹栈,先后弹出两个操作数,第一个是op2,第二是op1,然后根据操作符把计算结果再次压栈。所以栈顶值就是当前计算的结果。

这里的实现的简易dc,仅仅支持int整型,和int操作符+ - * /。

dc的用例如下(上述实现的简易版 dc,其测试用例同下):

432 13 * p
5616
3 p
3
* p
16848
* p
dc: stack empty
16848
42 / 4 p
4
p
4

上面的实现中,需要注意的是:

(1)数据封装方面,最好把stack做成一个单独模块,这样可以提供接口,隐藏stack数据本身。

(2)对于栈中元素<2时遇到操作符,不要执行单独一次的弹栈(即把两次弹栈看做一个原子操作,测试发现dc就是这样处理的)。

(3)由于输入数字不一定是一位数,所以需要一个buffer数组来存储多位数的字符表示并封装成字符串用标准库函数atoi转换,这个过程需要用ungetc(int, FILE*)来放回一个字符到缓冲区。

#返回上一级

怎么实现Linux下的逆波兰计算器dc?的更多相关文章

  1. Linux下的实用工具——计算器bc

    Linux下的实用工具——计算器   1. bc指令算加法,如图: 4. bc指令算除法(进阶),如图示,10/3之所以为3,是因为我们没有指定小数点后取几位,默认取到整数部分:而10/100之所以为 ...

  2. 6, java数据结构和算法: 栈的应用, 逆波兰计算器, 中缀表达式--> 后缀表达式

    直接上代码: public class PolandCalculator { //栈的应用:波兰计算器: 即: 输入一个字符串,来计算结果, 比如 1+((2+3)×4)-5 结果为16 public ...

  3. C#数据结构与算法系列(十):逆波兰计算器——逆波兰表达式(后缀表达式)

    1.介绍 后缀表达式又称逆波兰表达式,与前缀表达式相似,只是运算符位于操作数之后 2.举例说明 (3+4)*5-6对应的后缀表达式就是3 4 +5 * 6 - 3.示例 输入一个逆波兰表达式(后缀表达 ...

  4. 在UTF8(linux)下,逆置汉字字符串

    #include <stdio.h> int main() { char c[]="我是如此热爱编程!"; ,min=,max; while(c[index]) { i ...

  5. bc:linux下命令行计算器

    在linux下,存在一个命令行的计算器:bc.该程序一般随发行版发布. bc计算器能够执行一些基本的计算,包括+,-,×,\,%. 这些计算不经针对十进制,还可以使用二进制,八进制,十六进制,并且可以 ...

  6. Linux下的简单好用的计算器bc

    1. 关于bc bc是随意精度计算器语言,通常在linux下当计算器用,简单好用.相当于windows下的计算器. 2. 支持的运算符 主要的数学运算: + 加法 - 减法 * 乘法 / 除法 ^ 指 ...

  7. HDU1237 简单的计算器 【堆】+【逆波兰式】

    简单的计算器 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total Sub ...

  8. HDU1237 简单计算器 【栈】+【逆波兰式】

    简单计算器 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total Subm ...

  9. 《linux下的计算器:bc用法入门篇》

    说起电脑上的计算器,可能所有人的印象都是这样的:

随机推荐

  1. Python3.5 学习三

    对文件的操作 打开模式: 1 f=open("xxx","r",encoding=="utf-8") 只读 2 f=open("x ...

  2. linux安装spark-2.3.0集群

    (安装spark集群的前提是服务器已经配置了jdk并且安装hadoop集群(主要是hdfs)并正常启动,hadoop集群安装可参考<hadoop集群搭建(hdfs)>) 1.配置scala ...

  3. byte转文件流 下载到本地

    此方法将byte类型文件转为文件流保存到本地 byte 经过BASE64Decoder 进行编码之后的类型 所以需要解码 防止出现乱码及文件损毁 /** * byte 转文件 下载到本地 * @par ...

  4. JMeter PerfMon Metrics Collector性能监控插件

    官方文档地址https://jmeter-plugins.org/wiki/PerfMon/ JMeter是一款压力测试工具,我们也可以用它来监控服务器资源使用情况. JMeter正常自带可以通过To ...

  5. AssertJ断言系列-----------<数据库断言二>

    那么,在实际的接口测试中,我们除了要断言响应的数据正确之外,可能有的还需要断言数据层是否数据真的有入库. assertj db是可以直接对数据库进行断言和操作的. 一.创建一个students表 CR ...

  6. python学习,day3:函数式编程,带参数

    # coding=utf-8 # Author: RyAn Bi def test(x,y,z): print(x) print(y) print(z) test(y=2,z =3,x=1) #形参与 ...

  7. 生成allure测试报告之后,服务器端口无法访问查看生成的report,可能是这样引起的。

    1. 检查防火墙 2. 如果机器有安装ADsafe,请关闭adsafe后重试

  8. scp 一次拷贝多个文件

    用正则表达式去匹配即可, scp  *.tar  root@11.11.11.12:/root/ 拷贝当前目录下的所有tar类型的文件到服务器上

  9. 洛谷 P2014 选课(树形背包)

    洛谷 P2014 选课(树形背包) 思路 题面:洛谷 P2014 如题这种有依赖性的任务可以用一棵树表示,因为一个儿子要访问到就必须先访问到父亲.然后,本来本题所有树是森林(没有共同祖先),但是题中的 ...

  10. 区间DP的学习(持续更新)

    例题: 1.Multiplication Puzzle 原题地址:http://poj.org/problem?id=1651 2.Dire Wolf 原题地址:http://acm.split.hd ...