一、题目链接

  http://codeforces.com/gym/100198/problem/G

二、题意

  看样例就能明白,写表达式解析器。

三 、思路

  一看这题目,立马就会想到“后缀表达式”,考虑到这里有变量,在把中缀表达式变为后缀表达式时,把变量替换成常量即可。注意,define 2 3 表示,下面的2用3来替换。和C语言是一样的。

PS:然而,我用Java写的,总是RE在第四个样例,实在找不到哪里抛异常了。各位同行,如果您能找到我的代码的bug,烦请您在评论区留下出bug的样例。谢谢!

四、源代码

import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.InputStream;
import java.io.PrintStream;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.Map;
import java.util.Scanner;
import java.util.Stack;

public class Main {
    static final String digitPattern = "-?\\d+";
    static final File fin = new File("plcool.in");
    static final File fout = new File("plcool.out");
    static String input;
    static ArrayList<String> inExp = new ArrayList<String>();
    static ArrayList<String> postExp = new ArrayList<String>();
    static Map<Character, Integer> prior = new HashMap<Character, Integer>();
    static Map<String, String> define = new HashMap<String, String>();
    static InputStream is = null;
    static PrintStream ps = null;
    static {
        prior.put('^', 3);
        prior.put('*', 2);
        prior.put('/', 2);
        prior.put('%', 2);
        prior.put('+', 1);
        prior.put('-', 1);
        prior.put('(', 0);
        prior.put(')', 0);
    }

    static String findRoot(String key) {
        if(!define.containsKey(key))return key;
        else return findRoot(define.get(key));
    }

    static int power(int x, int y) {
        int res = 1;
        while(y > 0) {
            if((y & 1) == 1)res *= x;
            x = x * x;
            y >>= 1;
        }
        return res;
    }

    static int mod(int x, int y) {
        int sign = ((x >>> 31) ^ (y >>> 31)) == 0 ? 1 : -1;
        return (Math.abs(x) % Math.abs(y)) * sign;
    }

    static void split() {
        inExp.clear();
        int s = 0, t = 0, len = input.length();
        boolean befOper = false;
        for (int i = 0; i < len; ++i) {
            if(Character.isDigit(input.charAt(i)) || (input.charAt(i) == '-' && (befOper || i == 0))) {
                for(s = i++;i < len && Character.isDigit(input.charAt(i)); ++i);
                inExp.add(input.substring(s, i));
                --i;
                befOper = false;
            }
            else if(Character.isLetter(input.charAt(i))) {
                for(s = i;i < len && (Character.isLetter(input.charAt(i))
                        || Character.isDigit(input.charAt(i))); ++i);
                inExp.add(input.substring(s, i));
                --i;
            }
            else if(prior.containsKey(input.charAt(i))) {
                inExp.add(input.charAt(i) + "");
                befOper = true;
            }
        }
        //inExp.forEach(e -> ps.print(e + " "));
        for (int i = 0, sz = inExp.size(); i < sz; ++i) {
            String now = inExp.get(i);
            if (define.containsKey(now)) {
                String root = findRoot(now);
                if(!root.matches(digitPattern))root = "0";
                inExp.set(i, root);
            }
            else if(!prior.containsKey(now.charAt(0)) && !now.matches(digitPattern))inExp.set(i, "0");
        }
        //inExp.forEach(e -> ps.print(e + " "));
    }

    static Stack<Character> opers = new Stack<Character>();
    static void toPost() {
        opers.clear();
        postExp.clear();
        for (String s : inExp) {
            if (s.matches(digitPattern))
                postExp.add(s);
            else {
                if (opers.isEmpty())
                    opers.push(s.charAt(0));
                else {
                    char top = opers.peek();
                    char c = s.charAt(0);
                    if (s.equals(")")){
                        while (!opers.isEmpty() && opers.peek() != '(') {
                            postExp.add(opers.pop() + "");
                        }
                        if (!opers.isEmpty() && opers.peek() == '(')
                            opers.pop();
                    }
                    else if((c != '(' && c != '^' && prior.get(c) <= prior.get(top))){
                        for(top = opers.peek();prior.get(c) <= prior.get(top);) {
                            postExp.add(opers.pop() + "");
                            if(!opers.isEmpty())top = opers.peek();
                            else break;
                        }
                        opers.push(c);
                    }
                    else
                        opers.push(c);
                }
            }
        }
        while(!opers.isEmpty()) {
            char c = opers.pop();
            if(c == ')')continue;
            else postExp.add(c + "");
        }
        //postExp.forEach(e -> ps.print(e + " "));
    }

    static Stack<Integer> digits = new Stack<Integer>();
    static int calcResult() {
        digits.clear();
        for(String s : postExp) {
            if(s.matches(digitPattern))digits.push(Integer.parseInt(s));
            else {
                char c = s.charAt(0);
                int last1 = digits.pop(), last2 = digits.pop();
                switch (c) {
                case '-':{
                    digits.push(last2 - last1);
                    break;
                }
                case '+':{
                    digits.push(last2 + last1);
                    break;
                }
                case '*':{
                    digits.push(last2 * last1);
                    break;
                }
                case '/':{
                    digits.push(last2 / last1);
                    break;
                }
                case '%':{
                    digits.push(mod(last2, last1));
                    break;
                }
                case '^':{
                    digits.push(power(last2, last1));
                    break;
                }
                }
            }
        }
        return digits.pop();
    }

    static String[] parts;
    public static void main(String[] args) throws Exception {
//        is = new FileInputStream(fin);
//        ps = new PrintStream(new FileOutputStream(fout));
        is = System.in;
        ps = System.out;
        Scanner scanner = new Scanner(is);
        while (scanner.hasNextLine()) {
            input = scanner.nextLine().trim().toLowerCase();
            if (input.charAt(0) == 'p') {
                input = input.replaceAll(" ", "");
                input = input.substring(5);
                split();
                toPost();
                int res = calcResult();
                ps.println(res);
            }

            else {
                parts = input.split(" ");
                if (define.containsKey(parts[1]))
                    continue;
                else {
                    String root1 = findRoot(parts[1]);
                    String root2 = findRoot(parts[2]);
                    if(!root1.equals(root2))define.put(parts[1], parts[2]);
                }
            }
        }
        scanner.close();
    }
}

CFGym 100198G 题解的更多相关文章

  1. CFGym 101490J 题解

    一.题目链接 http://codeforces.com/gym/101490 二.题面 三.题意 给你n个点,代表学生所在位置,n个点,代表老师所在位置.每个学生分配一个老师.让你找出一个最小的学生 ...

  2. CFGym 101490E 题解

    一.题目链接 http://codeforces.com/gym/101490 二.题面 三.题意 给你一个图,n个点,m条边,一个x,从顶点1走到顶点n.假设从顶点1走到顶点n的最短路为d,x代表你 ...

  3. CFGym 101161I 题解

    一.题目链接 http://codeforces.com/gym/101161/problem/I 二.题意 给定一棵树,一个初始的省会城市,若干个询问,0表示修改省会城市,1表示查询去省会必须经过指 ...

  4. CFGym 101194L 题解

    一.题目链接 http://codeforces.com/gym/101194/problem/L 二.题意 有4个队伍,要打6场比赛(刚好每两个队伍都能相互比一次),若A和B比赛有3种结果: A赢B ...

  5. CFGym 101194D 题解

    一.题目链接 http://codeforces.com/gym/101194/problem/D 二.题意 给定一个数字n和一个数字k,一个n个整数的序列,让你在里面找尽可能多的长度为k的符合“要求 ...

  6. CFGym 101505I 题解

    一.题目链接 http://codeforces.com/gym/101505 二.题意 这题其实主要就是题意,理解题意后,就是水题了.我想了下,主要原因就是这几点: 1.题意太过英文化,很多句子不能 ...

  7. CFGym 100211J 题解

    一.题目 二.题意 给定一个字母表(最多也就是英文小写字母的前10个字母),一个交换表,两个字符串,判断字符串A能否通过交换表的交换方式变成字符串B. 三.思路 1.一开始,比赛时,我半模拟半记忆化地 ...

  8. [CF-GYM]Abu Tahun Mod problem题解

    前言 这道题比较简单,但我还是想了好一会 题意简述 Abu Tahun很喜欢回文. 一个数组若是回文的,那么它从前往后读和从后往前读都是一样的,比如数组\(\left\{1\right\},\left ...

  9. 2016 华南师大ACM校赛 SCNUCPC 非官方题解

    我要举报本次校赛出题人的消极出题!!! 官方题解请戳:http://3.scnuacm2015.sinaapp.com/?p=89(其实就是一堆代码没有题解) A. 树链剖分数据结构板题 题目大意:我 ...

随机推荐

  1. 前端工程化 - bower

    什么是bower bower是前端的包管理工具,类似于php的composer,python的pip,虽然先如今bower已经停止了更新(主要是因为Browserify和Webpack等包管理工具的兴 ...

  2. BZOJ 4011 【HNOI2015】 落忆枫音

    题目链接:落忆枫音 以下内容参考PoPoQQQ大爷的博客 首先我们先来考虑一下如果没有新加入的那条边,答案怎么算. 由于这是一个\(DAG\),所以我们给每个点随便选择一条入边,最后一定会构成一个树形 ...

  3. gcd 与 扩gcd 总结

    gcd 定理的证明: 模板: ll gcd(ll a,ll b) { ) return a; else return gcd(b,a%b); } 扩gcd证明: 模板: ll extgcd(ll a, ...

  4. windows使用git记录

    1.免密码clone远程服务器代码开启ssh 生成私钥公钥 命令:查看自己配置的邮箱 git config user.name git config user.email 生成密钥:邮箱填写上面查看出 ...

  5. 【hive】cube和rollup函数

    cube 数据立方体(Data Cube),是多维模型的一个形象的说法.(关于多维模型这里不讲述,在数据仓库设计过程中还挺重要的,有兴趣自行查阅) 立方体其本身只有三维,但多维模型不仅限于三维模型,可 ...

  6. promise的基础知识

    promise 相当于异步操作结果的占位符 它不会去订阅一个事件,也不会传递一个回调函数给目标函数,而是让函数返回一个promise,例如: let promise = readFile('a.txt ...

  7. 干掉某个用户的所有进程 ---slay和kill

    要杀掉指定进程,你可以: 1.sudo slay <name> 知道进程名字即可 2.kill[参数][进程号]  知道进程ID即可 用‘slay’干掉某个用户的所有进程 slay 是Ch ...

  8. CF1082E:E.increasing Frequency(贪心&最大连续和)

    You are given array a a of length n n . You can choose one segment [l,r] [l,r] (1≤l≤r≤n 1≤l≤r≤n ) an ...

  9. Nginx下载和安装与启动

    nginx是什么 nginx是一个开源的,支持高性能,高并发的www服务和代理服务软件.它是一个俄罗斯人lgor sysoev开发的,作者将源代码开源出来供全球使用. nginx比它大哥apache性 ...

  10. bzoj 4177 Mike的农场

    bzoj 4177 Mike的农场 思维有些江化了,一上来就想费用流做法,但其实就是个最小割啊. 考虑先将所有的收益拿到,再减去不能拿的以及三元组 \((i,j,k)\) 产生的代价.即,先让 \(a ...