「SCOI2016」围棋 解题报告
「SCOI2016」围棋
打CF后困不拉基的,搞了一上午...
考虑直接状压棋子,然后发现会t
考虑我们需要上一行的状态本质上是某个位置为末尾是否可以匹配第一行的串
于是状态可以\(2^m\)压住了,但还是会T
考虑到复杂度瓶颈在于每行的状态都要枚举上一行的状态,是按行转移的。
那么如果做一个轮廓线,就可以按格子转移
考虑有那些状态,当前格子\(i,j\),当前轮廓线是否可以匹配第一行的串的状态\(s\)
然后你试试发现如果想好好转移
得存一个\((i,j)\)匹配到第一行串的位置\(x\),和第二行串的位置\(y\)
这里的思考方向倾向于先想转移,转移不了再加状态
然后发现这个匹配的过程可以使用KMP优化,转移的均摊是\(O(1)\)的
复杂度\(O(nm2^{m-c+1}c^2)\)
这里复杂度写成\(2^m\)可能不太稳...
Code:
#include <cstdio>
#include <cctype>
#include <cstring>
template <class T>
void read(T &x)
{
x=0;char c=getchar();
while(!isdigit(c)) c=getchar();
while(isdigit(c)) x=x*10+c-'0',c=getchar();
}
const int mod=1e9+7;
void add(int &x,int y){x+=y;if(x>=mod)x-=mod;}
void mul(int &x,int y){x=1ll*x*y%mod;}
int n,m,c,q,dp[2][1<<12][7][7];
int nxt1[12],nxt2[12],d1[12],d2[12];
char s1[12],s2[12];
void init(char *s,int *d,int *nxt)
{
for(int i=1;i<=c;i++) d[i]=(s[i]=='W'?2:(s[i]=='B'));
for(int j=0,i=2;i<=c;i++)
{
while(j&&d[j+1]!=d[i]) j=nxt[j];
if(d[j+1]==d[i]) ++j;
nxt[i]=j;
}
}
int get(int *nxt,int p,int col,int *d)
{
while(p&&d[p+1]!=col) p=nxt[p];
if(d[p+1]==col) ++p;
return p;
}
int main()
{
read(n),read(m),read(c),read(q);
while(q--)
{
memset(d1,-1,sizeof d1),memset(nxt1,0,sizeof nxt1);
memset(d2,-1,sizeof d2),memset(nxt2,0,sizeof nxt2);
scanf("%s",s1+1),init(s1,d1,nxt1);
scanf("%s",s2+1),init(s2,d2,nxt2);
memset(dp,0,sizeof dp);
int sum=1,cur=0,tx,ty,t,l=m-c+1;
dp[cur][0][0][0]=1;
for(int i=1;i<=n;i++)
{
for(int j=0;j<m;j++)
{
memset(dp[cur^1],0,sizeof dp[cur^1]);
for(int s=0;s<1<<l;s++)
for(int x=0;x<=c;x++)
for(int y=0;y<=c;y++)
if(dp[cur][s][x][y])
for(int col=0;col<3;col++)
{
tx=get(nxt1,x,col,d1);
ty=get(nxt2,y,col,d2);
if(j+1>=c)
{
t=s&(~(1<<j+1-c));
t|=(tx==c)<<j+1-c;
}
else t=s;
if(j+1>=c&&(s>>j+1-c&1)&&ty==c) continue;
add(dp[cur^1][t][tx][ty],dp[cur][s][x][y]);
}
cur^=1;
mul(sum,3);
}
for(int s=0;s<1<<l;s++)
for(int x=0;x<=c;x++)
for(int y=0;y<=c;y++)
if(x||y)
add(dp[cur][s][0][0],dp[cur][s][x][y]),dp[cur][s][x][y]=0;
}
int ans=0;
for(int s=0;s<1<<l;s++)
add(ans,dp[cur][s][0][0]);
add(sum,mod-ans);
printf("%d\n",sum);
}
return 0;
}
2019.3.6
「SCOI2016」围棋 解题报告的更多相关文章
- 「SCOI2016」妖怪 解题报告
「SCOI2016」妖怪 玄妙...盲猜一个结论,然后过了,事后一证,然后假了,数据真水 首先要最小化 \[ \max_{i=1}^n (1+k)x_i+(1+\frac{1}{k})y_i \] \ ...
- 「SCOI2016」美味 解题报告
「SCOI2016」美味 状态极差无比,一个锤子题目而已 考虑每次对\(b\)和\(d\)求\(c=d \ xor \ (a+b)\)的最大值,因为异或每一位是独立的,所以我们可以尝试按位贪心. 如果 ...
- 「SCOI2016」萌萌哒 解题报告
「SCOI2016」萌萌哒 这思路厉害啊.. 容易发现有个暴力是并查集 然后我想了半天线段树优化无果 然后正解是倍增优化并查集 有这个思路就简单了,就是开一个并查集代表每个开头\(i\)每个长\(2^ ...
- 「ZJOI2016」旅行者 解题报告
「ZJOI2016」旅行者 对网格图进行分治. 每次从中间选一列,然后枚举每个这一列的格子作为起点跑最短路,进入子矩形时把询问划分一下,有点类似整体二分 至于复杂度么,我不会阿 Code: #incl ...
- 「HNOI2016」树 解题报告
「HNOI2016」树 事毒瘤题... 我一开始以为每次把大树的子树再接给大树,然后死活不知道咋做,心想怕不是个神仙题哦 然后看题解后才发现是把模板树的子树给大树,虽然思维上难度没啥了,但是还是很难写 ...
- 「HNOI2016」序列 解题报告
「HNOI2016」序列 有一些高妙的做法,懒得看 考虑莫队,考虑莫队咋移动区间 然后你在区间内部找一个最小值的位置,假设现在从右边加 最小值左边区间显然可以\(O(1)\),最小值右边的区间是断掉的 ...
- 「HNOI2016」网络 解题报告
「HNOI2016」网络 我有一个绝妙的可持久化树套树思路,可惜的是,它的空间是\(n\log^2 n\)的... 注意到对一个询问,我们可以二分答案 然后统计经过这个点大于当前答案的路径条数,如果这 ...
- 「HAOI2018」染色 解题报告
「HAOI2018」染色 是个套路题.. 考虑容斥 则恰好为\(k\)个颜色恰好为\(c\)次的贡献为 \[ \binom{m}{k}\sum_{i\ge k}(-1)^{i-k}\binom{m-k ...
- 「HNOI2016」最小公倍数 解题报告
「HNOI2016」最小公倍数 考虑暴力,对每个询问,处理出\(\le a,\le b\)的与询问点在一起的联通块,然后判断是否是一个联通块,且联通块\(a,b\)最大值是否满足要求. 然后很显然需要 ...
随机推荐
- awr format
AWR-Format工具 在Chrome高版本中配置使用AWR-Format for Chrome插件
- java 浅拷贝和深拷贝 对象克隆clone
分一下几点讨论: 为什么要克隆? 如何实现克隆 浅克隆和深克隆 解决多层克隆问题 总结 一:为什么要克隆? 大家先思考一个问题,为什么需要克隆对象?直接new一个对象不行吗? 答案是:克隆的对象可能包 ...
- Mybatis+Spring整合后Mapper测试类编写
public class UserMapperTest { private ApplicationContext applicationContext; @Before public void ini ...
- VMWARE中NAT下获取不到IP
1.编辑-虚拟网络编辑器-dhcp设置 2.虚拟机-可移动设备-网络适配器-设置,注意:这里一定要选nat,当初我就是选了桥接,死活上不去,搞了2个小时.
- Tomcat 常见的几个报错与启动问题
报错:A child container failed during start 1.Caused by: java.lang.IllegalArgumentException: Servlet ma ...
- WPF当属性值改变时利用PropertyChanged事件来加载动画
在我们的程序中,有时我们需要当绑定到UI界面上的属性值发生变化从而引起数据更新的时候能够加载一些动画,从而使数据更新的效果更佳绚丽,在我们的程序中尽量将动画作为一种资源放在xaml中,而不是在后台中通 ...
- caffe2学习
https://www.jianshu.com/p/50bf3bd4e3d0 知乎专栏 https://zhuanlan.zhihu.com/kingbob
- Express学习 ------模版引擎(handlebars)
Handlebars一款js模版引擎,我们在做客户端开发的时候,也可能已经使用过.它语法比较简单,和我们平常写的html 一样,只不过html 中可以加入handlebars 表达式. handleb ...
- hdu-1686(kmp)
题意:前面的都是废话...其实直接看输入要求和输出要求就可以了,就是给你两个字符串,问你第一个字符串在第二个字符串中出现几次: 解题思路:kmp... 代码: #include<iostream ...
- 源码分析: 图片加载框架Picasso源码分析
使用: Picasso.with(this) .load("http://imgstore.cdn.sogou.com/app/a/100540002/467502.jpg") . ...