怎么实现Linux下的逆波兰计算器dc?
@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?的更多相关文章
- Linux下的实用工具——计算器bc
Linux下的实用工具——计算器 1. bc指令算加法,如图: 4. bc指令算除法(进阶),如图示,10/3之所以为3,是因为我们没有指定小数点后取几位,默认取到整数部分:而10/100之所以为 ...
- 6, java数据结构和算法: 栈的应用, 逆波兰计算器, 中缀表达式--> 后缀表达式
直接上代码: public class PolandCalculator { //栈的应用:波兰计算器: 即: 输入一个字符串,来计算结果, 比如 1+((2+3)×4)-5 结果为16 public ...
- C#数据结构与算法系列(十):逆波兰计算器——逆波兰表达式(后缀表达式)
1.介绍 后缀表达式又称逆波兰表达式,与前缀表达式相似,只是运算符位于操作数之后 2.举例说明 (3+4)*5-6对应的后缀表达式就是3 4 +5 * 6 - 3.示例 输入一个逆波兰表达式(后缀表达 ...
- 在UTF8(linux)下,逆置汉字字符串
#include <stdio.h> int main() { char c[]="我是如此热爱编程!"; ,min=,max; while(c[index]) { i ...
- bc:linux下命令行计算器
在linux下,存在一个命令行的计算器:bc.该程序一般随发行版发布. bc计算器能够执行一些基本的计算,包括+,-,×,\,%. 这些计算不经针对十进制,还可以使用二进制,八进制,十六进制,并且可以 ...
- Linux下的简单好用的计算器bc
1. 关于bc bc是随意精度计算器语言,通常在linux下当计算器用,简单好用.相当于windows下的计算器. 2. 支持的运算符 主要的数学运算: + 加法 - 减法 * 乘法 / 除法 ^ 指 ...
- HDU1237 简单的计算器 【堆】+【逆波兰式】
简单的计算器 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) Total Sub ...
- HDU1237 简单计算器 【栈】+【逆波兰式】
简单计算器 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) Total Subm ...
- 《linux下的计算器:bc用法入门篇》
说起电脑上的计算器,可能所有人的印象都是这样的:
随机推荐
- B - Bridging signals (LIS)
点击打开链接 B - Bridging signals 'Oh no, they've done it again', cries the chief designer at the Waferlan ...
- springmvc 通过@ResponseBody 返回json的中文乱码解决方案2个
1.方法上面的RequestMapping要加上红色的部分. @ResponseBody @RequestMapping(value = "/search", prod ...
- IntelliJ IDEA 配置JSP & Servlet开发环境
首先我们要安装和配置好JAVA和TOMCAT,这里不再详细描述 下载地址: JAVA:https://www.oracle.com/technetwork/java/javase/downloads/ ...
- Weblogic wls-wsat组件反序列化漏洞(CVE-2017-10271)
CVE编号: CVE-2017-10271 漏洞描述: Weblogic wls-wsat组件反序列化漏洞 利用脚本: https://github.com/hanc00l/weblogic_wls_ ...
- php 递归数据,三维数组转换二维
public function sortarea($area, $parent_id = 0, $lev = 1){ static $list; foreach($area as $v){ if($v ...
- npm安装包很慢
每次安装时: 可以通过指定 --registry,指向国内镜像服务器地址来加快安装速度. npm install -gd express --registry=http://registry.npm. ...
- 浅谈mongodb与Python的交互
1. mongdb和python交互的模块 pymongo 提供了mongdb和python交互的所有方法 安装方式: pip install pymongo 2. 使用pymongo 导入pymon ...
- js 的概念和声明-js 的变量-js 的运算符和逻辑结构-js 的数组
js 的概念和声明Js的概念和声明:问题:在网页的发展历程中,发现网页不能对用户的数据进行自动校验,和提供一些特效造成用户体验极差解决:使用JavaScript作用:可以让网页和用户之间进行直接简单的 ...
- 实验一 c++简单程序设计
一.实验内容 1.ex 2_28 (1) 用if...else判断 #include<iostream> using namespace std; int main() { char i; ...
- MNIST手写数字识别 Tensorflow实现
def conv2d(x, W): return tf.nn.conv2d(x, W, strides=[1, 1, 1, 1], padding='SAME') 1. strides在官方定义中是一 ...