CSP历年复赛题-P7073 [CSP-J2020] 表达式

原题链接:https://www.luogu.com.cn/problem/P7073

题意解读:给定一个后缀表达式,其中变量都有初始值,q个询问,每次将一个变量取反,求后缀表达式的结果。

解题思路:

1、堆栈模拟法

我们知道,对于后缀表达式,可以借助堆栈进行运算,依次读取操作数和操作符,如果是操作数则入栈,如果是操作符则弹出栈顶1-2个元素进行运算(& |需要两个操作数,!只要一个操作数),运算结果再次入栈,最后栈顶即为表达式结果。

由于表达式最长10^6,q最大10^5,超时是一定的。

那么问题就集中在字符串处理,具体看代码。

30分代码:

#include <bits/stdc++.h>
using namespace std;

const int N = 100005;
string str;
int n, q;
int x[N];

int main()
{
    getline(cin, str);
    cin >> n;
    for(int i = 1; i <= n; i++) cin >> x[i]; //所有变量初始值

    int t;
    cin >> q;
    while(q--)
    {
        cin >> t;
        //变量取反
        x[t] = !x[t];
        //后缀表达式计算
        stack<int> stk;
        for(int i = 0; i < str.length(); i++)
        {
            if(str[i] == 'x')
            {
                int num = 0;
                int j = i + 1; //读变量x后面的数字
                while(str[j] >= '0' && str[j] <= '9')
                {
                    num = num * 10 + str[j] - '0';
                    j++;
                }
                stk.push(x[num]); //变量值入栈
                i = j - 1;
            }
            else if(str[i] == '&')
            {
                int a = stk.top(); stk.pop();
                int b = stk.top(); stk.pop();
                int c = a & b;
                stk.push(c);
            }
            else if(str[i] == '|')
            {
                int a = stk.top(); stk.pop();
                int b = stk.top(); stk.pop();
                int c = a | b;
                stk.push(c);
            }
            else if(str[i] == '!')
            {
                int a = stk.top(); stk.pop();
                int c = !a;
                stk.push(c);
            }
        }
        cout << stk.top() << endl;
        //变量恢复
        x[t] = !x[t];
    }
}

2、表达式树

对于任何一个中缀表达式,都可以转化为一棵树形结构,如样例中:

x1 x2 & x3 | 对应的中缀表达式为 x1 & x2 | x3

转化为树形结构为:

可以看出,变量都是叶子节点,操作符都是中间节点或根节点,根节点的值就是表达式的值。

接下来要解决三个关键问题:

第一、如何构建表达式树?

通过堆栈计算后缀表达式的过程,可以提取出变量和操作符,变量编号是树中的节点,操作符编号(在最大变量编号基础上递增)也是树中的节点,计算之后的结果关联在操作符节点上,如此最终就能构建出一棵树,最后栈顶的元素就是根节点。

由于&/|的子节点有两个,!的子节点只有一个,可以直接采用邻接表来存储树形结构。

    cnt = n; //节点编号跳过变量编号
    for(int i = 0; i < str.length(); i++)
    {
        if(str[i] == 'x')
        {
            int num = 0;
            int j = i + 1; //读变量x后面的数字
            while(str[j] >= '0' && str[j] <= '9')
            {
                num = num * 10 + str[j] - '0';
                j++;
            }
            stk.push(num); //变量编号入栈
            i = j - 1;
        }
        else if(str[i] == '&' || str[i] == '|')
        {
            int a = stk.top(); stk.pop();
            int b = stk.top(); stk.pop();
            cnt++; //符号节点累加
            stk.push(cnt); //符号节点编号入栈
            tree[cnt].push_back(a); //节点cnt和a之间建立一条边
            tree[cnt].push_back(b); //节点cnt和b之间建立一条边
            o[cnt] = str[i];
        }
        else if(str[i] == '!')
        {
            int a = stk.top(); stk.pop();
            cnt++;
            stk.push(cnt);
            tree[cnt].push_back(a);
            o[cnt] = '!';
        }
    }

第二、如何计算树中每个节点的值?

从根节点出发,通过递归对所有子树的值,用操作符进行计算。

int dfs1(int root)
{
    if(root <= n) return v[root]; //变量叶子节点直接返回变量值
    if(o[root] == '&') return v[root] = dfs1(tree[root][0]) & dfs1(tree[root][1]);
    if(o[root] == '|') return v[root] = dfs1(tree[root][0]) | dfs1(tree[root][1]);
    if(o[root] == '!') return v[root] = !dfs1(tree[root][0]);
}

第三、将某个叶子节点的值取反,根节点的值是否改变?

如果每次都将某个叶子节点值取反,然后利用dfs1重新算一次表达式的值,事件复杂度也是nq。

其实,只需要计算一次初始表达式的值dfs1

然后需要看哪些变量的变化会影响表达式的值即可,如果影响则表达式的值取反,如果不影响表达式的值不变。

如何来标记哪些变量的变化会影响结果呢?

我们引入dfs2,核心思想是从根节点出发,标记每个对结果能产生影响的节点

如果当前符号是&,如果左子树是1,则标记右子树会对结果产生影响,反之如果右子树是1,则标记左子树会对结果产生影响;

如果当前符号是|,如果左子树是0,则标记右子树会对结果产生影响,反之如果右子树是0,则标记左子树会对结果产生影响;

如果当前符号是!,则标记子节点会对结果产生影响。

void dfs2(int root)
{
    flag[root] = true;
    if(o[root] == '&')
    {
        if(v[tree[root][0]] == 1) dfs2(tree[root][1]); //如果左子树值是1,则标记右子树的值会影响表达式结果
        if(v[tree[root][1]] == 1) dfs2(tree[root][0]); //如果右子树值是1,则标记左子树的值会影响表达式结果
    }
    if(o[root] == '|')
    {
        if(v[tree[root][0]] == 0) dfs2(tree[root][1]); //如果左子树结果是0,则标记右子树的值会影响表达式结果
        if(v[tree[root][1]] == 0) dfs2(tree[root][0]); //如果右子树结果是0,则标记左子树的值会影响表达式结果
    }
    if(o[root] == '!') dfs2(tree[root][0]); //子树的值肯定影响结果
}

100分代码:

#include <bits/stdc++.h>
using namespace std;

const int N = 1000005; //树的节点比变量数多,但肯定不超过10^6
string str;
int n, q, cnt;
vector<int> tree[N]; 
char o[N]; //树中每个节点对应的操作符,如果有
int v[N]; //树中每个节点的值
stack<int> stk;
bool flag[N]; //标记某个变量的值的变化是否影响最终的结果

//计算表达式的值
int dfs1(int root)
{
    if(root <= n) return v[root]; //变量叶子节点直接返回变量值
    if(o[root] == '&') return v[root] = dfs1(tree[root][0]) & dfs1(tree[root][1]);
    if(o[root] == '|') return v[root] = dfs1(tree[root][0]) | dfs1(tree[root][1]);
    if(o[root] == '!') return v[root] = !dfs1(tree[root][0]);
}

//标记所有节点的变化会影响表达式的结果
void dfs2(int root)
{
    flag[root] = true;
    if(o[root] == '&')
    {
        if(v[tree[root][0]] == 1) dfs2(tree[root][1]); //如果左子树值是1,则标记右子树的值会影响表达式结果
        if(v[tree[root][1]] == 1) dfs2(tree[root][0]); //如果右子树值是1,则标记左子树的值会影响表达式结果
    }
    if(o[root] == '|')
    {
        if(v[tree[root][0]] == 0) dfs2(tree[root][1]); //如果左子树结果是0,则标记右子树的值会影响表达式结果
        if(v[tree[root][1]] == 0) dfs2(tree[root][0]); //如果右子树结果是0,则标记左子树的值会影响表达式结果
    }
    if(o[root] == '!') dfs2(tree[root][0]); //子树的值肯定影响结果
}

int main()
{
    getline(cin, str);
    cin >> n;
    for(int i = 1; i <= n; i++) cin >> v[i]; //所有变量初始值

    cnt = n; //节点编号跳过变量编号
    for(int i = 0; i < str.length(); i++)
    {
        if(str[i] == 'x')
        {
            int num = 0;
            int j = i + 1; //读变量x后面的数字
            while(str[j] >= '0' && str[j] <= '9')
            {
                num = num * 10 + str[j] - '0';
                j++;
            }
            stk.push(num); //变量编号入栈
            i = j - 1;
        }
        else if(str[i] == '&' || str[i] == '|')
        {
            int a = stk.top(); stk.pop();
            int b = stk.top(); stk.pop();
            cnt++; //符号节点累加
            stk.push(cnt); //符号节点编号入栈
            tree[cnt].push_back(a); //节点cnt和a之间建立一条边
            tree[cnt].push_back(b); //节点cnt和b之间建立一条边
            o[cnt] = str[i];
        }
        else if(str[i] == '!')
        {
            int a = stk.top(); stk.pop();
            cnt++;
            stk.push(cnt);
            tree[cnt].push_back(a);
            o[cnt] = '!';
        }
    }
    int res = dfs1(stk.top());
    dfs2(stk.top());

    int t;
    cin >> q;
    while(q--)
    {
        cin >> t;
        if(flag[t]) cout << !res << endl;
        else cout << res << endl;
    }

    return 0;
}

 

posted @ 2024-06-14 15:33  五月江城  阅读(21)  评论(0编辑  收藏  举报