SWUST OJ 1042: 中缀表达式转换为后缀表达式【表达式转逆波兰表达式】

题目描述

中缀表达式是一个通用的算术或逻辑公式表示方法,操作符是以中缀形式处于操作数的中间(例:3 + 4),中缀表达式是人们常用的算术表示方法。后缀表达式不包含括号,运算符放在两个运算对象的后面,所有的计算按运算符出现的顺序,严格从左向右进行(不再考虑运算符的优先规则,如:(2 + 1) * 3 , 即2 1 + 3 *。利用栈结构,将中缀表达式转换为后缀表达式。(测试数据元素为单个字符)

输入

中缀表达式

输出

后缀表达式

样例输入复制

A+(B-C/D)*E

样例输出复制

ABCD/-E*+

这个题我看到解法当中有通过编译原理当中的文法表达式来处理的,也有用堆栈模拟的。

我的方案是用递归。

重点是要把表达式按优先级进行切割。
我是这么想的:
对于表达式:A+B*C-(D*F*(F+B+C*D)-(A*H))*T
先按优先级最低的+、-号进行切分,但是主要切分时要把括号内的表达式当成整体,于是可以切割成:
A
B*C
(D*F*(F+B+C*D)-(A*H))*T
三个子表达式
【如果+、-号无法切割,就扫描*、/能不能切割,还是不能就说明要么这个表达式只有一个原子,要么就是整个表达式都用括号包起来了。】
这里我们可以继续递归切割,A已经是原子,无需切割。第二个表达式可以继续切割成B和C
这样,每次切割,我们可以得到两个列表,一个用来装切割的表达式,一个用来装两个表达式之间的符号。
最后,假设我们的处理函数为f
那么对于这个例子,实际上我要求:f( A+B*C-(D*F*(F+B+C*D)-(A*H))*T)
根据刚才的分析,显然他可以通过递归化简为: f((D*F*(F+B+C*D)-(A*H))*T)f(A)f(B*C)+-
这样不断的递归f当中的表达式,最后合并答案就是我们需要的逆波兰表达式!
可以观看代码:
#define _CRT_SECURE_NO_WARNINGS
#include<stdio.h>
#include<string.h>
#include<math.h>
#include<algorithm>
#include<vector>
#include<string>
#include<map>
#include<queue>
#include<iostream>
#include<list>
#include<set>
#include<stack>

using namespace std;

bool expr_cut(string root_expr,vector<string>& expr_list,vector<char>& op_list)
{
    stack<int> s;
    int len=root_expr.size();
    if(len==1)
        return true;
    string expr_now;
    bool first_flag=false,second_flag=false;
    for(int i=0;i<len;++i)
    {
        if((root_expr[i]=='+' || root_expr[i]=='-') && s.empty())
        {
            expr_list.push_back(expr_now);
            op_list.push_back(root_expr[i]);
            expr_now.clear();
            first_flag=true;
            continue;
        }
        else if(root_expr[i]=='(')
        {
            s.push(i);
        }
        else if(root_expr[i]==')')
        {
            s.pop();
        }
        expr_now=expr_now+root_expr[i];
    }
    expr_list.push_back(expr_now);
    if(first_flag)
        return false;
    expr_list.clear(),op_list.clear(),expr_now.clear();
    for(int i=0;i<len;++i)
    {
        if((root_expr[i]=='*' || root_expr[i]=='/') && s.empty())
        {
            expr_list.push_back(expr_now);
            op_list.push_back(root_expr[i]);
            expr_now.clear();
            second_flag=true;
            continue;
        }
        else if(root_expr[i]=='(')
        {
            s.push(i);
        }
        else if(root_expr[i]==')')
        {
            s.pop();
        }
        expr_now=expr_now+root_expr[i];
    }
    expr_list.push_back(expr_now);
    if(second_flag)
        return false;
    expr_list.clear(),op_list.clear(),expr_now.clear();
    root_expr=root_expr.substr(1,root_expr.size()-2);
    return expr_cut(root_expr,expr_list,op_list);
}

string f(string root_expr)
{
    vector<string> expr_list;
    vector<char> op_list;
    bool is_atom=expr_cut(root_expr,expr_list,op_list);
    if(is_atom)
        return root_expr;
    else
    {
        string result="";
        string first_item=f(expr_list[0]);
        for(int i=1;i<expr_list.size();i++)
        {
            string second_item=f(expr_list[i]);
            result=first_item+second_item+result+op_list[i-1];
            first_item=second_item;
        }
        return result;
    }
}

int main()
{
    string expr;
    while(cin>>expr)
    {
        string result=f(expr);
        cout<<result;
    }
    return 0;
}

/*
A+B*C-(D*F*(F+B+C*D)-(A*H))*T
(A+B*C)
*/

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 题目描述: 输入一个字符串,将其中的每个单词翻后输出。 输入格式: 一个字符串,长度不超过100。 输出格式: 翻后的字符串。 样例输入: hello world 样例输出: olleh dlrow 解题思路: 先将整个字符串翻,然后再将每个单词翻。 代码实现: ### 回答2: 博客园链接:https://www.cnblogs.com/Tomorrow-I-Love-You/p/14562869.html 题目描述 在本题中,将给出一个长度不超过 $100$ 个字符的字符串。 请你将这个字符串中的单词翻过来,标点可以不处理,但是要保证单词内部正确的顺序。 请注意,在一些编程语言中(如 C++),存在着一些规定中的数组边界处理错误,如果您的程序产生了这样的错误,您只需要考虑修正这个规定中的错误而不需要花费更多的时间在调试上面。 输入格式 输入数据在第一行有一个正整数 $n$。 接下来有 $n$ 行,每行表示一个字符串。 输出格式 输出每组数据的结果,每个字符串占一行。 样例输入 3 hello world. how are you? i'm fine, thank you. 样例输出 olleh dlrow. woh era ?uoy i'm enif, uoyknaht. 算法 字符串 参考文献 无 讲解 第一次写独立完成的题解,如有错误,敬请指正。这一题思想非常简单,就是考虑如何将字符串中单词顺序颠倒,具体请看ac代码。$\color{#EEC900}code$ ### 回答3: SWUST OJ1162: 单词翻,是一道经典的字符串处理题目。本题需要程序员编写程序,输入一行英文句子,并将每个单词翻输出,但是所有非字母字符(包括空格、逗号、句号等)必须保持原来的位置不变。 解题思路主要分为以下几个步骤: 1. 输入字符串 首先,我们需要输入一行英文句子,可以使用字符串的输入输出函数,如scanf或gets函数。在输入的同时,要考虑单词中间的空格和标点符号。 2. 遍历字符串 我们需要使用循环语句,遍历整个字符串,找到每个单词的首尾位置,以供后续反。 3. 反单词 使用指针或字符串操作函数(如strrev函数),将每个单词反。注意,在反字符串的同时,还需要保留所有非字母字符。 4. 输出结果 最后,我们需要输出经过反后的整个句子。 在代码实现过程中,需要注意以下几个细节: 1. 所有非字母字符必须保持原来的位置不变。可以使用判断函数(如isalpha函数),判断字符是否为字母。 2. 单词的首尾位置可以使用指针或者字符串操作函数(如strtok函数)实现。 3. 字符串反时,需要考虑边界情况,如字符串为空或者只有一个字符。 综上所述,本题是一道字符串处理的基础性题目,可以帮助程序员提高对字符串操作的熟练度和实践能力,对于进一步深入学习算法和数据结构也具有一定的启发作用。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值