C语言实现数学表达式运算

 更新时间:2021年11月26日 17:36:48   作者:_牛仔很忙  
这篇文章主要为大家详细介绍了c语言实现数学表达式运算,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

本文实例为大家分享了C语言实现数学表达式运算的具体代码,供大家参考,具体内容如下

1、开发思路: (假设有表达式 2 * 3 * ( 1 + 2) )

数字要一个一个取出放在内存中,根据相邻前后2个计算符号,判断是否要取出数字进行计算,2个数字的计算值重新放在内存中且顺序放置。考虑使用栈这种数据结构去保存数字和符号,用2个栈,1个栈保存数字,一个栈保存运算符号。

2、因要使用栈这种数据结构,本代码使用纯C语言开发,故先编写栈的代码,参考:
c语言实现通用数据结构(三):通用椎栈

3、重要处理逻辑

(1)如何判断前后2个运算符的优先级关系

(2)如何字符转换为数字

因键盘输入的内容为字符类型,需要判断输入的字符类型且进行必要转换
ASCII码表,表头依次为:二进制 十进制 十六进制 字符

(3)如何判断表达式处理完毕

默认先预置一个符号#,输入内容2 * 3 * ( 1 + 2)# ,当符号栈内为#,且当前处理的字符为#。则表达式处理完毕。

4、代码实现

#define _CRT_SECURE_NO_DEPRECATE

#define _CRT_SECURE_CPP_OVERLOAD_STANDARD_NAMES 1

#include <stdlib.h>
#include <stdio.h>
#include "myStack.h"

// 判断是否操作符
int ifOp(char c) {
    switch (c) {
    case '+':
        return 1;
    case '-':
        return 1;
    case '*':
        return 1;
    case '/':
        return 1;
    case '(':
        return 1;
    case ')':
        return 1;
    case '#':
        return 1;
    default:
        break;
    }
    return 0;
}

int findOffset(char * str,char c,int len) {
    for (int i = 0; i < len;i++) {
        if (str[i] == c) {
            return i;
        }
    }
    return -1;
}

//判断任意相继出现的2个运算符的优先级
char yxji(char op1,char op2) {
    char ops[] = "+-*/()#";
    char oplist[7][7] = { 
        ">><<<>>",
        ">><<<>>",
        ">>>><>>",
        ">>>><>>" ,
        "<<<<<=&",
        ">>>>&>>",
        "<<<<<&="};
    int len = sizeof(ops) / sizeof(char);
    return oplist[findOffset(ops,op1, len)][findOffset(ops, op2, len)];
}

//基础运算 a-第1个数  b-第2个数
void baseOp(char op,int a,int b,int * value) {
    printf("baseOp %d %c %d",a,op,b);
    //int value = 0;
    //int* p = &value;
    switch (op) {
    case '+':
        *value = a + b;
        break;
    case '-':
        *value = a-b;
        break;
    case '*':
        *value = a*b;
        break;
    case '/':
        *value = a/b;
        break;
    default:
        printf("运算符不合法");
        exit(1);
    }
}

//转换字符为数字
void transValue(char  c,int * v) {
    if (c > 47 && c < 58) {
        *v =  (c - 48);
    }
}

void printstack(MyStack * stack1,MyStack * stack2) {
    int len1 = myListGetSize(stack1);
    int len2 = myListGetSize(stack2);
    printf("stack1值:");
    for (int i = 0; i < len1;i++) {
        char* m = (char*)myListGetDataAt(stack1,i );
        printf("%c ", *m);
    }
    printf("\nstack2值:");
    for (int i = 0; i < len2; i++) {
        int* m = (int*)myListGetDataAt(stack2, i);
        printf("%d(%p) ", *m,m);
    }
    printf("\n");
}

// 计算,该方法只能对数字 0-9 运算(可掌握栈、指针的使用)
// 2*3*(1+2)#
void calculate(char bds[]) {
    int i = 0;
    char flag = '#';
    MyStack* stack1 = createMyStack();//stack1中放运算符
    myStackPush(stack1, &flag);
    MyStack* stack2 = createMyStack();//stack2中放数字
    //char c = bds[i]; // 等价于*(bds+i) 
    while (bds[i] != '#' || *(char*)myStackGetTop(stack1)!='#') {
        printstack(stack1,stack2);
        if (!ifOp(bds[i])) {
             /*
             *  这种写法不行!
                 int vv = 0;
                 transValue(bds[i], &vv)
             */
            int* vu = (int*)malloc(sizeof(int));
            transValue(bds[i], vu);
            printf("is number:%d\n", *vu);//打印出数字
            myStackPush(stack2, vu);
            i++;
        }
        else {
            printf("is fuhao:%c\n", bds[i]);
            char * op1 = (char*)myStackGetTop(stack1);
            printf("top1 op:%c\n",*op1);
            if (*op1 == '#') {
                myStackPush(stack1, &bds[i]);
                i++;
                continue;
            }
            char res = yxji(*op1, bds[i]);
            printf("yxji:%c\n",res);
            switch (res) {
                case '>':{
                    char* curop = (char*)myStackPop(stack1);//取出当前运算符
                    printf("top2 op:%c\n", *op1);
                    int* b = (int*)myStackPop(stack2);//第2个运算数
                    int* a = (int*)myStackPop(stack2);//第1个运算数
                    /*
                    *  这种写法不行! 
                        int value = 0;
                        baseOp(*curop,transValue(*a), transValue(*b),&value);
                    */
                    int* value = (int*)malloc(sizeof(int));
                    baseOp(*curop, *a, *b, value);
                    printf("=%d\n", *value);
                    myStackPush(stack2, value);
                    break;
                }
                case '<':
                    myStackPush(stack1, &bds[i]);
                    i++;
                    break;
                case '=': {
                    printf("()==");
                    myStackPop(stack1);//取出右括号 (
                    i++;
                    break;
                }
                default:
                    printf("表达式错误!");
                    exit(1);
            }
        }
    }
    int * valueRes = (int*)myStackPop(stack2);
    printf("计算结果值为:%d\n",*valueRes);

    freeMyList(stack1);
    freeMyList(stack2);
}

int main() {
    printf("输入表达式:\n");
    char bds[50];
    scanf("%s",bds);// 数组变量名,传入的相当于是数组第一个元素的地址。方法形参是个指针变量,指针变量才能存放地址
    calculate(bds);
    return 0;
}

5、代码开发过程总结 (踩坑填坑真实记录)

将符号转为数字并把数字放入栈中,若写为如下形式不行
int vv = 0;
transValue('1‘, &vv);
myStackPush(stack2, vv);

因为临时变量地址始终不变,第2个值赋值后,等于是把已放入栈内的第一个值修改了(程序中通过打印出指针变量值,即变量的地址,发现地址确实没变)
应该用如下方式:

int* vv= (int*)malloc(sizeof(int));
transValue('1‘, vv);
myStackPush(stack2, vv);

掌握指针用法很重要!此处记录如下:

6. 代码测试结果(开发环境:visual studio 2019)

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

相关文章

  • Objective-C不带加减号的方法实例

    Objective-C不带加减号的方法实例

    显而易见的事实是,Objective-C 中,+ 表示类方法,- 表示实例方法,这篇文章主要给大家介绍了关于Objective-C不带加减号的相关资料,需要的朋友可以参考下
    2021-06-06
  • c++网络编程下Linux的epoll技术和Windows下的IOCP模型

    c++网络编程下Linux的epoll技术和Windows下的IOCP模型

    c++ 网络编程LINUX-epoll/windows-IOCP下socket opoll函数用法 优于select方法的epoll 以及windows下IOCP 解决多进程服务端创建进程资源浪费问题,感兴趣的小伙伴一起来学习吧
    2021-08-08
  • c语言网络编程-标准步骤(比较简单)

    c语言网络编程-标准步骤(比较简单)

    这篇文章主要介绍了c语言网络编程-标准步骤(比较简单),需要的朋友可以参考下
    2014-01-01
  • C语言数据结构递归之斐波那契数列

    C语言数据结构递归之斐波那契数列

    这篇文章主要介绍了C语言数据结构递归之斐波那契数列的相关资料,希望通过本文能帮助到大家,让大家理解掌握这部分内容,需要的朋友可以参考下
    2017-10-10
  • C++中的操作符重载详细解析

    C++中的操作符重载详细解析

    运算符重载后不能改变运算符的操作对象(操作数)的个数;如:"+"是实现两个操作数的运算符,重载后仍然为双目运算符
    2013-09-09
  • C++类常量和类枚举

    C++类常量和类枚举

    这篇文章主要介绍了C++类常量和类枚举,给类当中定义一些常量,可以给所有类的对象使用,比如说我们在类当中定义一个数组,希望可以定义一个常量,用来初始化数组的长度,那么下面我i吗就来看看过程当如何吧
    2022-01-01
  • C语言、C++中的union用法总结

    C语言、C++中的union用法总结

    这篇文章主要介绍了C语言、C++中的union用法总结,本文讲解了什么是union、C中使用union、当union遇到对象等内容,需要的朋友可以参考下
    2014-10-10
  • VC小技巧汇总之控件技巧

    VC小技巧汇总之控件技巧

    这篇文章主要介绍了VC小技巧汇总之控件技巧,对于VC的开发很有借鉴价值,需要的朋友可以参考下
    2014-07-07
  • 浅析结束程序函数exit, _exit,atexit的区别

    浅析结束程序函数exit, _exit,atexit的区别

    在一个程序中最多可以用atexit()注册32个处理函数,这些处理函数的调用顺序与其注册的顺序相反,也即最先注册的最后调用,最后注册的最先调用
    2013-09-09
  • OpenCV数字图像处理基于C++之图像形态学处理详解

    OpenCV数字图像处理基于C++之图像形态学处理详解

    OpenCV是一款由Intel公司俄罗斯团队发起并参与和维护的一个计算机视觉处理开源软件库,支持与计算机视觉和机器学习相关的众多算法,下面这篇文章主要给大家介绍了关于OpenCV数字图像处理基于C++之图像形态学处理的相关资料,需要的朋友可以参考下
    2022-12-12

最新评论