「JOISC 2016 Day 1」棋盘游戏
「JOISC 2016 Day 1」棋盘游戏
先判无解:第1,3行有连续的空格或四个角有空格。
然后可以发现有解的情况第1,3行可以在任意时间摆放。
对于某一列,若第2行放有棋子,那么显然可以把棋盘分开两边来计算,然后再排列一下。
所以目前要处理的是一段 第二行都没有棋子的棋盘的方案数。
对于该段棋盘:
定义\(dp[i][j][2]\)为前\(i\)列,当前列的第二行是第\(j\)个放置的,\(0/1\)表示是否为通过行的方式放置。
这里为了避免重复和方便,若可以通过列的方式放置,就不通过行的方式放置。
具体的转移相当于前\(i-1\)列放置棋子的顺序序列中插入第\(i\)列的放置顺序。
第一列在dp前处理掉。
转移就分为3种(显然两列不能同时通过行的方式放置)。
令v为\(i\)列中1,3行没放置棋子格子数。
以T来指代\(i\)列第二行的棋子,\(T_1\)为\(i-1\)列第二行棋子。
0->0
只需要保证在放T前放好v。
0->1
要保证在放T后才放好v,\(T_1\)在T前放。
1->0
保证在放T前放好v,放T在\(T_1\)前。
整个棋盘的第一和最后一列都是不用特判的,因为它们都是列放置。
对于其他整段,第一列的第二行前面和最后一列的第二行后面都是有棋子的。
#include<bits/stdc++.h>
#define rep(q,a,b) for(int q=a,q##_end_=b;q<=q##_end_;++q)
#define dep(q,a,b) for(int q=a,q##_end_=b;q>=q##_end_;--q)
#define mem(a,b) memset(a,b,sizeof a )
#define debug(a) cerr<<#a<<' '<<a<<"___"<<endl
using namespace std;
void in(int &r) {
static char c;
r=0;
while(c=getchar(),!isdigit(c));
do r=(r<<1)+(r<<3)+(c^48);
while(c=getchar(),isdigit(c));
}
const int mn=2005;
int dp[mn][mn*3][2];
int had[mn][4],n;//其实这里had是没有的意思,没有改过来
const int mod=1e9+7;
int mul[mn*3],inv[mn*3];
int C(int a,int b){
return 1LL*mul[b]*inv[a]%mod*inv[b-a]%mod;
}
int A(int a,int b){
return 1LL*mul[b]*inv[b-a]%mod;
}
int tot_sum,ans=1;
void add(int &a,long long b){
a=(a+b)%mod;
}
int Sum[mn*3][2];
void get_ans(int l,int r){
if(l>r)return;
int sum=0;
int v=had[l][1]+had[l][3];
dp[l][v+1][0]=A(v,v);
rep(q,1,v)dp[l][q][1]=A(v,v);
sum+=v+1;
rep(q,l+1,r){
v=had[q][1]+had[q][3];
rep(w,1,sum)Sum[w][0]=dp[q-1][w][0],Sum[w][1]=dp[q-1][w][1];
rep(w,1,sum)add(Sum[w][0],Sum[w-1][0]),add(Sum[w][1],Sum[w-1][1]);
rep(w,v+1,sum+v+1){
add(dp[q][w][0],1LL*A(v,w-1)*Sum[sum][0]);
// rep(e,1,sum){
// add(dp[q][w][0],1LL*A(v,w-1)*dp[q-1][e][0]);
// }
}
rep(w,v+1,sum+v){
add(dp[q][w][0],1LL*A(v,w-1)*(Sum[sum][1]-Sum[w-v-1][1]));
// rep(e,w-v,sum){
// add(dp[q][w][0],1LL*A(v,w-1)*dp[q-1][e][1]);
// }
}
rep(r,0,v-1){
int ty=C(r,v);
rep(w,r+1,sum+r+1){
add(dp[q][w][1],1LL*ty*A(r,w-1)*A(v-r,sum+v+1-w)%mod*Sum[w-r-1][0]);
// rep(e,0,w-r-1){
// add(dp[q][w][1],1LL*ty*A(r,w-1)*dp[q-1][e][0]%mod*A(v-r,sum+v+1-w));
// }
}
}
sum+=v+1;
}
int tot=0;
rep(q,0,sum){
add(tot,dp[r][q][0]);
add(tot,dp[r][q][1]);
}
tot_sum+=sum,ans=1LL*ans*tot%mod*C(sum,tot_sum)%mod;
}
bool check_it_can_be_solve(){
if(had[1][1]||had[1][3]||had[n][1]||had[n][3])return 0;
rep(q,2,n)if(had[q][1]&&had[q-1][1]||had[q][3]&&had[q-1][3])return 0;
return 1;
}
char as[mn];
int main(){
freopen("game.in","r",stdin);
freopen("game.out","w",stdout);
in(n);
mul[0]=1;
rep(q,1,n*3)mul[q]=1LL*mul[q-1]*q%mod;
inv[0]=inv[1]=1;
rep(q,2,n*3)inv[q]=1LL*(mod-mod/q)*inv[mod%q]%mod;
rep(q,1,n*3)inv[q]=1LL*inv[q-1]*inv[q]%mod;
rep(q,1,3){
scanf("%s",as+1);
rep(w,1,n)had[w][q]=as[w]=='x';
}
if(!check_it_can_be_solve()){
puts("0");
return 0;
}
int now=1;
while(now<=n&&had[now][2])++now;
if(now==n+1)get_ans(1,now-1);
else{
get_ans(1,now-1);
int v=had[now][1]+had[now][3];
tot_sum+=v,ans=1LL*ans*A(v,tot_sum)%mod;
++now;
while(1){
int last=now;
while(now<=n&&had[now][2])++now;
if(now==n+1)get_ans(last,now-1);
else{
get_ans(last,now-1);
v=had[now][1]+had[now][3];
tot_sum+=v,ans=1LL*ans*A(v,tot_sum)%mod;
++now;
}
if(now==n+1)break;
}
}
printf("%d\n",(ans+mod)%mod);
return 0;
}
「JOISC 2016 Day 1」棋盘游戏的更多相关文章
- Loj #2731 「JOISC 2016 Day 1」棋盘游戏
Loj 2731 「JOISC 2016 Day 1」棋盘游戏 JOI 君有一个棋盘,棋盘上有 \(N\) 行 \(3\) 列 的格子.JOI 君有若干棋子,并想用它们来玩一个游戏.初始状态棋盘上至少 ...
- LOJ #2731. 「JOISC 2016 Day 1」棋盘游戏(dp)
题意 JOI 君有一个棋盘,棋盘上有 \(N\) 行 \(3\) 列 的格子.JOI 君有若干棋子,并想用它们来玩一个游戏.初始状态棋盘上至少有一个棋子,也至少有一个空位. 游戏的目标是:在还没有放棋 ...
- LOJ 2737 「JOISC 2016 Day 3」电报 ——思路+基环树DP
题目:https://loj.ac/problem/2737 相连的关系形成若干环 / 内向基环树 .如果不是只有一个环的话,就得断开一些边使得图变成若干链.边的边权是以它为出边的点的点权. 基环树的 ...
- LOJ 2736 「JOISC 2016 Day 3」回转寿司 ——堆+分块思路
题目:https://loj.ac/problem/2736 如果每个询问都是 l = 1 , r = n ,那么每次输出序列的 n 个数与本次操作的数的最大值即可.可以用堆维护. 不同区间的询问,可 ...
- loj2734「JOISC 2016 Day 2」女装大佬 || 洛谷P3615 如厕计划
loj2734 洛谷P3615 http://218.5.5.242:9021/problem/185 不会做... 题解(来自ditoly): 这一步更详细的解释(来自kkksc03): 还是从后面 ...
- 「JOISC 2016 Day 3」回转寿司
https://loj.ac/problem/2736 题解 挺有意思的题. 考虑这种操作不好直接维护,还有时限比较长,所以考虑分块. 考虑一个操作对整个块的影响,无非就是可能把最大的拿走,再把新的元 ...
- [LOJ#2732] 「JOISC 2016 Day 2」雇佣计划
参考博文 (不过个人感觉我讲的稍微更清楚一点) 题目就是让我们求图中满足数值大于等于B的连通块数量 然后我们可以尝试转换为求连通块两端所产生的“谷”的数量,显然一个连通块对谷可以贡献2的答案,最终答案 ...
- 题解 「JOISC 2016 Day 3」电报
题目传送门 题目大意 给出一个\(n\)个点\(n\)条边的图,每个点有且仅有一个出边,改变每条边都会有对应的花费.求最小的花费使得整个图强连通. 思路 很显然,最后的图就是一个环.那我们要求的答案实 ...
- 「JOISC 2015 Day 1」卡片占卜
题目描述 K 理事长是占卜好手,他精通各种形式的占卜.今天,他要用正面写着 I ,背面写着 O 的卡片占卜一下日本 IOI 国家队的选手选择情况. 占卜的方法如下: 首先,选取五个正整数 A,B,C, ...
随机推荐
- 【嵌入式AI】全志 XR806 say hello world
欢迎关注我的公众号 [极智视界],回复001获取Google编程规范 O_o >_< o_O O_o ~_~ o_O 大家好,我是极智视界,本文介绍了全志 ...
- codevs 1300:文件排版(DP)
题目描述 写电子邮件是有趣的,但不幸的是经常写不好看,主要是因为所有的行不一样长,你的上司想要发排版精美的电子邮件,你的任务是为他编写一个电子邮件排版程序. 完成这个任务最简单的办法是在太短的行中的单 ...
- CS5202Capstone|CS5202芯片|CS5202芯片方案
一.CS5202功能概述CS5202结合了DisplayPort输入接口和模拟RGB DAC输出接口.嵌入式单片机基于工业标准8051核心,适用于多个细分市场和显示器应用程序,如笔记本电脑.主板.台式 ...
- 编写Java程序,车站只剩 50 张从武汉到北京的车票,现有 3 个窗口售卖,用程序模拟售票的过程,使用Runnable解决线程安全问题
查看本章节 查看作业目录 需求说明: 车站只剩 50 张从武汉到北京的车票,现有 3 个窗口售卖,用程序模拟售票的过程,要求使用同步方法保证售票过程中票数的正确性 实现思路: 创建 Java 项目,在 ...
- Git创建本地仓库
使用Git创建本地仓库, 可以记录文本文件变化, 这里以自己的文章为例, 利用TortoiseGit工具创建一个本地Git仓库. 1. 切换到需要创建仓库的目录下, 比如:D:\MyDoc\Devel ...
- html基础 button按钮标签
场景:在网页中显示用户点击的按钮标签名:button 注意:form不能少,少了不会出效果 html代码 <form > 昵称: <input type="text&quo ...
- nginx安装,手动源码安装
什么是Nginx? Nginx (engine x) 是一个高性能的HTTP和反向代理web服务器,同时也提供了IMAP/POP3/SMTP服务.Nginx是由伊戈尔·赛索耶夫为俄罗斯访问量第二的Ra ...
- mysql 外连接
自连接:最大的特点是:一张表看做两张表.自己连接自己. 找出每个员工的上级领导,要求显示员工名和对应的领导名. select e.ename,ee.ename from emp e join emp ...
- 论文翻译:2021_AEC IN A NETSHELL: ON TARGET AND TOPOLOGY CHOICES FOR FCRN ACOUSTIC ECHO CANCELLATION
论文地址:https://ieeexploreieee.53yu.com/abstract/document/9414715 Netshell 中的 AEC:关于 FCRN 声学回声消除的目标和拓扑选 ...
- Web开发之Servlet
当一个请求到达服务端,服务器怎么处理? 当一个请求到达服务端时,由服务端的引擎来进行分析.它根据工程名找到工程, 然后拿到URL的资源地址和web.XML文件的所有的进行对比,和哪一个对比上就找到了具 ...