【kuangbin带你飞-区间DP-2】 非常好的dfs+dp题 CodeForces - 149 D-Coloring Brackets

题目链接:https://codeforces.com/contest/149/problem/D

题意:给你一个合法的括号序列,现在让你给这个序列染色,染色的条件如下:

  1. 每对()有且只有一个括号被染色
  2. 相邻的括号如果都被染色了,那么其颜色不能相同
  3. 每个括号只能涂蓝色,红色,或者不涂任何颜色

现在让你求染色方案总数,并且最后结果mod 1e9+7

算法思路:

  1. 依据每对括号进行dp,因为括号存在嵌套,所以需要使用dfs进行递归处理
  2. 为了知道每个左括号对应右括号,需要提前进行预处理
  3. dp状态方程为dp[l][r][i][j] 表示区间【l,r】的左右两个端点分别涂颜色i和颜色j方案数
  4. 需要注意区间是一对括号和不是一对括号这两种情况对应的染色方案不一样。注意状态转移的那部分如果不确定一定是一个区间端点匹配的情况,都染色情况的限制都较弱于区间端点配对的情况,详情见代码

代码:

#include<bits/stdc++.h>
#define ll long long
using namespace std;
const int maxn = 800;
const ll mod = 1e9 + 7;
ll dp[maxn][maxn][3][3];
string s;
ll tail[maxn];//tail[i]等于某个位置的左括号对应的右括号的位置
void dfs(int l, int r) {
    if(l >= r)
        return;
    if(l + 1 == r) {
        for(int i = 0; i < 3; i++)
            for(int j = 0; j < 3; j++) {
                if(i != 0 && j != 0 || (i == 0 && j == 0))
                    continue;
                dp[l][r][i][j] = 1;
            }
        return;
    }
    if(tail[l] == r) {
        dfs(l + 1, r - 1);
        for(int i = 0; i < 3; i++) {
            for(int j = 0; j < 3; j++) {
                if(i == 0 && j == 0 || (i != 0 && j != 0))
                    continue;
                for(int k = 0; k < 3; k++) {
                    for(int m = 0; m < 3; m++) {
                        if((k != 0 && k == i) || (m != 0 && m == j))
                            continue;
                        dp[l][r][i][j] = (dp[l][r][i][j] + dp[l + 1][r - 1][k][m]) % mod;
                    }
                }
            }
        }
    } else {
        dfs(l, tail[l]);
        dfs(tail[l] + 1, r);
        for(int i = 0; i < 3; i++) {
            for(int j = 0; j < 3; j++) {
                if(i == 0 && j == 0 || (i != 0 && j != 0))
                    continue;
                for(int k = 0; k < 3; k++) {
                    if(k != 0 && k == j)
                        continue;
                    for(int m = 0; m < 3; m++) {
                        dp[l][r][i][m] = (dp[l][r][i][m] + (dp[l][tail[l]][i][j] * dp[tail[l] + 1][r][k][m]) % mod) % mod;
                    }
                }
            }
        }
    }
}
int main() {
    ios::sync_with_stdio(false);//优化
    cin >> s;
    ll n = s.size();
    stack<ll> st;
    //预处理
    for(ll i = 0; i < s.size(); i++) {
        if(s[i] == '(')
            st.push(i);
        else {
            tail[st.top()] = i;
            st.pop();
        }
    }
    dfs(0, n - 1);
    ll res = 0;
    for(ll i = 0; i < 3; i++) {
        for(ll j = 0; j < 3; j++) {
            res = (res + dp[0][n - 1][i][j]) % mod;
        }
    }
    cout << res << endl;
}

 

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值