C语言实现逆波兰式实例

所属分类: 软件编程 / C 语言 阅读数: 131
收藏 0 赞 0 分享

复制代码 代码如下:

#include<stdio.h>
#include<string.h>

typedef struct{char s[20][20];int top;}SQ;

void copystr(char *a,char *b)
{
    int i=0;
    do
    {
        b[i]=a[i];
        i++;
    }
    while(a[i]!='\0');
    b[i]='\0';
}

void voidSQ(SQ *s)
{
    s->top=-1;
}

int ifempty(SQ *s)
{
    return(s->top==-1);
}

void push(SQ *S,char *c)
{
    if(S->top==19)
        printf("over flow\n");
    else
    {

        S->top++;
        copystr(c,S->s[S->top]);
    }
}
char *pop(SQ *S)
{
    if(ifempty(S))
    {
        printf("over flow!\n");
        return(NULL);
    }
    else
        return(S->s[S->top--]);
}

int judge(char *c)
{
    if(c[1]=='\0')
        switch(c[0])
    {
        case '+':return(3);
        case '-':return(3);
        case '*':return(2);
        case '/':return(2);
        default:return(1);
    }
    else
        return(1);
}

void write(char *a,char *b,char *c)
{
    strcat(a,c);
    strcat(a,b);
}

int seek(char *c,int start)
{
    int signal=1;
    for(start=start++;c[start]!='\0'&&signal!=0;start++)
    {
        if(c[start]==')')
            signal--;
        else if(c[start]=='(')
            signal++;
    }
    if(signal==0)
        return(start-1);
    else
    {
        printf("输入无效式子\n");
        return(-1);
    }
}
void FB(SQ *A,SQ *B)
{
    for(;!ifempty(A);)
    {
        push(B,A->s[A->top]);
        pop(A);
    }
}

char *rewrite(char *A)
{
    SQ front;
    SQ back;
    int i,j,k,flag=0;
    char *result;
    char mid[20];
    voidSQ(&front);
    voidSQ(&back);
    for(i=0;A[i]!='\0';)
    {
        if(A[i]=='(')
        {
            j=seek(A,i);
            for(k=i+1;k<j;k++)
            {
                mid[k-i-1]=A[k];
            }
            mid[j-i-1]='\0';
            copystr(rewrite(mid),mid);
            push(&back,mid);
            i=j+1;
        }
        else if(A[i]!='(')
        {
            mid[0]=A[i];
            mid[1]='\0';
            push(&back,mid);
            i++;
        }
    }
    FB(&back,&front);
    for(;front.top>=2;)
    {
        flag=0;
        for(i=0;i<=front.top;i++)
        {
            if(judge(front.s[i])==2)
            {
                flag=1;
                break;
            }

        }
        if(flag==1)
        {
            for(;front.top>=2;)
            {
                if(judge(front.s[front.top])==1&&judge(front.s[front.top-1])==2&&judge(front.s[front.top-2])==1)
                {
                    write(front.s[front.top],front.s[front.top-1],front.s[front.top-2]);
                    push(&back,front.s[front.top]);
                     pop(&front);
                    pop(&front);
                     pop(&front);
                }
                else
                {
                    push(&back,front.s[front.top]);
                    pop(&front);
                }
            }
            FB(&front,&back);
            FB(&back,&front);
        }
        else
        {
            for(;front.top>=2;)
            {
                if(judge(front.s[front.top])==1&&judge(front.s[front.top-1])==3&&judge(front.s[front.top-2])==1)
                {
                    write(front.s[front.top],front.s[front.top-1],front.s[front.top-2]);

                    push(&back,front.s[front.top]);
                     pop(&front);
                    pop(&front);
                     pop(&front);
                }
                else
                {
                    push(&back,front.s[front.top]);
                    pop(&front);
                }
            }
            FB(&front,&back);
            FB(&back,&front);
        }
    }
    result=front.s[front.top];
    return(result);
}

typedef struct{char c[20];int top;}sq;
int execute(char a,char b,char c)
{
    switch(a)
    {
    case('+'):return((c-48)+(b-48));
    case('-'):return((c-48)-(b-48));
    case('*'):return((c-48)*(b-48));
    case('/'):return((c-48)/(b-48));
    }
}

void voidsq(sq *s)
{
    s->top=-1;
}

int ifsqempty(sq *s)
{
    return(s->top==-1);
}

void pushsq(sq *s,char x)
{
    if(s->top==19)
        printf("over flow!\n");
    else
    {
        s->top=s->top+1;
        s->c[s->top]=x;
    }
}

void popsq(sq *s)
{
    if(ifsqempty(s))
        printf("over flow!\n");
    else
        s->top--;
}

int just(char c)
{   
    switch(c)
    {
        case ('+'):return(0);
        case ('-'):return(0);
        case ('*'):return(0);
        case ('/'):return(0);
        default:return(1);
    }
}

void restread(sq *a,sq *b)
{
    for(;!ifsqempty(a);)
    {
        pushsq(b,a->c[a->top]);
        popsq(a);
    }
}


int calculate(char *c)
{
    sq rest,read;
    int i,re;
    voidsq(&rest);
    voidsq(&read);
    for(i=0;c[i]!='\0';i++)
        pushsq(&read,c[i]);
    for(;read.top>=2;)
    {
        for(;read.top>=2;)
        {
            if(just(read.c[read.top])==0&&just(read.c[read.top-1])==1&&just(read.c[read.top-2]) ==1)
            {

                re=execute(read.c[read.top],read.c[read.top-1],read.c[read.top-2]);
                pushsq(&rest,re+48);
                popsq(&read);
                popsq(&read);
                 popsq(&read);
            }
            else
            {
                pushsq(&rest,read.c[read.top]);
                popsq(&read);
            }
        }
        restread(&read,&rest);
        restread(&rest,&read);
    }
    return(read.c[0]-48);
}


void main()
{
    char re[20];
    char a[20];
    printf("请输入算式:\n");
    scanf("%s",a);
    copystr(rewrite(a),re);
    printf("逆波兰式:\n%s\n",re);
    printf("求值结果:\n%d\n",calculate(re));

}

更多精彩内容其他人还在看

用标准c++实现string与各种类型之间的转换

这个类在头文件中定义, < sstream>库定义了三种类:istringstream、ostringstream和stringstream,分别用来进行流的输入、输出和输入输出操作。另外,每个类都有一个对应的宽字符集版本
收藏 0 赞 0 分享

C++如何通过ostringstream实现任意类型转string

再使用整型转string的时候感觉有点棘手,因为itoa不是标准C里面的,而且即便是有itoa,其他类型转string不是很方便。后来去网上找了一下,发现有一个好方法
收藏 0 赞 0 分享

C/C++指针小结

要搞清一个指针需要搞清指针的四方面的内容:指针的类型,指针所指向的类型,指针的值或者叫指针所指向的内存区,还有指针本身所占据的内存区
收藏 0 赞 0 分享

C++ 类的静态成员深入解析

在C++中类的静态成员变量和静态成员函数是个容易出错的地方,本文先通过几个例子来总结静态成员变量和成员函数使用规则,再给出一个实例来加深印象
收藏 0 赞 0 分享

C++类的静态成员初始化详细讲解

通常静态数据成员在类声明中声明,在包含类方法的文件中初始化.初始化时使用作用域操作符来指出静态成员所属的类.但如果静态成员是整型或是枚举型const,则可以在类声明中初始化
收藏 0 赞 0 分享

C++类静态成员与类静态成员函数详解

静态成员不可在类体内进行赋值,因为它是被所有该类的对象所共享的。你在一个对象里给它赋值,其他对象里的该成员也会发生变化。为了避免混乱,所以不可在类体内进行赋值
收藏 0 赞 0 分享

C++中的friend友元函数详细解析

友元可以是一个函数,该函数被称为友元函数;友元也可以是一个类,该类被称为友元类。友元函数的特点是能够访问类中的私有成员的非成员函数。友元函数从语法上看,它与普通函数一样,即在定义上和调用上与普通函数一样
收藏 0 赞 0 分享

static全局变量与普通的全局变量的区别详细解析

以下是对static全局变量与普通的全局变量的区别进行了详细的分析介绍,需要的朋友可以过来参考下,希望对大家有所帮助
收藏 0 赞 0 分享

C++ explicit关键字的应用方法详细讲解

C++ explicit关键字用来修饰类的构造函数,表明该构造函数是显式的,既然有"显式"那么必然就有"隐式",那么什么是显示而什么又是隐式的呢?下面就让我们一起来看看这方面的知识吧
收藏 0 赞 0 分享

教你5分钟轻松搞定内存字节对齐

随便google一下,人家就可以跟你解释的,一大堆的道理,我们没怎么多时间,讨论为何要对齐.直入主题,怎么判断内存对齐规则,sizeof的结果怎么来的,请牢记以下3条原则
收藏 0 赞 0 分享
查看更多