BalkanOI 2018 Parentrises(贪心+基础DP)
题意
思路
对于 \(\text{P1}\) 的档,首先可以看出 \(O(n^3)\) 的方法,即用 \(O(n^3)\) 的 \(\text{DP}\) 判断合法性以及记录路径。具体是这样的,因为括号匹配可以用一个弹栈的模型去表示(前括号入,后括号弹),用一个整数就可以表示当前的匹配状态,所以用 \(dp_{i,j,k}\) 表示第 \(i\) 个括号,忽视蓝色括号栈中有 \(j\) 个前括号,忽视红色括号栈中有 \(k\) 个前括号。则如果加入一个红前括号,则 \(j\) 加一;若加入一个蓝前括号,则 \(k\) 加一;若加入一个绿前括号,则 \(j,k\) 均加一。后括号同理。
从这个转移来看,似乎就是一个走棋盘的模型,但是单看走棋盘似乎也看不出什么。那么把棋盘拍扁成一维,只保存当前位置到起点的距离。设 \((0,0)\) 在最左上角,那么向右或下走距离加 \(1\) ,右下则加 \(2\) ,反之同理。不难发现,如果存在把距离变回零的方案,则棋盘上可以走回 \((0,0)\) 的方案肯定可以构造出来。
一顿操作,问题变成了给定一个加减号序列,你需要在里面适当位置填上 \(1\) 或 \(2\) ,满足任意前缀大于等于 \(0\) ,最终总和等于 \(0\) 。
这时候,贪心策略也渐渐显然,通过维护某一时刻最大的前缀 \(u\) ,最小的前缀 \(d\) 。扫到某一时刻,碰到加号(左括号)则 \(u+2,d+1\),碰到减号(右括号)则 \(u-1,d-2\) ,最大前缀小于零则要求一无法满足,而最小前缀小于零就补到零(表示可以将其中一个的 \(-2\) 变成了 \(-1\) ),最后得到的 \(d\) 不是零就说明加号过多,要求二无法满足,同样不合法。
我们得到的 \(u\) 就告诉我们如果只有 \(+2,-1\) 时,最后会是几,那我们就倒着扫这么多个数,把 \(+2\) 变成 \(+1\) ,\(-1\) 变成 \(-2\) 。
这样就得到了这个加减号序列,最后我们只用把 \(+1\) 换成红蓝交替的前括号,\(-1\) 换成红蓝交替的后括号,\(+2,-2\) 分别换成绿色前后括号即可。
会 \(O(n)\) 判断合法性,就直接按照这个 \(\text{DP}\) 就可以切 \(\text{P2}\) 档了,可以直接看代码。
代码
#include<bits/stdc++.h>
#define FOR(i,x,y) for(int i=(x),i##END=(y);i<=i##END;++i)
#define DOR(i,x,y) for(int i=(x),i##END=(y);i>=i##END;--i)
template<typename T,typename _T>inline bool chk_min(T &x,const _T y){return y<x?x=y,1:0;}
template<typename T,typename _T>inline bool chk_max(T &x,const _T y){return x<y?x=y,1:0;}
typedef long long ll;
namespace Subtask1
{
const int N=1e6+5;
char str[N];
int ans[N];
int n;
int check()
{
int d=0,u=0;
FOR(i,1,n)
{
if(str[i]=='(')d+=1,u+=2;
else
{
d-=2,u-=1;
chk_max(d,0);
if(u<0)return -1;
}
}
if(d>0)return -1;
return u;
}
void Solve()
{
int T;
scanf("%d",&T);
while(T--)
{
scanf("%s",str+1);
n=strlen(str+1);
int res=check();
if(res==-1){printf("impossible\n");continue;}
FOR(i,1,n)ans[i]=0;
int flg=-1;
if(res)DOR(i,n,1)
{
if(str[i]=='(')ans[i]=flg,flg=-flg;
else ans[i]=2;
res--;
if(!res)break;
}
flg=-1;
FOR(i,1,n)if(str[i]==')')
{
if(ans[i]==2)break;
ans[i]=flg,flg=-flg;
}
FOR(i,1,n)
{
if(ans[i]==1)putchar('R');
else if(ans[i]==-1)putchar('B');
else putchar('G');
}
puts("");
}
}
};
namespace Subtask2
{
const int P=1e9+7;
const int N=305;
int dp[N][N][N<<1];
int ans[N];
void pls(int &x,int y){x+=y;if(x>=P)x-=P;}
void Solve()
{
memset(dp,0,sizeof(dp));
dp[0][0][0]=1;
FOR(i,0,299)
{
FOR(j,0,i)FOR(k,0,2*i)
{
pls(dp[i+1][j+1][k+2],dp[i][j][k]);
if(k)pls(dp[i+1][std::max(0,j-2)][k-1],dp[i][j][k]);
}
FOR(j,0,2*i)pls(ans[i+1],dp[i+1][0][j]);
}
int T;
scanf("%d",&T);
while(T--)
{
int n;
scanf("%d",&n);
printf("%d\n",ans[n]);
}
}
};
int main()
{
int knd;
scanf("%d",&knd);
if(knd==1)Subtask1::Solve();
else if(knd==2)Subtask2::Solve();
return 0;
}
BalkanOI 2018 Parentrises(贪心+基础DP)的更多相关文章
- 「BalkanOI 2018 Day2」Parentrises
「BalkanOI 2018 Day2」Parentrises part1 显然可以直接贪心. 右括号记-1,左括号记1. 默认起始全部绿色,不染色. 策略如下: 从左往右扫,如果右括号个数大于左括号 ...
- 基础dp
队友的建议,让我去学一学kuangbin的基础dp,在这里小小的整理总结一下吧. 首先我感觉自己还远远不够称为一个dp选手,一是这些题目还远不够,二是定义状态的经验不足.不过这些题目让我在一定程度上加 ...
- 贪心/构造/DP 杂题选做Ⅱ
由于换了台电脑,而我的贪心 & 构造能力依然很拉跨,所以决定再开一个坑( 前传: 贪心/构造/DP 杂题选做 u1s1 我预感还有Ⅲ(欸,这不是我在多项式Ⅱ中说过的原话吗) 24. P5912 ...
- poj2709 贪心基础
D - 贪心 基础 Crawling in process... Crawling failed Time Limit:1000MS Memory Limit:65536KB 64bi ...
- uva11292贪心基础题目
C - 贪心 基础 Crawling in process... Crawling failed Time Limit:1000MS Memory Limit:65536KB 64bi ...
- hdu 1009 贪心基础题
B - 贪心 基础 Crawling in process... Crawling failed Time Limit:1000MS Memory Limit:32768KB 64bi ...
- L贪心基础
<span style="color:#330099;">/* L - 贪心 基础 Time Limit:1000MS Memory Limit:65536KB 64b ...
- ALGO-13_蓝桥杯_算法训练_拦截导弹(贪心,DP)
问题描述 某国为了防御敌国的导弹袭击,发展出一种导弹拦截系统.但是这种导弹拦截系统有一个缺陷:虽然它的第一发炮弹能够到达任意的高度,但是以后每一发炮弹都不能高于前一发的高度.某天,雷达捕捉到敌国的导弹 ...
- 基础DP(初级版)
本文主要内容为基础DP,内容来源为<算法导论>,总结不易,转载请注明出处. 后续会更新出kuanbin关于基础DP的题目...... 动态规划: 动态规划用于子问题重叠的情况,即不同的子问 ...
随机推荐
- awk 实战
awk 一些好玩的用法.有什么不错的点子可以留言,发挥出awk牛逼功能 分离mac地址 ifconfig wlan0 | grep eth | awk '{n=split($2,arr,": ...
- 创建一个vue项目()
1.打开cmd,选定路径 2. vue init webpack "项目名称“ 3.在项目路径下,安装一下项目依赖 cnpm install 4.运行 cnpm run dev 5.在浏 ...
- JAVA中使用alibaba fastjson实现JSONObject、Object、Json字符串的转换
JAVA中使用alibaba fastjson实现JSONObject.Object.Json字符串的转换 Object转JSON字符串:String jsonStr = JSONObject.toJ ...
- linux 查看磁盘文件大小
du -sh : 查看当前目录总共占的容量.而不单独列出各子项占用的容量 du -lh --max-depth=1 : 查看当前目录下一级子文件和子目录占用的磁盘容量 df -h 查看整个服务器的磁盘 ...
- hook
hook的定义 hook,钩子,勾住系统的程序逻辑. 在某段SDK源码逻辑执行的构成中,通过代码手段拦截执行该程序,加入自己的代码逻辑 使用价值 hook是安卓面向切面(aop)编程的基础,可以让我们 ...
- 用php实现斐波那契数列,如: 1, 1, 2, 3, 5, 8, 13, 21, 34。用数组求出第20个数的值。
<?php //用数组 function fib($n){ $array = array(); $array[0] = 1; $array[1] = 1; for($i=2;$i<$n;$ ...
- 设计 mysql的单例模式及完整功能
class MySQLDB{ private $host; private $port; private $username; private $password; private $charset; ...
- Oracle函数中对于NO_DATA_FOUND异常处理的研究
一直以来有一个困惑,一直没解决,昨天一哥们问我这个问题,决心弄清楚,终于得到了答案.先看下面这个函数: create or replace function fn_test(c_xm varchar) ...
- github爬虫100项目
为了更好的巩固所学,在github上开始100爬虫项目,记录学习过程,也希望对他人的学习有帮助,目前还在持续更新中,有兴趣可以看看 地址: https://github.com/mapyJJJ/100 ...
- python headers missing
系统环境 win7 64 bit,cygwin,Gvim8.1 问题 Gvim通过插件管理器Vundle下载好了YouCompleteMe插件的全部文件. 利用cygwin 进入在YouComplet ...