算术表达式的实现,支持加减乘除,括号运算,表达式转二叉树


基本思路

首先,用户输入的待求表达式,也就是中缀表达式,对于人来说,这个很好理解,但是对于计算机,后缀表达式求值更加容易。如果看成一棵二叉树,其实中缀表达式就是对一个二叉树的中序遍历,后缀表达式(也叫逆波兰表达式)就是后序遍历的结果。那么主要思路就来了:先把中缀表达式转换成后缀表达式,再对后缀表达式进行求值。

步骤一,中缀表达式转后缀表达式

两个数据结构:

1,后缀表达式队列

用于存放最后的后缀表达式

2,操作符栈

对用户输入的操作符进行处理

算法伪代码:

/*
依次读入用户输入的中缀表达式
如果是数字,直接入队
如果是操作符,如果栈为空,直接入栈
   如果操作符为 ( 直接入栈
   如果操作符为 ) ,依次出栈,直到遇到第一个(,(和)都不入栈
   如果是其他操作符(+ - * /),则和栈顶元素进行比较优先级
           如果栈顶元素优先级大于等于(*和/ > +和-)操作符 ,则出栈,直到栈顶元素优先级小于操作符或栈为空
*/
//伪代码
//后缀表达式队列
queue num;
//操作符栈
stack option;
while(scanf("%c",&x)){
     if(x 是数){
           num.addBack(x);    //入队
     }else if(option.empty() || x == '('){
           //如果栈为空,直接入栈,如果操作符为 ( 直接入栈
           option.push(x) 
    }else if(x == ')'){
           //如果操作符为 ) ,依次出栈,直到遇到第一个(,(和)都不入栈
           while((tmp = option.pop()) != '('){
                   num.addBack(tmp); 
          }
    }else{
        //只是获取栈首元素,没有出栈
        while(1){
                 tmp = option.head();
                 if(tmp 优先级大于等于x){
                     tmp1 = option.pop();
                     num.addBack(tmp1); 
                 }else{
                     break;
                 }
        }
        option.push(x);
    }
}
//最后把所有操作符入队
while(!option.empty()){
        tmp = option.pop();
        num.addBack(tmp);
}

步骤二,后缀表达式计算值

数据结构:

结果栈

用于存放计算的中间过程的值和最终结果

算法伪代码:
/*
依次从后缀表达式队列中取值
如果是值,直接入栈
如果是操作符,从队列中取出两个数,进行运算(后取出来的数作为第一个运算符),计算结果再次压栈
*/
//用于存放结果的栈
stack res;
//后缀表达式队列,上一步求出来的
queue num;
while(!num.empty()){
     //取队首
     tmp = num.getFront();
     if(tmp 是数){
          res.push(tmp);
     }else{
           num1 = num.pop();
           num2 = num.pop();
           num3 = num2 tmp num1;
           res.push(num3);
     }
}

步骤三,转二叉树

算法伪代码:

    /*
    主要是在计算逆波兰表达式的时候,构造出一棵二叉树,
     构造思路和计算时差不多,如果是数,就直接构造节点,如果是操作符,就把前面两个节点提出来作为左右节点
    */
    //用于存放所有临时节点和最后根节点的数组
treeNode all[MAX];
int n=0;
//后缀表达式队列,上一步求出来的
queue num;
while(!num.empty()){
     //取队首
     tmp = num.getFront();
     if(tmp 是数){
          res[n++] = tmp;
     }else{
          num1 = all[--n];
          num2 = all[--n];
          tmp.left = num1;
          tmp.right = num2;
          all[n++] = tmp;
     }
}

运行情况

算术表达式的实现,支持加减乘除,括号运算,表达式转二叉树_第1张图片
运行情况

完整代码

   /**
 * 计算算术表达式的值,支持 加减乘除括号 运算
 * 思路是,先把用户输入的表达式(中缀表达式,中序遍历),格式化成逆波兰表达式(后缀表达式,后序遍历序列)
 * 再对后缀表达式进行求值
 */
#include 
#include 
//通用链表
#include "cl_link.h"

#define MAX 1024
#define IS_OPTION '0'
#define IS_NUM '1'

char* optionAll = "+-*/()";

typedef struct bTree{
    char type;
    double num;
    char option;
    struct bTree* left;
    struct bTree* right;
}bTree;

//运算符节点
typedef struct optionNode{
    //用于标志运算符
    char type;
    cl_link_node node;
    char option;
}optionNode;

//操作数节点
typedef struct numNode{
    // 用于标志操作数
    char type;
    cl_link_node node;
    double num;
}numNode;

//运算符栈
cl_link* optionStack = NULL;
//数字队列
cl_link* numQueue = NULL;
/**
 * 比较是否是运算符
 * @param  optionAll 运算符表
 * @param  aim       比较对象
 * @return           1 是 0 不是
 */
int isOption(const char* optionAll,const char aim)
{
    for (size_t i = 0; i < strlen(optionAll); i++) {
        if(aim == *(optionAll+i)){
            return 1;
        }
    }
    return 0;
}
/**
 * 比较优先级
 * @return 1 src >= des
 *         0 src < des
 */
int isMaxLevel(const char src, const char des)
{
    if(src == '*' || src == '/')
        return 1;
    if((src == '+' || src == '-') && (des == '+' || des == '-'))
        return 1;
    return 0;
}
void* show(void* arg)
{
    numNode* node = cl_link_get_data(arg, numNode, node);
    if(node->type == IS_NUM){

        printf("%0.2f,", node->num);
    }else{
        optionNode* node1 = cl_link_get_data(arg, optionNode, node);
        printf("%c,", node1->option);

    }
    return NULL;
}
//计算
double cal(double num1, double num2, char option)
{
    switch (option) {
        case '+':
            return num1+num2;
        case '-':
            return num1-num2;
        case '*':
            return num1*num2;
        case '/':
            return num1/num2;
    }
    return 0;
}


void pre(bTree* root){
    if(root != NULL)
    {
        if(root->type == IS_NUM){
            printf("%0.2f,", root->num);
        }else{
            printf("%c,", root->option);
        }
        pre(root->left);
        pre(root->right);
    }
}
void mid(bTree* root){
    if(root != NULL)
    {
        mid(root->left);
        if(root->type == IS_NUM){
            printf("%0.2f,", root->num);
        }else{
            printf("%c,", root->option);
        }
        mid(root->right);
    }
}
void back(bTree* root){
    if(root != NULL)
    {
        back(root->left);
        back(root->right);
        if(root->type == IS_NUM){
            printf("%0.2f,", root->num);
        }else{
            printf("%c,", root->option);
        }
    }
}
int main()
{
    //初始化运算符栈
    optionStack = cl_link_create();
    //初始化逆波兰空队列,即后序遍历结果
    numQueue = cl_link_create();
    //把输入格式化成后缀表达式 开始
    char expr[MAX];
    scanf("%s", expr);
    for (size_t i = 0; i < strlen(expr); i++) {
        //假设输入完全合法,不是运算符就是运算数
        if(!isOption(optionAll, *(expr+i))){
            //运算数,直接入队,numQueue
            numNode* node = malloc(sizeof(numNode));
            node->num = atoi(expr+i);
            node->type = IS_NUM;
            int tmp = node->num;
            while(tmp/10 > 0){
                i++;
                tmp /= 10;
            }
            //入队
            cl_link_add_back(numQueue, cl_link_get_node(node, numNode, node));
        }else{
            /**
             * 如果栈为空,直接入栈
             * 如果是( 直接入栈
             * 如果是) 则依次出栈到队列中,直到遇到第一个(
             * 如果X是其他运算符,则依次与栈顶元素比较优先级,
             * 如果栈顶元素的优先级大于等于X,
             * 则出栈,直到栈顶元素的优先级小于X或者栈为空。
             * 否则 入栈
             */
            optionNode* node = malloc(sizeof(optionNode));
            node->type = IS_OPTION;
            node->option = *(expr+i);
            if(optionStack->sum == 0 || *(expr+i) == '('){
                //如果栈为空,或者是(,直接入栈
                cl_link_push(optionStack, cl_link_get_node(node, optionNode, node));
            }else if(*(expr+i) == ')'){
                //如果是) 则依次出栈到队列中,直到遇到第一个(
                while(1){
                    optionNode* tmp = cl_link_get_data(cl_link_pop(optionStack), optionNode, node);
                    if(tmp->option == '('){
                        break;
                    }else{
                        cl_link_add_back(numQueue, cl_link_get_node(tmp, optionNode, node));
                    }
                }
            }else{
                printf("%C,", *(expr+i));
                //如果X是其他运算符,则依次与栈顶元素比较优先级,如果栈顶元素的优先级大于等于X,则出栈,直到栈顶元素的优先级小于X或者栈为空。
                while(1){
                    //得到栈顶元素
                    optionNode* tmp = cl_link_get_data(cl_link_get_head(optionStack), optionNode, node);
                    if(isMaxLevel(tmp->option, *(expr+i))){
                        //如果优先级大于运算符,就出栈到逆波兰队列
                        optionNode* tmp = cl_link_get_data(cl_link_pop(optionStack), optionNode, node);
                        cl_link_add_back(numQueue, cl_link_get_node(tmp, optionNode, node));
                    }else{

                        break;
                    }
                }
                optionNode* node = malloc(sizeof(optionNode));
                node->type = IS_OPTION;
                node->option = *(expr+i);
                cl_link_push(optionStack, cl_link_get_node(node, optionNode, node));
            }
        }
    }
    /**
     * 若运算符还有,则直接出栈
     */
    while(optionStack->sum != 0){
        optionNode* tmp = cl_link_get_data(cl_link_pop(optionStack), optionNode, node);
        cl_link_add_back(numQueue, cl_link_get_node(tmp, optionNode, node));
    }
    //把输入格式化成后缀表达式 结束

    printf("\n逆波兰表达式(后序遍历):");
    //打印逆波兰式,就是后序遍历结果
    cl_link_each(numQueue, NULL, show);
    bTree* allTreeNode[MAX];
    int n = 0;
    //计算逆波兰表达式结果,用于存放临时结果和最终结果
    cl_link* res = cl_link_create();
    while(numQueue->sum != 0){
        numNode* tmp = cl_link_get_data(cl_link_pop(numQueue), numNode, node);
        if(tmp->type == IS_NUM){
            //如果是数,直接入栈
            cl_link_push(res, cl_link_get_node(tmp, numNode, node));
            bTree* newNode = malloc(sizeof(bTree));
            newNode->left = NULL;
            newNode->right = NULL;
            newNode->type = IS_NUM;
            newNode->num = tmp->num;
            allTreeNode[n++] = newNode;
        }else{
            //如果是操作符,则pop出栈顶两个元素,进行运算,再入栈
            optionNode* tmpOption = (optionNode*)tmp;
            numNode* num1 = cl_link_get_data(cl_link_pop(res), numNode, node);
            numNode* num2 = cl_link_get_data(cl_link_pop(res), numNode, node);
            numNode* node = malloc(sizeof(numNode));
            node->num = cal(num2->num, num1->num, tmpOption->option);
            node->type = IS_NUM;
            cl_link_push(res, cl_link_get_node(node, numNode, node));
            bTree* newNode = malloc(sizeof(bTree));
            newNode->left = NULL;
            newNode->right = NULL;
            newNode->type = IS_OPTION;
            newNode->option = tmpOption->option;
            newNode->right = allTreeNode[--n];
            newNode->left = allTreeNode[--n];
            allTreeNode[n++] = newNode;
        }
    }
    numNode* resData = cl_link_get_data(cl_link_get_head(res), numNode, node);
    printf("\n运算结果是%0.2f\n", resData->num);
    printf("前序遍历\n");
    pre(allTreeNode[0]);
    printf("\n中序遍历\n");
    mid(allTreeNode[0]);
    printf("\n后序遍历\n");
    back(allTreeNode[0]);
    printf("\n\n");

    return 0;
}

涉及的队列和栈数据结构代码

cl_link.h

#ifndef _CL_LINK_H
#define _CL_LINK_H

#include 
#include 
#include 
#include 

#define ADD_SUCCESS (0)
#define ADD_FAIL (-1)
#define DELETE_SUCCESS (0)
#define SELETE_FAIL (-1)
#define CANFIND (1)
#define NOTFIND (0)

typedef struct cl_link_node cl_link_node;
typedef struct cl_link cl_link;



/**
 * 链表节点,其他结构体需要使用链表数据结构,只需包含此节点
 * @return
 */
typedef struct cl_link_node{
    cl_link_node* next;
    cl_link_node* prev;
}cl_link_node;

/**
 * 通用链表对象
 * @return
 */
typedef struct cl_link{
    pthread_mutex_t     cl_link_mutex;  //链表锁
    cl_link_node        cl_link_head;   //链表头
    cl_link_node        cl_link_tail;   //链表尾
    int                 sum;            //节点数
}cl_link;

#define cl_link_get_node(aim, type, node)       \
    (cl_link_node *) ((u_char *) aim + offsetof(type, node))


#define cl_link_get_data(aim, type, node)      \
    (type *) ((u_char *) aim - offsetof(type, node))

/**
 * 创建一个链表对象
 * @return 链表对象地址
 */
cl_link* cl_link_create();

/**
 * 链表的压栈操作
 * @param  link 链表对象
 * @param  node 新节点
 * @return      压栈状态
 */
int cl_link_push(cl_link* link, void* node);

/**
 * 链表的出栈操作
 * @param link 链表对象
 */
void* cl_link_pop(cl_link* link);
//获取栈顶元素
void* cl_link_get_head(cl_link* link);

/**
 * 对每个节点进行操作
 * @param link    链表对象
 * @param res     返回值
 * @param handler 处理函数
 */
void cl_link_each(cl_link* link, void* res[], void* (*handler)(void* node));

/**
 * 队尾添加元素
 * @param  link 队列对象
 * @param  node 新节点
 * @return      添加状态
 */
int cl_link_add_back(cl_link* link, void* node);

/**
 * 队头获取元素
 * @param  link 队列对象
 * @return      取得的元素
 */
void* cl_link_get_front(cl_link* link);

/**
 * 根据key查找节点
 * @param link      链表对象
 * @param key       关键字
 * @param condition 条件
 */
void* cl_link_find(cl_link* link, void* key, int (*condition)(void* node, void* key));


#endif

cl_link.c

#include "cl_link.h"

/**
 * 创建一个链表对象
 * @return 链表对象地址
 */
cl_link* cl_link_create()
{
    cl_link* link = malloc(sizeof(cl_link));
    if(link == NULL){
        write(STDERR_FILENO, "malloc error", sizeof("malloc error"));
        return NULL;
    }
    pthread_mutex_init(&(link->cl_link_mutex), NULL);

    pthread_mutex_lock(&(link->cl_link_mutex));

    link->cl_link_head.next = &(link->cl_link_tail);
    link->cl_link_head.prev = &(link->cl_link_tail);

    link->cl_link_tail.next = &(link->cl_link_head);
    link->cl_link_tail.prev = &(link->cl_link_head);

    link->sum = 0;

    pthread_mutex_unlock(&(link->cl_link_mutex));

    return link;
}

/**
 * 链表的压栈操作
 * @param  link 链表对象
 * @param  node 新节点
 * @return      压栈状态
 */
int cl_link_push(cl_link* link, void* node)
{
    cl_link_node* new_node = (cl_link_node*)node;
    pthread_mutex_lock(&(link->cl_link_mutex));
    if(link)
    {
        new_node->next = link->cl_link_head.next;
        link->cl_link_head.next->prev = new_node;

        link->cl_link_head.next = new_node;
        new_node->prev = &(link->cl_link_head);
        link->sum++;
        pthread_mutex_unlock(&(link->cl_link_mutex));
        return ADD_SUCCESS;
    }else{
        pthread_mutex_unlock(&(link->cl_link_mutex));
        return ADD_FAIL;
    }
}


/**
 * 链表的出栈操作
 * @param link 链表对象
 */
void* cl_link_pop(cl_link* link)
{
    pthread_mutex_lock(&(link->cl_link_mutex));
    if(link->sum){
        cl_link_node* aim = link->cl_link_head.next;
        link->cl_link_head.next = aim->next;
        aim->next->prev = &(link->cl_link_head);
        link->sum--;
        pthread_mutex_unlock(&(link->cl_link_mutex));
        return aim;
    }else{
        pthread_mutex_unlock(&(link->cl_link_mutex));
        return NULL;
    }
}

void* cl_link_get_head(cl_link* link)
{
    return link->cl_link_head.next;
}

/**
 * 对每个节点进行操作
 * @param link    链表对象
 * @param res     返回值
 * @param handler 处理函数
 */
void cl_link_each(cl_link* link, void* res[], void* (*handler)(void* node))
{
    pthread_mutex_lock(&(link->cl_link_mutex));
    int n = link->sum;
    void* r;

    cl_link_node* p = link->cl_link_head.next;
    cl_link_node* todo = p;
    while(n)
    {
        todo = p;
        p = p->next;
        r = handler(todo);
        if(res != NULL)
        {
            res[link->sum-n] = r;
        }
        n--;
    }
    pthread_mutex_unlock(&(link->cl_link_mutex));
}


/**
 * 队尾添加元素
 * @param  link 队列对象
 * @param  node 新节点
 * @return      添加状态
 */
int cl_link_add_back(cl_link* link, void* node)
{
    cl_link_node* new_node = (cl_link_node*)node;
    pthread_mutex_lock(&(link->cl_link_mutex));
    if(link)
    {
        new_node->next = &(link->cl_link_tail);
        new_node->prev = link->cl_link_tail.prev;

        link->cl_link_tail.prev->next = new_node;
        link->cl_link_tail.prev = new_node;
        link->sum++;
        pthread_mutex_unlock(&(link->cl_link_mutex));
        return ADD_SUCCESS;
    }else{
        pthread_mutex_unlock(&(link->cl_link_mutex));
        return ADD_FAIL;
    }
}

/**
 * 队头获取元素
 * @param  link 队列对象
 * @return      取得的元素
 */
void* cl_link_get_front(cl_link* link)
{
    return cl_link_pop(link);
}

/**
 * 根据key查找节点
 * @param link      链表对象
 * @param key       关键字
 * @param condition 条件
 */
void* cl_link_find(cl_link* link, void* key, int (*condition)(void* node, void* key))
{
    pthread_mutex_lock(&(link->cl_link_mutex));
    int n = link->sum;
    cl_link_node* p = link->cl_link_head.next;
    while(n)
    {
        if(condition(p, key) == CANFIND)
        {
            pthread_mutex_unlock(&(link->cl_link_mutex));
            return p;
        }

        p = p->next;
        n--;
    }
    pthread_mutex_unlock(&(link->cl_link_mutex));
    return NULL;
}

你可能感兴趣的:(算术表达式的实现,支持加减乘除,括号运算,表达式转二叉树)