#返回上一级

@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. robot framework接口测试之一-完整的测试用例

    *** Settings *** Library Collections Library json Library requests Library RequestsLibrary Library H ...

  2. 某种带权有向无环图(graph)的所有路径的求法

    // 讨论QQ群:135202158 最近做某个东西,最后用图实现了,这里总结一下算法. 假设有以下带权有向无环图(连通或非连通,我这里用的是非连通的): 每个节点(node)可能与其他节点有向地相连 ...

  3. (转)使用VS实现XML2CS

    转自 StackOverFlow Method 1 XSD tool Suppose that you have your XML file in this location C:\path\to\x ...

  4. leetcode-209-长度最小的子数组

    题目描述: 给定一个含有 n 个正整数的数组和一个正整数 s ,找出该数组中满足其和 ≥ s 的长度最小的连续子数组.如果不存在符合条件的连续子数组,返回 0. 示例: 输入: s = 7, nums ...

  5. linux克隆后修配置

    第一步:克隆 第二步:vi /etc/sysconfig/network-scripts/ifcfg-eth0   编辑 DEVICE=eth0 TYPE=Ethernet ONBOOT=yes NM ...

  6. easyui实现多选框,并且获取值

    在easyui官方文档里面是没有combobox下拉框,可以进行多选的,但是其实是可以多选的, <td align="left">大区:   <input typ ...

  7. C. Edgy Trees Codeforces Round #548 (Div. 2) 【连通块】

    一.题面 here 二.分析 这题刚开始没读懂题意,后来明白了,原来就是一个数连通块里点数的问题.首先在建图的时候,只考虑红色路径上的点.为什么呢,因为为了不走红色的快,那么我们可以反着想只走红色的路 ...

  8. 小程序中实时将less编译成wxss

    1.npm或者yarn全局安装wxss-cli npm install -g wxss-cli 2.运行wxss-cli命令(weuiTest为小程序目录) wxss ./weuiTest 实时监听w ...

  9. Cisco ISR4400 Netflow 配置模板

    flow exporter NAME destination 145.0.1.200 transport udp 9991 export-protocol netflow-v5 flow monito ...

  10. Mac下在Shell终端下使用open快速打开窗口文件夹

    Ubuntu下可以使用nautilus打开,但是在Mac替代的是open. 打开当前路径的窗口 oepn . 打开其他窗口 open /dirname 其实open不只可以打开窗口,应用同样支持. 关 ...