[CSP-S模拟测试]:题(DP+数学)
题目描述
出个题就好了.这就是出题人没有写题目背景的原因。
你在平面直角坐标系上。
你一开始位于$(0,0)$。
每次可以在上/下/左/右四个方向中选一个走一步。
即:从$(x,y)$走到$(x,y+1),(x,y-1),(x-1,y),(x+1,y)$四个位置中的其中一个。
允许你走的步数已经确定为$n$,现在你想走$n$步之后回到$(0,0)$。
但这太简单了,你希望知道有多少种不同的方案能够使你在n步之后回到$(0,0)$,当且仅当两种方案至少有一步走的方向不同,这两种方案被认为是不同的.
答案可能很大所以只需要输出答案对${10}^9+7$取模后的结果。
这还是太简单了,所以你给能够到达的格点加上了一些限制,一共有三种限制,加上没有限制的情况,一共有四种情况,用$0,1,2,3$标号:
$0.$没有任何限制,可以到达坐标系上所有的点。即能到达的点集为$\{ (x,y)|x,y$为整数$\}$
$1.$只允许到达x轴非负半轴上的点。即能到达的点集为$\{ (x,y)\|x$为非负数$,y=0\}$
$2.$只允许到达坐标轴上的点。即能到达的点集为$\{ (x,y)\|x=0$或$y=0\}$
$3.$只允许到达$x$轴非负半轴上的点,$y$轴非负半轴上的点以及第$1$象限的点。即能到达的点集为$\{ (x,y)\| x\geqslant 0,y\geqslant 0\}$
输入格式
一行两个整数(空格隔开)n和typ,分别表示你必须恰好走的步数和限制的种类。
typ的含义见【题目描述】
输出格式
一行一个整数ans,表示不同的方案数对${10}^9+7$取模后的结果。
样例
样例输入0:
100 0
样例输出0:
383726909
样例输入1:
100 1
样例输出1:
265470434
样例输入2:
100 2
样例输出2:
376611634
样例输入3:
100 3
样例输出3:
627595255
数据范围与提示
$10\%$的数据,$typ=0,n\leqslant 100$
$10\%$的数据,$typ=0,n\leqslant 1,000$
$ 5\%$的数据,$typ=0,n\leqslant 100,000$
$10\%$的数据,$typ=1,n\leqslant 100$
$10\%$的数据,$typ=1,n\leqslant 1,000$
$ 5\%$的数据,$typ=1,n\leqslant 100,000$
$10\%$的数据,$typ=2,n\leqslant 100$
$15\%$的数据,$typ=2,n\leqslant 1,000$
$10\%$的数据,$typ=3,n\leqslant 100$
$10\%$的数据,$typ=3,n\leqslant 1,000$
$ 5\%$的数据,$typ=3,n\leqslant 100,000$
以上$11$部分数据没有交集。
$100%$的数据,保证$n$为偶数,$2\leqslant n\leqslant 100,000$,$0\leqslant typ\leqslant 3$。
题解
一个数学和$DP$的大结合。
首先,这四问都可以用$\Theta (n^3)$的$DP$拿到部分分。
定义$dp[t][i][j]$表示$t$时刻在点$(i,j)$的方案数。
$DP$式子为:
$dp[t][i][j]=dp[t-1][i-1][j]+dp[t-1][i][j-1]+dp[t-1][i+1][j]+dp[t-1][i][j+1]$。
注意边界就好了。
期望得分:$40$分。
$typ=0$的情况,显然让我们想到了visit那道题,25分就轻松拿到了。
$typ=1$的情况,我们可以理解为向右走为进栈,想左走为出栈,那么就是求卡特兰数的第$\frac{n}{2}$项,直接用公式解决就可以了,附带一个卡特兰数的求法。
$typ=2$的情况,注意数据范围$n\leqslant 1,000$,所以我们可以考虑$\Theta (n^2)$的做法,定义$dp[t][i]$表示在$t$时刻位于距离原点为$i$的方案数,现在我们只考虑它在$x$轴上的情况,那么$DP$式子为:$dp[t][i]=dp[t-1][i-1]+dp[t-1][i+1]$,现在放回在坐标系上的情况,那么只有在原点的时候才会有四个方向向它贡献答案,那么$DP$式子只在$i=0$时发生改变,为:$dp[t][0]=4\times dp[t-1][1]$,也可以用滚动数组节省空间。注:正解为卡特兰数,在此介绍另一种做法。
$typ=3$的情况,只是在$typ=1$的情况上加一个枚举横向走的步数$i$纵向的步数即为$n-i$,需要注意的是横向的步数肯定为偶数步,那么方案数就是:$C_n^i\times catalan\frac{i}{2}\times catalan\frac{n-i}{2}$。
代码时刻
#include<bits/stdc++.h>
using namespace std;
int n,typ;
long long inv[200001],jget_C[200001];
long long dp[1001][1001];
long long qpow(long long x,long long y)
{
long long res=1;
while(y)
{
if(y%2)res=(res*x)%1000000007;
y>>=1;
x=(x*x)%1000000007;
}
return res;
}
void pre_work()
{
jget_C[0]=1; jget_C[1]=1;
for(int i=2;i<=n;i++)jget_C[i]=1LL*jget_C[i-1]*i%1000000007;
inv[0]=1; inv[1]=1;
for(int i=2;i<=n;i++)inv[i]=1LL*(1000000007-1000000007/i)*inv[1000000007%i]%1000000007;
for(int i=1;i<=n;i++)inv[i]=1LL*inv[i]*inv[i-1]%1000000007;
}
long long get_C(int n,int m)
{
if(n<m)return 0;
if(m==0)return 1;
return (jget_C[n]%1000000007*inv[m]%1000000007*inv[n-m]%1000000007)%1000000007;
}
long long get_Catalan(int x){return get_C(2*x,x)%1000000007*qpow((long long)(x+1),1000000005)%1000000007;}
int main()
{
scanf("%d%d",&n,&typ);
pre_work();
switch(typ)
{
case 0:
{
long long ans=0;
for(int i=0;i<=n;i+=2)
ans=(ans+get_C(n,i)%1000000007*get_C(i,i/2)%1000000007*get_C(n-i,(n-i)/2)%1000000007)%1000000007;
printf("%lld",ans%1000000007);
break;
}
case 1:
{
printf("%lld",get_Catalan(n/2));
break;
}
case 2:
{
dp[0][0]=1;
for(int i=1;i<=n;i++)
{
dp[i][0]=dp[i-1][1]*4%1000000007;
for(int j=1;j<=n/2;j++)
dp[i][j]=(dp[i-1][j-1]+dp[i-1][j+1])%1000000007;
}
printf("%lld\n",dp[n][0]);
break;
}
case 3:
{
long long ans=0;
for(int i=0;i<=n;i+=2)
ans=(ans+get_C(n,i)%1000000007*get_Catalan(i/2)%1000000007*get_Catalan((n-i)/2)%1000000007)%1000000007;
printf("%lld",ans);
break;
}
}
return 0;
}
rp++
[CSP-S模拟测试]:题(DP+数学)的更多相关文章
- [CSP-S模拟测试]:题(DP)
		
题目描述 由于出题人赶时间所以没办法编故事来作为背景.一开始有$n$个苹果,$m$个人依次来吃苹果,第$i$个人会尝试吃$u_i$或$v_i$号苹果,具体来说分三种情况.$\bullet 1.$两个苹 ...
 - [CSP-S模拟测试]:装饰(数学)
		
题目传送门(内部题147) 输入格式 每个测试点第一行一个正整数$T$,表示该测试点内的数据组数. 接下来$T$行,每行三个非负整数$a,b,c$,含义如题目中所示. 输出格式 对每组数据输出一行一个 ...
 - [CSP-S模拟测试]:Six(数学)
		
题目传送门(内部题85) 输入格式 一个正整数$N$. 输出格式 一个数表示答案对$1000000007$取模后的结果 样例 样例输入1: 样例输出1: 样例输入2: 样例输出2: 样例输入3: 样例 ...
 - [CSP-S模拟测试]:chess(数学)
		
题目描述 $dirty$在一个棋盘上放起了棋子. 棋盘规格为$n\times m$,他希望任意一个$n\times n$的区域内都有$C$个棋子.$dirty$很快就放置好了一个满足条件的棋盘方案,但 ...
 - [CSP-S模拟测试]:旅行(数学+线段树)
		
题目传送门(内部题12) 输入格式 第一行,一个整数$n$,代表树的点数.第二行,$n$个整数,第$i$个整数是$B_i$,描述排列$B$.接下来$n−1$行,每行两个整数$u,v$,描述一条树边$( ...
 - [CSP-S模拟测试]:A(数学)
		
题目传送门(内部题44) 输入格式 一行四个整数,分别表示$S,T,a,b$. 输出格式 输出最小步数,数据保证有解. 样例 样例输入: 10 28 4 2 样例输出: 数据范围与提示 样例解释: 先 ...
 - [CSP-S模拟测试]:最大值(数学+线段树)
		
题目背景 $Maxtir$最喜欢最大值. 题目传送门(内部题128) 输入格式 第$1$行输入四个正整数$n,m,q$. 第$2$至$n+1$行中,第$i+1$行输入魔法晶石$i$的三种属性$(x_i ...
 - [CSP-S模拟测试]:求和(数学)
		
题目传送门(内部题107) 输入格式 一行五个正整数$x_1,y_1,x_2,y_2,m$ 输出格式 输出一个整数,为所求的答案对$m$取模后的结果. 样例 样例输入: 2 1 5 3 10007 样 ...
 - [CSP-S模拟测试]:数列(数学)
		
题目传送门(内部题95) 输入格式 第一行三个整数$n,a,b$,第二行$n$个整数$x_1\sim x_n$表示数列. 输出格式 一行一个整数表示答案.无解输出$-1$. 样例 样例输入:2 2 3 ...
 
随机推荐
- Java IO NIO详细讲解
			
1.IO Java IO概述 2.NIO Java NIO浅析
 - exists、in和join比较
			
这个根据实际情况具体分析 遇到问题了再具体分析就行.
 - 剑指Offer编程题(Java实现)——从尾到头打印链表
			
题目描述 输入一个链表,按链表值从尾到头的顺序返回一个ArrayList. 解题思路 思路一:使用头插法 使用头插法可以得到一个逆序的链表.遍历链表,每次将所遍历节点插入到链表的头部. 头结点和第一个 ...
 - Ubantu创建热点并共享——2019年5月10日更新
			
只需要两步,参考以下两篇文章: ubuntu16.04上安装配置DHCP服务的详细过程 Ubuntu18.04 创建与编辑热点的方法
 - 逆向与反汇编实战(一)--PEiD分析复现
			
1.准备 简介: PEiD(PE Identifier)是一款著名的查壳工具,其功能强大,几乎可以侦测出所有的壳,其数量已超过470 种PE 文档 的加壳类型和签名. 整个过程需要测试文件成品:htt ...
 - 在cmd下用cd怎么进不了其他的盘
			
你当前就是在C盘目录下的,可以切换到别的盘比如D:,然后在切换E:!然后可以切换C:,然后可以用cd\回到根目录. cd是打开文件根目录里面文件夹的,比如C:目录下可以cd Windows打开Wind ...
 - luogu P1399 [NOI2013]快餐店
			
传送门 注意到答案为这个基环树直径\(/2\) 因为是基环树,所以考虑把环拎出来.如果直径不过环上的边,那么可以在环上每个点下挂的子树内\(dfs\)求得.然后如果过环上的边,那么环上的部分也是一条链 ...
 - 【问题解决方案】CentOS7替换yum的问题:使用yum makecache出现File contains no section headers
			
参考链接 CentOS7替换yum的问题:使用yum时出现File contains no section headers centos安装网络repo源及错误说明 一.centos替换yum的步骤 ...
 - 关于jsp删除成功,添加成功等之后 页面自动跳转的js写法
			
因为比较常用,所以写在博客里保存起来,防止以后忘了不会写了: 删除成功,<span id="time" style="background:red"> ...
 - SVN本地搭建服务
			
本地搭建svn 服务端并使用 1.下载SVN服务端软 VisualSVN server 地址:http://subversion.apache.org/packages.html 2.安装 Visu ...