zoukankan      html  css  js  c++  java
  • 表达式转表达式二叉树

    表达式树

    二叉树是表达式处理的常用工具,例如,a+b*(c-d)-e/f可以表示成如下所示的二叉树

    其中,每个非叶子节点表示一个运算符,左子树是第一个运算数对应的表达式,右子树是第二个表达式对应的表达式。每个叶子节点都是数。

    其在空间利用上也非常高效,节点数等于表达式的长度。

    表达式转二叉树

    lrj说方法有很多种,下面介绍他讲的一种:找到“最后计算”的运算符(它是整个表达式树的根),然后递归处理左右两边。

     1 const int maxn = 1000 + 10; 
     2 char str[maxn];
     3 int lch[maxn + 1], rch[maxn + 1]; char op[maxn + 1];     //每个结点的左右子结点编号和字符
     4 int nc = 0;         //结点数 
     5 int build_tree(char* s, int x, int y) 
     6 {  
     7     int i, c1=-1, c2=-1, p=0;  
     8     int u; 
     9     if(y-x == 1)   //仅一个字符,建立单独结点 
    10     {        
    11         u = ++nc;    
    12         lch[u] = rch[u] = 0; 
    13         op[u] = s[x];   
    14         return u;
    15      } 
    16 
    17      for (i = x; i < y; i++)   //寻找根节点的位置
    18      {
    19          switch (s[i]) 
    20          {
    21             case '(':  p++; break;     
    22             case ')':  p--; break;      
    23             case '+':
    24             case '-':  if (!p) c1 = i; break;      
    25             case '*': 
    26             case '/':  if (!p) c2 = i; break;
    27          }
    28      }  
    29     if (c1 < 0) c1 = c2;     //找不到括号外的加减号,就用乘除号  
    30     if(c1 < 0) return build_tree(s, x+1, y-1);     //整个表达式被一对括号括起来  
    31     u = ++nc;  
    32     lch[u] = build_tree(s, x, c1); 
    33     rch[u] = build_tree(s, c1+1, y);  
    34     op[u] = s[c1];  
    35     return u;
    36 }

    前缀式、中缀式、后缀式

    前缀表达式和后缀表达式分别对应表达式树前序和后序遍历的结果,如果不考虑括号,中缀表达式对应表达式树中序遍历的结果。

     1 //中序遍历
     2 void InOrder(int root)
     3 {
     4     if (lch[root] > 0)  InOrder(lch[root]);
     5     printf("%c ", op[root]);
     6     if (rch[root] > 0)  InOrder(rch[root]);
     7 }
     8 
     9 //后序遍历
    10 void PostOrder(int root)
    11 {
    12     if (lch[root] > 0)  PostOrder(lch[root]);
    13     if (rch[root] > 0)  PostOrder(rch[root]);
    14     printf("%c ", op[root]);
    15 }

    表达式求值

    有了表达式树求值就非常简单了,先求左子树的值,再求右子树值,然后根据根节点的运算符计算最终的值。整个过程采用递归求解。

    (注意!!这种方法只能求个位数表达式的值)

     1 //根据表达式二叉树求值
     2 int cal(int root)
     3 {
     4     int ans = 0;
     5     char ch = op[root];
     6     if (isdigit(ch))  return ch - '0';
     7     switch (ch)
     8     {
     9     case '+':  ans = cal(lch[root]) + cal(rch[root]); break;
    10     case '-':  ans = cal(lch[root]) - cal(rch[root]); break;
    11     case '*':  ans = cal(lch[root]) * cal(rch[root]); break;
    12     case '/':  ans = cal(lch[root]) / cal(rch[root]); break;
    13     }
    14     return ans;
    15 }

    打印二叉树(括号表达法)

    root(lch,rch)像这样的格式,其中lch,rch本身也是这样的格式(递归定义)。整个过程类似于先序遍历,先打印根节点,再打印左子树,再打印右子树。

    注意格式,如果缺少左、右子树用空格代替(如root( ,rch)),如果都没有,则括号也省掉。

    //打印表达式二叉树
    void PrintTree(int root)
    {
        int flag = 0;        //标记是否有左右子树
        printf("%c", op[root]);
        if (lch[root] > 0)  flag += 1;
        if (rch[root] > 0)  flag += 2;
        if (flag == 0)  return;
        if (flag == 1)        //只有左子树
        {
            printf("(");
            PrintTree(lch[root]);
            printf(", )");
        }
        if (flag == 2)        //只有右子树
        {
            printf("( ,");
            PrintTree(rch[root]);
            printf(")");
        }
        if (flag == 3)        //左右子树都有
        {
            printf("(");
            PrintTree(lch[root]);
            printf(",");
            PrintTree(rch[root]);
            printf(")");
        }
    }

    最一个加一个认真写的 胡乱写的能实现嵌套括号、多位数相加减乘除的完整代码

      1 #include<stdio.h>
      2 #include<string>
      3 
      4 const int maxn = 1000 + 10; 
      5 char expr[maxn];        //原表达式
      6 int str[maxn];            //转换后的表达式
      7 int lch[maxn + 1], rch[maxn + 1],op[maxn + 1];     //每个结点的左右子结点编号和字符
      8 bool is_alpha1[maxn], is_alpha2[maxn];        //是否为操作符或括号,前者对s数组,后者对op数组
      9 int nc = 0;         //结点数 
     10 int cnt = 0;        //转换后字符串的长度
     11 
     12 // 把表达式exp转化成参数形式,并存到str中
     13 void analyse(char* expr)
     14 {
     15     int len = strlen(expr);
     16     int i = 0;
     17     while(i < len)
     18     {
     19         if (!isdigit(expr[i]))
     20         {
     21             str[cnt] = expr[i++];
     22             is_alpha1[cnt++] = true;
     23         }
     24         else
     25         {
     26             int tmp = 0;
     27             while (isdigit(expr[i]))
     28             {
     29                 tmp = tmp * 10 + expr[i] - '0';
     30                 i++;
     31             }
     32             str[cnt] = tmp;
     33             is_alpha1[cnt++] = false;
     34         }
     35     }
     36 }
     37 
     38 //表达式转表达式树
     39 int build_tree(int* s, int x, int y) 
     40 {  
     41     int i, c1=-1, c2=-1, p=0;  
     42     int u; 
     43     if(y-x == 1)   //仅一个字符,建立单独结点 
     44     {        
     45         u = ++nc;    
     46         lch[u] = rch[u] = 0; 
     47         op[u] = s[x];   
     48         if (is_alpha1[x])  is_alpha2[u] = true;
     49         return u;
     50      } 
     51 
     52      for (i = x; i < y; i++)   //寻找根节点的位置
     53      {
     54          if(s[i] == '(' && is_alpha1[i])   p++;
     55          if(s[i] == ')' && is_alpha1[i])   p--;
     56          if((s[i] == '+' || s[i] == '-') && is_alpha1[i])   if (!p) c1 = i;
     57          if((s[i] == '*' || s[i] == '/') && is_alpha1[i])    if (!p) c2 = i;
     58      }  
     59     if (c1 < 0) c1 = c2;     //找不到括号外的加减号,就用乘除号  
     60     if(c1 < 0) return build_tree(s, x+1, y-1);     //整个表达式被一对括号括起来  
     61     u = ++nc;  
     62     lch[u] = build_tree(s, x, c1); 
     63     rch[u] = build_tree(s, c1+1, y);  
     64     op[u] = s[c1];  
     65     if (is_alpha1[c1])  is_alpha2[u] = true;
     66     return u;
     67 }
     68 
     69 //先序遍历
     70 void PreOrder(int root)
     71 {
     72     if(!is_alpha2[root])  printf("%d ", op[root]);
     73     else  printf("%c ", op[root]);
     74 
     75     if (lch[root] > 0)  PreOrder(lch[root]);
     76     if (rch[root] > 0)  PreOrder(rch[root]);
     77 }
     78 
     79 //中序遍历
     80 void InOrder(int root)
     81 {
     82     if (lch[root] > 0)  InOrder(lch[root]);
     83 
     84     if (!is_alpha2[root])  printf("%d ", op[root]);
     85     else  printf("%c ", op[root]);
     86 
     87     if (rch[root] > 0)  InOrder(rch[root]);
     88 }
     89 
     90 //后序遍历
     91 void PostOrder(int root)
     92 {
     93     if (lch[root] > 0)  PostOrder(lch[root]);
     94     if (rch[root] > 0)  PostOrder(rch[root]);
     95 
     96     if (!is_alpha2[root])  printf("%d ", op[root]);
     97     else  printf("%c ", op[root]);
     98 }
     99 
    100 //根据表达式二叉树求值
    101 int cal(int root)
    102 {
    103     int ans = 0;
    104     int ch = op[root];
    105     if (!is_alpha2[root])  return ch;
    106     switch (ch)
    107     {
    108     case '+':  ans = cal(lch[root]) + cal(rch[root]); break;
    109     case '-':  ans = cal(lch[root]) - cal(rch[root]); break;
    110     case '*':  ans = cal(lch[root]) * cal(rch[root]); break;
    111     case '/':  ans = cal(lch[root]) / cal(rch[root]); break;
    112     }
    113     return ans;
    114 }
    115 
    116 //打印表达式二叉树
    117 void PrintTree(int root)
    118 {
    119     int flag = 0;        //标记是否有左右子树
    120     if (!is_alpha2[root])  printf("%d", op[root]);
    121     else  printf("%c", op[root]);
    122 
    123     if (lch[root] > 0)  flag += 1;
    124     if (rch[root] > 0)  flag += 2;
    125     if (flag == 0)  return;
    126     if (flag == 1)        //只有左子树
    127     {
    128         printf("(");
    129         PrintTree(lch[root]);
    130         printf(", )");
    131     }
    132     if (flag == 2)        //只有右子树
    133     {
    134         printf("( ,");
    135         PrintTree(rch[root]);
    136         printf(")");
    137     }
    138     if (flag == 3)        //左右子树都有
    139     {
    140         printf("(");
    141         PrintTree(lch[root]);
    142         printf(",");
    143         PrintTree(rch[root]);
    144         printf(")");
    145     }
    146 }
    147 
    148 int main()
    149 {
    150     printf("表达式:");
    151     scanf("%s", expr);
    152 
    153     analyse(expr);   //转化
    154 
    155     
    156     build_tree(str, 0, cnt);        //建树
    157 
    158     int root = 1;                            
    159     printf("先序遍历:"); PreOrder(root); printf("
    ");            //三种遍历
    160     printf("中序遍历:"); InOrder(root); printf("
    ");
    161     printf("后序遍历:"); PostOrder(root); printf("
    ");
    162 
    163     printf("表达式二叉树:"); PrintTree(root); printf("
    ");        //打印表达式二叉树
    164 
    165     int ans;
    166     ans = cal(root);                //表达式求值
    167     printf("表达式值:%d
    ", ans);
    168 
    169     return 0;
    170 }
  • 相关阅读:
    数学基础详解 1——微积分
    logistic回归梯度上升优化算法
    决策树
    西瓜书学习笔记(1)——模型评估与选择
    关于map与set的一点理解;
    STL中的set容器的一点总结(转)
    HDOJ 题目分类
    Train Problem I(栈)
    猜数字(规律)
    Change the ball(找规律)
  • 原文地址:https://www.cnblogs.com/lfri/p/9985993.html
Copyright © 2011-2022 走看看