CF149D Coloring Brackets
CF149D Coloring Brackets
题面:
给出一个配对的括号序列(如"\((())()\)"、"\(()\)"等, "\()()\)"、"\((()\)"是不符合要求的 ),对该序列按以下方法进行染色:
- 1.一个括号可以染红色、蓝色或不染色
- 2.一对匹配的括号需要且只能将其中一个染色
- 3.相邻两个括号颜色不能相同(但可以都不染色) 求符合条件的染色方案数(对1000000007取模)
输入:一行,表示括号序列
输出:一个数表示方案数(对1000000007取模) 数据范围:
这道题转移很恶心,并且也比较难想。是一道很仙的区间 \(dp\) 题。
想了半天,发现只会打爆搜,最后看了看题解 才把这题搞懂
按照区间dp的套路我们还是设 \(f[l][r]\) 表示从 \(l\) 刷到 \(r\) 的合法方案数。
这是,我们还要填上一维颜色,因为限制是和颜色相关的。
\(f[l][r][0,1,2][0,1,2]\) 表示从\(l\) 刷到 \(r\) 且 \(l\) 不涂/涂红色/涂蓝色, \(r\) 不涂/涂红色/涂蓝色的方案数。
边界 \(l +1 == r\):
\(l\) 和 \(r\) 不配对,就是这种情况 \("(("\) ,就要保证 \(l\) 和 \(r\) 的颜色不同就行,还要算上他们两个都不涂的情况。
if(match[l] != r) f[l][r][0][0] = f[l][r][0][1] = f[l][r][0][2] = f[l][r][1][0] = f[l][r][2][0] = 1
\(l\) 和 \(r\) 配对的情况,就不能算 两个都不涂的情况
if(match[l] == r) f[l][r][0][1] = f[l][r][0][2] = f[l][r][1][0] = f[l][r][2][0] = 1
转移的时候:
\(l\) 和 \(r\) 匹配的时候,也就是 \("(....)"\) 的情况
我们就只需要考虑 \(l\) 和 \(l+1\) 以及 \(r-1\) 和 \(r\) 颜色不相同的情况,大力枚举 \(l+1\) 和 \(r\) 的颜色就行。
至于为什么不枚举 \(l\) 和 \(r\) 的颜色,因为那样写太复杂了,你完全可以只考虑 \(l\) 和 \(r\) 的颜色符合限制的方案数。
至于其他 \(l\) 和 \(r\) 的颜色情况都不符合条件,可以直接赋为 \(0\)
Code
for(int i = 0; i <= 2; i++)//l+1的颜色
{
for(int j = 0; j <= 2; j++)//r-1的颜色
{//注意相邻的颜色不能相同,也就是不能发生转移
if(j != 1) f[l][r][0][1] = (f[l][r][0][1] + f[l+1][r-1][i][j]) % p;
if(j != 2) f[l][r][0][2] = (f[l][r][0][2] + f[l+1][r-1][i][j]) % p;
if(i != 1) f[l][r][1][0] = (f[l][r][1][0] + f[l+1][r-1][i][j]) % p;
if(i != 2) f[l][r][2][0] = (f[l][r][2][0] + f[l+1][r-1][i][j]) % p;
}
}
\(l\) 和 \(r\) 不匹配的时候,也就是\("()()"\) 的情况
这个我们把它分为两个小区间 \(l -> match[l]\) 以及 \({match[l]+1} -> r\)
就可以递归求解了,至于为什么要分成这两个小区间。
因为你分成别的区间的话,要考虑的情况比较多,你要考虑 与 \(l\) 相邻的情况以及和他 匹配的那个括号的情况。
这样就可以少讨论与 \(l\) 相邻的情况,只需要考虑 \(match[l]\) 与 \(match[l]+1\) 相邻的情况。
然后大力枚举一下这四个点的颜色就可以了,只不过写起来有点费劲
Code
for(int i = 0; i <= 2; i++)//l的颜色
{
for(int j = 0; j <= 2; j++)//r的颜色
{
for(int k = 0; k <= 2; k++)//match[l] 的颜色
{
for(int u = 0; u <= 2; u++)//match[l+1] 的颜色
{
if(k == u && k != 0 && u != 0) continue;//相邻的颜色不能相同,但可以都不染色
f[l][r][i][j] = (f[l][r][i][j] + f[l][match[l]][i][k] * f[match[l]+1][r][j][u] % p) % p;
}
}
}
}
至于这个循环考虑了 \(l\) 和 \(match[l]\) 都不涂的情况,但那种情况的方案数为 \(0\) ,在乘另一个数对答案没有贡献。
最后的答案就是 \(\displaystyle \sum_{i=0}^2 \sum_{j=0}^{2} f[1][n][i][j]\)。 枚举一下起点和终点的颜色就可以了。
Code(我写的是记忆化搜索的放法)
当然了,你也可以写平常的那种写法。
#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
using namespace std;
#define int long long
const int p = 1e9+7;
char s[710];
int n,top,ans;
int sta[710],match[710],f[710][710][3][3];
void slove(int l,int r)
{
if(l+1 == r)
{
if(match[l] != r) f[l][r][0][0] = 1;
f[l][r][0][1] = f[l][r][0][2] = f[l][r][1][0] = f[l][r][2][0] = 1;
return;
}
if(match[l] == r)//()
{
slove(l+1,r-1);
for(int i = 0; i <= 2; i++)//l+1的颜色
{
for(int j = 0; j <= 2; j++)//r-1的颜色
{
if(j != 1) f[l][r][0][1] = (f[l][r][0][1] + f[l+1][r-1][i][j]) % p;
if(j != 2) f[l][r][0][2] = (f[l][r][0][2] + f[l+1][r-1][i][j]) % p;
if(i != 1) f[l][r][1][0] = (f[l][r][1][0] + f[l+1][r-1][i][j]) % p;
if(i != 2) f[l][r][2][0] = (f[l][r][2][0] + f[l+1][r-1][i][j]) % p;
}
}
}
else//()....()
{
slove(l,match[l]); slove(match[l]+1,r);
for(int i = 0; i <= 2; i++)//l的颜色
{
for(int j = 0; j <= 2; j++)//r的颜色
{
for(int k = 0; k <= 2; k++)//match[l] 的颜色
{
for(int u = 0; u <= 2; u++)//match[l+1] 的颜色
{
if(k == u && k != 0 && u != 0) continue;//相邻的颜色不能相同,但可以都不染色
f[l][r][i][j] = (f[l][r][i][j] + f[l][match[l]][i][k] * f[match[l]+1][r][j][u] % p) % p;
}
}
}
}
}
}
signed main()
{
scanf("%s",s+1);
n = strlen(s+1);
for(int i = 1; i <= n; i++)
{
if(s[i] == '(')
{
sta[++top] = i;
}
else if(s[i] == ')')
{
int t = sta[top--];
match[t] = i;
match[i] = t;
}
}
slove(1,n);
for(int i = 0; i <= 2; i++)
{
for(int j = 0; j <= 2; j++)
{
ans = (ans + f[1][n][i][j]) % p;;
}
}
printf("%lld\n",ans);
return 0;
}
CF149D Coloring Brackets的更多相关文章
- CF149D. Coloring Brackets[区间DP !]
题意:给括号匹配涂色,红色蓝色或不涂,要求见原题,求方案数 区间DP 用栈先处理匹配 f[i][j][0/1/2][0/1/2]表示i到ji涂色和j涂色的方案数 l和r匹配的话,转移到(l+1,r-1 ...
- CodeForces 149D Coloring Brackets
Coloring Brackets time limit per test: 2 seconds memory limit per test: 256 megabytes input: standar ...
- Codeforces Round #106 (Div. 2) D. Coloring Brackets 区间dp
题目链接: http://codeforces.com/problemset/problem/149/D D. Coloring Brackets time limit per test2 secon ...
- Codeforces 149D Coloring Brackets(树型DP)
题目链接 Coloring Brackets 考虑树型DP.(我参考了Q巨的代码还是略不理解……) 首先在序列的最外面加一对括号.预处理出DFS树. 每个点有9中状态.假设0位不涂色,1为涂红色,2为 ...
- Codeforces Round #106 (Div. 2) D. Coloring Brackets —— 区间DP
题目链接:https://vjudge.net/problem/CodeForces-149D D. Coloring Brackets time limit per test 2 seconds m ...
- codeforces 149D Coloring Brackets (区间DP + dfs)
题目链接: codeforces 149D Coloring Brackets 题目描述: 给一个合法的括号串,然后问这串括号有多少种涂色方案,当然啦!涂色是有限制的. 1,每个括号只有三种选择:涂红 ...
- CF 149D Coloring Brackets(区间DP,好题,给配对的括号上色,求上色方案数,限制条件多,dp四维)
1.http://codeforces.com/problemset/problem/149/D 2.题目大意 给一个给定括号序列,给该括号上色,上色有三个要求 1.只有三种上色方案,不上色,上红色, ...
- Coloring Brackets (区间DP)
Once Petya read a problem about a bracket sequence. He gave it much thought but didn't find a soluti ...
- CF 149D Coloring Brackets 区间dp ****
给一个给定括号序列,给该括号上色,上色有三个要求 1.只有三种上色方案,不上色,上红色,上蓝色 2.每对括号必须只能给其中的一个上色 3.相邻的两个不能上同色,可以都不上色 求0-len-1这一区间内 ...
随机推荐
- Promise、Generator,Async/await
我们知道JavaScript是单线程语言,如果没有异步编程非得卡死. 以前,异步编程的方法有下面四种 回调函数 事件监听 发布/订阅 Promise对象 现在据说异步编程终极解决方案是——async/ ...
- 浅谈 FTP、FTPS 与 SFTP
无论是网盘还是云存储,上传都是一项很简单的操作.那些便捷好用的上传整理工具所用的 FTP 协议到底是什么意义,繁杂的模式又有何区别? 二狗子最近搭建了一个图片分享网站,每天都有好多人在他的网站上传许多 ...
- 不支持原子性的 Redis 事务也叫事务吗?
文章收录在 GitHub JavaKeeper ,N线互联网开发必备技能兵器谱 假设现在有这样一个业务,用户获取的某些数据来自第三方接口信息,为避免频繁请求第三方接口,我们往往会加一层缓存,缓存肯定要 ...
- stack 数据结构
栈定义 栈:后进先出(永远从栈顶取元素)LIFO last-in-first-out 栈实现 class Stack { constructor() { this.items = [] this. ...
- HTTPS建立过程
由于HTTPS建立过程经常忘记,固记录 HTTPS建立流程 客户端发起SSL连接 服务端发送证书(证书中包含了服务端公钥) 客户端根据已有证书判断是否是可信任的(客户端决定是否继续) 服务端要求客户端 ...
- 非构造函数方式创建DbContext实例的方法
using Microsoft.EntityFrameworkCore;using Microsoft.EntityFrameworkCore.Design;using Microsoft.Entit ...
- 从架构到部署,全面了解K3s
Kubernetes无处不在--开发者的笔记本.树莓派.云.数据中心.混合云甚至多云上都有Kubernetes.它已然成为现代基础设施的基础,抽象了底层的计算.存储和网络服务.Kubernetes隐藏 ...
- 【新阁教育】基于Log4Net实现日志信息双向存储(含源码)
1.引言 在上位机开发中,日志记录是必不可少的,我们可以通过日志记录做日志分析及错误追踪.初学者会采用txt文本写入来实现日志保存,但是文本写入不是线程安全,当存在多个线程同时写入日志时,就会出现一些 ...
- 分布式事务框架.NetCore CAP总结
来自CAP原作者yang-xiaodong的原理图: 本文撰写者:cmliu,部分内容引用自官方文档,部分内容待更新# .NetCore CAP # 1,简介 CAP 是一个遵循 .NET Stand ...
- oracle之同义词
同义词 从字面上理解就是别名的意思,和视图的功能类似.就是一种映射关系. 14.1 私有同义词; 一般是普通用户自己建立的同义词,创建者需要create synonym 权限. sys:SQL> ...