@题目描述@

小 D 特别喜欢玩游戏。这一天,他在玩一款填数游戏。

这个填数游戏的棋盘是一个 n×m 的矩形表格。玩家需要在表格的每个格子中填入一个数字(数字 0 或者数字 1 ),填数时需要满足一些限制。

下面我们来具体描述这些限制。

为了方便描述,我们先给出一些定义:

我们用每个格子的行列坐标来表示一个格子,即(行坐标,列坐标)。(注意: 行列坐标均从 0 开始编号)

一条路径是合法的当且仅当:

(1)这条路径从矩形表格的左上角的格子(0,0)出发,到矩形的右下角格子 (n−1,m−1)结束;

(2)在这条路径中,每次只能从当前的格子移动到右边与它相邻的格子,或者 从当前格子移动到下面与它相邻的格子。

对于一条合法的路径 P,我们可以用一个字符串 w(P) 来表示,该字符串的长度为 n+m−2,其中只包含字符“R”或者字符“D”, 第 i 个字符记录了路径 P 中第 i 步的移动 方法,“ R”表示移动到当前格子右边与它相邻的格子,“ D”表示移动到当前格子下面 与它相邻的格子。

同时,将每条合法路径 P 经过的每个格子上填入的数字依次连接后,会得到一个长 度为 n+m−1 的 01 字符串,记为 s(P) 。

游戏要求小 D 找到一种填数字 0、 1 的方法,使得对于两条路径P1,P2,如果w(P1) > w(P2),那么必须s(P1) ≤ s(P2)。

但是仅仅是找一种方法无法满 足小 D 的好奇心,小 D 更想知道这个游戏有多少种玩法,也就是说,有多少种填数字 的方法满足游戏的要求?

小 D 能力有限,希望你帮助他解决这个问题,即有多少种填 0、1 的方法能满足题目要求。由于答案可能很大,你需要输出答案对 10^9 + 7 取模的结果。

输入

输入共一行,包含两个正整数 n,m,由一个空格分隔,表示矩形的大小。其中 n 表示矩形表格的行数,m 表示矩形表格的列数。

输出

输出共一行,包含一个正整数,表示有多少种填 0、1 的方法能满足游戏的要求。 注意:输出答案对 10^9+7 取模的结果。

输入样例#1

2 2

输出样例#1

12

输入样例#2

3 3

输出样例#2

112

输入样例#3

5 5

输出样例#3

7136

数据规模与约定

n<=8,m<=10^6。

@题解@

傻逼规律题,我竟然不会做。

我太弱了……

题目中的限制等价于:对于所有的点,如图示的红色路径的数字串要 ≥ 蓝色路径的数字串。

我们再把这个限制进行转换:

(1)每一条自右上至左下的对角线的数字总是先出现一堆 0,再出现一堆 1。

(2)如果 (a, b+1) = (a+1, b),则以(a+1, b+1)为左上角,以(n-1, m-1)为右下角的矩形,每一条自右上至左下的对角线的数字全部相同。

可以证明以上的转换都是等价的。



然后就会发现一件很有意思的事情:对于这条对角线 2,肯定是000,001,011,111。这条对角线上一定有 2 个元素相同,也就说限制(2)会限制掉很大一个区域!

于是我们就可以根据这个设计算法了:

先手推 n = 1,n = 2, n = 3 的情况,将它们特判掉。

当 n = 1 时,ans = 2m

当 n = 2 时,ans = 4*3m-1

当 n = 3 时,ans = 112*3m-3

然后:如果交换 n 和 m ,答案不会变。所以我们不妨假设 n <= m。

计算出上图中对角线 1 为 00, 11 的方案数。除此之外对角线 1 一定为 01。

对于对角线 2,先考虑它为 000,111 的情况,然后考虑它为 001,011 的情况。

接下来考虑如何具体计数。

假如对角线 1 为 00, 11:



将剩余的对角线分为三类:两个端点没有被限制(形如 1),对答案的贡献为 4;两个端点被限制了一个(形如 2),对答案的贡献为 3;两个端点都被限制(形如 3),对答案的贡献为 2。将所有对角线的贡献乘起来即可。可以用快速幂。

假如对角线 2 为 000, 111:



图中所标出的对角线是不受限制的。蓝色限制与紫色限制相互组合,可以发现和上面那种情况是差不大多的(除了边缘的行列都被限制)。因此,依然还是按照上文所提到的方法进行计数。

假如对角线 2 为 001,011。因为是对称的,我们只需要弄清 001 的求解方式即可类比推导出 011 的求解方法。



紫色是限制区域。如果绿色矩形中有一个对角线相同,则限制区域会如下图所示。



对比上面那种情况,我们只是将蓝色矩形往下平移了一点。具体的计数方式还是跟上面那种情况是一样的。

还要讨论绿色矩形内对角线都互不相同的方案数。

注意要分类讨论 n = m 和 n < m 的 情况。

@代码@

我觉得我的题解可能描述的不是很具体……因为把这道题解释清楚是一件对我来说不太容易的事情……但是我觉得关键的地方我都写上了。

还是请大家多多见谅啦,如果有疑问就留言提出来吧,我会尽力解答的。

#include<cstdio>
#include<algorithm>
using namespace std;
const int MOD = int(1E9) + 7;
int pow_mod(int b, int p) {
int ret = 1;
while( p ) {
if( p & 1 ) ret = 1LL*ret*b%MOD;
b = 1LL*b*b%MOD;
p >>= 1;
}
return ret;
}
int main() {
int n, m;
scanf("%d%d", &n, &m);
if( n > m ) swap(n, m);
if( n == 1 ) {
printf("%d\n", pow_mod(2, m));
return 0;
}
else if( n == 2 ) {
printf("%lld\n", 4LL*pow_mod(3, m-1)%MOD);
return 0;
}
else if( n == 3 ) {
printf("%lld\n", 112LL*pow_mod(3, m-n)%MOD);
return 0;
}
else {
int ans = 0;
if( n == m ) {
ans = (ans + 2LL*pow_mod(4, n-2)%MOD*pow_mod(2, n-1)%MOD)%MOD;
ans = (ans + 2LL*5LL*pow_mod(4, n-4)%MOD*pow_mod(2, n-1)%MOD)%MOD;
for(int i=4;i<=m;i++) {
if( i == m ) ans = (ans + 4LL*3LL*pow_mod(2, n-2)%MOD)%MOD;
else ans = (ans + 4LL*5LL*pow_mod(4, n-i-1)%MOD*pow_mod(2, n-1)%MOD)%MOD;
}
ans = (ans + 3LL*pow_mod(2, n-2))%MOD;
for(int i=4;i<=n;i++) {
if( i == n ) ans = (ans + 4LL*3LL*pow_mod(2, m-2)%MOD)%MOD;
else ans = (ans + 4LL*5LL*pow_mod(4, m-i-1)%MOD*pow_mod(2, m-1)%MOD)%MOD;
}
ans = (ans + 3LL*pow_mod(2, n-2))%MOD;
}
else {
ans = (ans + 2LL*pow_mod(4, n-2)%MOD*pow_mod(3, m-n)%MOD*pow_mod(2, n-1)%MOD)%MOD;
ans = (ans + 2LL*5LL*pow_mod(4, n-4)%MOD*pow_mod(3, m-n)%MOD*pow_mod(2, n-1)%MOD)%MOD;
for(int i=4;i<=m;i++) {
if( i == m ) ans = (ans + 3LL*3LL*pow_mod(2, n-2)%MOD)%MOD;
else if( i > n ) ans = (ans + 3LL*4LL*pow_mod(3, m-i-1)%MOD*pow_mod(2, n-1)%MOD)%MOD;
else if( i == n ) ans = (ans + 4LL*4LL*pow_mod(3, m-n-1)%MOD*pow_mod(2, n-1)%MOD)%MOD;
else if( i < n ) ans = (ans + 4LL*5LL*pow_mod(4, n-i-1)%MOD*pow_mod(3, m-n)%MOD*pow_mod(2, n-1)%MOD)%MOD;
}
ans = (ans + 3LL*pow_mod(2, n-2))%MOD;
for(int i=4;i<=n;i++) {
if( i == n ) ans = (ans + 4LL*4LL*pow_mod(3, m-n-1)%MOD*pow_mod(2, n-1)%MOD)%MOD;
else ans = (ans + 4LL*5LL*pow_mod(4, n-i-1)%MOD*pow_mod(3, m-n)%MOD*pow_mod(2, n-1)%MOD)%MOD;
}
ans = (ans + 4LL*pow_mod(3, m-n-1)%MOD*pow_mod(2, n-1)%MOD)%MOD;
}
printf("%d\n", 2LL*ans%MOD);//*2:左上角的填数方式
}
}

@NOIP2018 - D2T2@ 填数游戏的更多相关文章

  1. [NOIP2018 TG D2T2]填数游戏

    题目大意:$NOIP2018\;TG\;D2T2$ 题解:在skip2004的博客基础上修改的,也是暴搜. 说明一下把vector改成数组并不可以通过此题,记录. 结论:在$m>n+1$时答案为 ...

  2. NOIP2018 Day2T2 填数游戏

    下面先给出大家都用的打表大法: 首先我们可以发现 \(n \le 3\) 的情况有 \(65pts\),而 \(n\) 这么小,打一下表何乐而不为呢?于是我写了一个爆枚每个位置再 \(check\) ...

  3. UOJ#440. 【NOIP2018】填数游戏 动态规划

    原文链接www.cnblogs.com/zhouzhendong/p/UOJ440.html 前言 菜鸡选手到省选了才做联赛题. 题解 首先我们分析一下性质: 1. 假如一个格子是 0,那么它的右上角 ...

  4. 【逆向笔记】2017年全国大学生信息安全竞赛 Reverse 填数游戏

    2017年全国大学生信息安全竞赛 Reverse 填数游戏 起因是吾爱破解大手发的解题思路,觉得题挺有意思的,就找来学习学习 这是i春秋的下载链接 http://static2.ichunqiu.co ...

  5. [Noip2018]填数游戏

    传送门 Description 耳熟能详,就不多说了 Solution 对于一个不会推式子的蒟蒻,如何在考场优雅地通过此题 手玩样例,发现对于 \(n=1\) , \(ans=2^m\) .对于 \( ...

  6. NOIP2018 填数游戏 搜索、DP

    LOJ 感觉这个题十分好玩于是诈尸更博.一年之前的做题心得只有这道题还记得清楚-- 设输入为\(n,m\)时的答案为\(f(n,m)\),首先\(f(n,m)=f(m,n)\)所以接下来默认\(n \ ...

  7. 【题解】NOIP2018 填数游戏

    题目戳我 \(\text{Solution:}\) 题目标签是\(dp,\)但是纯暴力打表找规律可以有\(65\)分. 首先是对于\(O(2^{nm}*nm)\)的暴力搜索,显然都会. 考虑几条性质: ...

  8. luogu P5023 填数游戏

    luogu loj 被这道题送退役了 题是挺有趣的,然而可能讨论比较麻烦,肝了2h 又自闭了,鉴于CSP在即,就只能先写个打表题解了 下面令\(n<m\),首先\(n=1\)时答案为\(2^m\ ...

  9. JZOJ5965【NOIP2018提高组D2T2】填数游戏

    题目 作为NOIP2018的题目,我觉得不需要把题目贴出来了. 大意就是,在一个n∗mn*mn∗m的010101矩阵中,从左上角到右下角的路径中,对于任意的两条,上面的那条小于下面的那条.问满足这样的 ...

随机推荐

  1. H5C3--媒体查询(向上兼容,向下覆盖),link中引入的ont and only

    一.使用 实例: <!DOCTYPE html> <html lang="en"> <head> <meta charset=" ...

  2. 适配器模式--在NBA我需要翻译

     适配器模式:将一个类的接口转换成客户希望的另外一个接口.Adapter模式使得原本由于接口不兼容而不能一起工作的那些类可以一起工作. 在软件开发中,也就是系统的数据和行为都正确,但接口不符时,我们应 ...

  3. iOS 7: 如何为iPhone 5s编译64位应用

    随着iPhone 5S的推出,大家开始关心5S上所使用的64位CPU A7. 除了关心A7的性能以外,大家还会关心一个问题,那就是使用A7的64位系统对应用有没有什么要求.特别是应用开发者,大家都比较 ...

  4. 机器学习之决策树(ID3)算法与Python实现

    机器学习之决策树(ID3)算法与Python实现 机器学习中,决策树是一个预测模型:他代表的是对象属性与对象值之间的一种映射关系.树中每个节点表示某个对象,而每个分叉路径则代表的某个可能的属性值,而每 ...

  5. OWIN启动类检测

    每个OWIN应用程序都有一个启动类,可以在这个类里为应用程序管道指定组件.有不同的方式可以将启动类与运行时关联起来,这依赖于选择的托管模型(OwinHost,IIS,IIS-Express).本教程中 ...

  6. python pattern 类

  7. 使用cnpm真的会有诡异的Bug

    前端网页居然会出现堆栈溢出,然后网页崩溃,退出的问题. 出现这个Bug的时候,我非常的怀疑我自己的一些操作能力,比如,git的操作. 毕竟我是本地代码然后拉取远程分支,还会暂存自己的代码,然后暂存区代 ...

  8. MSSQL2008 数据压缩方法

    数据压缩功能使得SOL Server 2008允许在表.索引和分区中执行数据压缩,这样不仅可以节省磁盘空间,而且允许更多数据置入RAM中,从而提升数据库查询的性能. 1.启用行压缩 如果我们要在指定的 ...

  9. php中实现多线程的的两个方法具体是怎样?

    假设你要建立一个服务来检查正在运行的n台服务器,以确定他们还在正常运转.你可能会写下面这样的代码:代码如下: <?php$hosts = array("host1.sample.com ...

  10. yii生成Model出错:yii-gii-generators-model-Generator.json No such file or dictory

    讲runtime 这个文件夹添加权限 chmod o+w runtime