洛谷 P4663 - [BalticOI 2008]魔法石(dp)
A:我该是有多无聊来写这种题的题解啊
B:大概是因为这题题解区里没有题解所以我来写一篇了,说明我有高尚的济世情怀(大雾
跑题了跑题了
首先看到字典序第 \(i\) 小小可以自然地想到按位决策,也就是说从高位到低位枚举,对于每一位我们强制令它为 I
,看看有多少个符合要求的字符串,如果小于 \(i\) 就将这位改为 \(1\) 并继续往下枚举。
那么怎么求有多少个符合要求的字符串呢?这时候就要用到 DP 了。显然一个字符串可以成为某个字符串的正规读法当且仅当它的字典序 \(\le\) 反串的字典序,因此我们很容易设计出一个 \(dp\) 状态:\(dp_{i,j,o,x,y}\) 表示当前决策了字符串的前 \(i\) 位和后 \(i\) 位,有 \(j\) 对 XI
相邻,\(o\) 表示反串字典序是否等于原串字典序,从前往后数第 \(i\) 位为 \(x\),从后往前数第 \(i\) 位为 \(y\) 的方案数。转移就枚举第 \(i+1\) 位填的数 \(u\) 和第 \(n-i\) 位填的数 \(v\),如果 \(o=1\) 且 \(u>v\) 就会导致原串字典序 \(>\) 反串字典序,不合法。否则显然有如下转移方程式子:
- \(dp_{i+1,j+[u\ne x]+[v\ne y],o\land[u==v],u,v}\leftarrow dp_{i,j,o,x,y}\)
注意特判 \(i+1=n-i\) 的情况和 \(i+2=n-i\) 的情况,对于前者而言必须有 \(u=v\),因为它俩是同一位,对于后者而言转移的第二维还需额外加上 \([u\ne v]\)。
时间复杂度 \(32n^3\),跑得飞快(
const int MAXN=60;
const char str[3]="IX";
int n,k;ll m,dp[MAXN+5][MAXN+5][2][2][2];
int lim[MAXN+5];
ll calc(){
memset(dp,0,sizeof(dp));
for(int a=0;a<2;a++) for(int b=a;b<2;b++){
if(~lim[1]&&(lim[1]^a)) continue;
if(~lim[n]&&(lim[n]^b)) continue;
dp[1][0][a==b][a][b]=1;
} ll ans=0;
for(int i=1;(i<<1|1)<=n;i++) for(int j=0;j<=k;j++)
for(int o=0;o<2;o++) for(int x=0;x<2;x++) for(int y=0;y<2;y++){
if(!dp[i][j][o][x][y]) continue;
if((i<<1|1)==n){
for(int u=0;u<2;u++){
if(~lim[i+1]&&(lim[i+1]^u)) continue;
if(j+(u^x)+(u^y)<=k) ans+=dp[i][j][o][x][y];
}
} else{
for(int u=0;u<2;u++) for(int v=0;v<2;v++){
if(~lim[i+1]&&(lim[i+1]^u)) continue;
if(~lim[n-i]&&(lim[n-i]^v)) continue;
if(o&&u>v) continue;
if((i+1<<1)==n){
if(j+(u^x)+(u^v)+(v^y)<=k) ans+=dp[i][j][o][x][y];
} else dp[i+1][j+(u^x)+(v^y)][o&&u==v][u][v]+=dp[i][j][o][x][y];
}
}
}
return ans;
}
int main(){
scanf("%d%d%lld",&n,&k,&m);
if(n==1) return ((m>2)?puts("NO SUCH STONE"):(putchar(str[m-1]))),0;
if(n==2){
if(k==0) return ((m>2)?puts("NO SUCH STONE"):(putchar(str[m-1]),putchar(str[m-1]))),0;
else return ((m>3)?puts("NO SUCH STONE"):((m&1)?(putchar(str[m>>1]),putchar(str[m>>1])):puts("IX"))),0;
} memset(lim,-1,sizeof(lim));
if(m>calc()) return puts("NO SUCH STONE")&0;
for(int i=1;i<=n;i++){
lim[i]=0;ll sum=calc();
if(m>sum) m-=sum,lim[i]=1;
}
for(int i=1;i<=n;i++) putchar(str[lim[i]]);
return 0;
}
洛谷 P4663 - [BalticOI 2008]魔法石(dp)的更多相关文章
- 洛谷P2507 [SCOI2008]配对 题解(dp+贪心)
洛谷P2507 [SCOI2008]配对 题解(dp+贪心) 标签:题解 阅读体验:https://zybuluo.com/Junlier/note/1299251 链接题目地址:洛谷P2507 [S ...
- 洛谷 P4665 [BalticOI 2015]Network
洛谷 P4665 [BalticOI 2015]Network 你有一棵 $ n $ 个节点的树,你可以在树上加一些边,使这棵树变成一张无重边.自环的图,且删掉任意一条边它仍然联通.求最少要加多少条边 ...
- 洛谷 P3177 [HAOI2015]树上染色 树形DP
洛谷 P3177 [HAOI2015]树上染色 树形DP 题目描述 有一棵点数为 \(n\) 的树,树边有边权.给你一个在 \(0 \sim n\)之内的正整数 \(k\) ,你要在这棵树中选择 \( ...
- 洛谷 P4072 [SDOI2016]征途 斜率优化DP
洛谷 P4072 [SDOI2016]征途 斜率优化DP 题目描述 \(Pine\) 开始了从 \(S\) 地到 \(T\) 地的征途. 从\(S\)地到\(T\)地的路可以划分成 \(n\) 段,相 ...
- 洛谷P1880 石子合并(区间DP)(环形DP)
To 洛谷.1880 石子合并 题目描述 在一个园形操场的四周摆放N堆石子,现要将石子有次序地合并成一堆.规定每次只能选相邻的2堆合并成新的一堆,并将新的一堆的石子数,记为该次合并的得分. 试设计出1 ...
- 洛谷P1063 能量项链(区间DP)(环形DP)
To 洛谷.1063 能量项链 题目描述 在Mars星球上,每个Mars人都随身佩带着一串能量项链.在项链上有N颗能量珠.能量珠是一颗有头标记与尾标记的珠子,这些标记对应着某个正整数.并且,对于相邻的 ...
- 洛谷P1282 多米诺骨牌 (DP)
洛谷P1282 多米诺骨牌 题目描述 多米诺骨牌有上下2个方块组成,每个方块中有1~6个点.现有排成行的 上方块中点数之和记为S1,下方块中点数之和记为S2,它们的差为|S1-S2|.例如在图8-1中 ...
- 洛谷 P5469 - [NOI2019] 机器人(区间 dp+拉格朗日插值)
洛谷题面传送门 神仙题,放在 D1T2 可能略难了一点( 首先显然对于 P 型机器人而言,将它放在 \(i\) 之后它会走到左边第一个严格 \(>a_i\) 的位置,对于 Q 型机器人而言,将它 ...
- 【题解】洛谷P4158 [SCOI2009] 粉刷匠(DP)
次元传送门:洛谷P4158 思路 f[i][j][k][0/1]表示在坐标为(i,j)的格子 已经涂了k次 (0是此格子涂错 1是此格子涂对)涂对的格子数 显然的是 每次换行都要增加一次次数 那么当j ...
随机推荐
- 初识HTML01
什么是页面? 页面是基于浏览器的应用程序 页面是数据展示的载体,由浏览器和服务器共同执行产物. 浏览器的功能 向服务器发送用户请求指令 接收并解析数据展示给用户 服务器的功能 存储页面资源 处理并响应 ...
- UltraSoft - Alpha - Scrum Meeting 3
Date: Apr 15th, 2020. 会议内容为 贡献分确定与进度汇报. Scrum 情况汇报 进度情况 组员 负责 昨日进度 后两日任务 CookieLau PM.后端 学习前后端分离技术的项 ...
- oo第四次博客-UML暨学期总结
一. 本单元两次作业架构设计 这两次作业实际上难度不大,不存在算法上的难题,大部分时间都是用在处理UML图中各个元素的关系上. 第一次UML主要处理UML类图.有UMLclass,UMLinterfa ...
- TVS管相关知识
在设计中,使用到了TVS管,在之前的设计中没有特别关注TVS管.今天查了一些资料,算是简单的有个了解. TVS管是一种保护器件.它的英文全称为 transient voltage suppressor ...
- C++ string类型小结
目录 构造函数 string.append() string.assign() string.at() string.back() string.begin() string.capasity() s ...
- 有关fgetc配合feof逐行读取文件最后一行读取两遍的错觉?
最近在做一个wifiap设置的接口,用户首先获取到当前wifi 热点的ssid 和pwd,然后修改,保存. 获取信息的时候是fopen对应的hostapd.conf文件,逐行读取,查找匹配的参数. 修 ...
- 数据流中的中位数 牛客网 剑指Offer
数据流中的中位数 牛客网 剑指Offer 题目描述 如何得到一个数据流中的中位数?如果从数据流中读出奇数个数值,那么中位数就是所有数值排序之后位于中间的数值.如果从数据流中读出偶数个数值,那么中位数就 ...
- 替换空格 牛客网 剑指Offer
替换空格 牛客网 剑指Offer 题目描述 请实现一个函数,将一个字符串中的每个空格替换成"%20".例如,当字符串为We Are Happy.则经过替换之后的字符串为We%20A ...
- 高度最小的BST 牛客网 程序员面试金典 C++ Python
高度最小的BST 牛客网 程序员面试金典 C++ Python 题目描述 对于一个元素各不相同且按升序排列的有序序列,请编写一个算法,创建一棵高度最小的二叉查找树. 给定一个有序序列int[] val ...
- poj 3020 Antenna Placement(二分图最大匹配)
题意: N行M列的矩阵,每个格子里不是 * 就是 O . * :是一个利益点. O:是一个空白点. 每次可以用一个圈覆盖相邻的两个*.(左右相邻或上下相邻). 问最少需要多少个圈可以覆盖所有的*. 思 ...