题目描述:小度和小良最近又迷上了下棋。棋盘一共有N行M列,我们可以把左上角的格子定为(1,1),右下角的格子定为(N,M)。在他们的规则中,“王”在棋盘上的走法遵循十字路线。也就是说,如果“王”当前在(x,y)点,小度在下一步可以移动到(x+1, y), (x-1, y), (x, y+1), (x, y-1), (x+2, y), (x-2, y), (x, y+2), (x, y-2) 这八个点中的任意一个。

  小度觉得每次都是小良赢,没意思。为了难倒小良,他想出了这样一个问题:如果一开始“王”在(x0,y0)点,小良对“王”连续移动恰好K步,一共可以有多少种不同的移动方案?两种方案相同,当且仅当它们的K次移动全部都是一样的。也就是说,先向左再向右移动,和先向右再向左移动被认为是不同的方案。

  输入:输入包括多组数据。输入数据的第一行是一个整数T(T≤10),表示测试数据的组数。每组测试数据只包括一行,为五个整数N,M,K,x0,y0。(1≤N,M,K≤1000,1≤x0≤N,1≤y0≤M)

  输出:对于第k组数据,第一行输出Case #k:,第二行输出所求的方案数。由于答案可能非常大,你只需要输出结果对9999991取模之后的值即可。

  这个题目,开始做的时候挺纠结,最直接的思路就是从王的起始点一步一步的计算到达其它位置的方案数,不过复杂度太高,会超时,下面先说下自己开始的思路,然后通过这个思路改进来解决这道题目。定义一个数组dp[x][i][j](0<=x<=k,1<=i<=n,1<=j<=m),表示经过x步到达位置(i,j)的方案数,move[8][2] = {{1,0},{-1,0},{0,1},{0,-1},{2,0},{-2,0},{0,2},{0,-2}},那么dp[x+1][i][j] = sumo(dp[x][i - move[o][0]][j - move[o][1]])(0<=o<8),sumo表示以o变化来累加,最终只要将dp[k][i][j](1<=i<=n,1<=j<=m)都加就是总的方案数了。但由于k,n,m的最大值都为1000,所以直接定义dp[1001][1001][1001]占用的存储空间过大,再看前面的推导式子dp[x+1][i][j]只与dp[x][i'][j']有关,所以可以定义滚动数组dp[2][1001][1001]来节约空间。可以很容易的看出这个思路的复杂度是O(k*n*m),显然这个复杂度不能满足题目要求。

  可不可以根据上面思路优化下呢,最终我们需要计算dp[k][i][j](1<=i<=n,1<=j<=m)的和,前面的思路是一步一步经过k步来推导出dp[k][i][j],可以不可以用起来的方法求dp[k][i][j]呢?考虑到这是一个矩阵,移动的方向只能是行或者列,所有很自然想到行列步数的组合来求dp[k][i][j],即dp[k][i][j] = sumt(C[k][t] * dp1[t][i] * dp2[k - t][j]),dp1[t][i]表示在列方向经过t步到达i,dp2[k - t][j]表示在行方向经过k - t步到达j,C[k][t]是组合数。计算dp1和dp2可以通过前面一步一步推的思路求解,只是维度变成一维的了,复杂度分别为O(k * n), O(k * m)。求最终结果也是将所有dp[k][i][j](1<=i<=n,1<=j<=m)加起来,这个计算式和前面不同的是由于行列式独立的,所以最终的ans = sumi(sumj(sumt(C[k][t] * dp1[t][i] * dp2[k - t][j]))) = sumt(C[k][t] * sumi(dp1[t][i]) * sumj(dp2[k - t][j])),C[k][t]可以根据组合数的性质C[k][t] = C[k - 1][t] + C[k - 1][t - 1]求得,所以这个思路的算法复杂度为O(max( k * n, k * m, k^2)),可以满足题目要求了。

  具体代码如下:

 #include <iostream>
#include <cstring>
using namespace std; int n, m, k, x0, y0;
long long dp1[][], dp2[][], c[][];
int move[] = {, -, , -}; void dp(long long dp[][], int start, int end, int x)
{
dp[][x] = ;
for (int i = ; i < k; i++)
{
for (int j = start; j <= end; j++)
if (dp[i][j])
{
for (int k = ; k < ; k++)
{
int ii = j + move[k];
if (ii >= start && ii <= end)
dp[i + ][ii] = (dp[i + ][ii] + dp[i][j]) % ;
}
}
}
}
void pl()
{
for (int i = ; i < ; i++)
c[i][] = ;
for (int i = ; i < ; i++)
for (int j = ; j <= i ; j++)
c[i][j] = (c[i - ][j - ] + c[i - ][j]) % ;
} int main()
{
int t;
int ans;
pl();
cin >> t;
for (int i = ; i <= t; i++)
{
cin >> n >> m >> k >> x0 >> y0; memset(dp1, , sizeof(dp1));
memset(dp2, , sizeof(dp2));
dp(dp1, , n, x0);
dp(dp2, , m, y0);
ans = ;
for (int j = ; j <= k; j++)
{
for (int ii = ; ii <= n; ii++)
dp1[j][] = (dp1[j][] + dp1[j][ii]) % ;
for (int ii = ; ii <= m; ii++)
dp2[j][] = (dp2[j][] + dp2[j][ii] ) % ;
}
for (int j = ; j <= k; j++)
ans = (ans + (((c[k][j] * dp1[j][]) % ) * dp2[k - j][]) % ) % ;
cout << "Case #" << i << ":" << endl << ans << endl;
}
return ;
}

2014年百度之星程序设计大赛 - 初赛(第二轮)Chess的更多相关文章

  1. 2014年百度之星程序设计大赛 - 初赛(第二轮)JZP Set

    题目描述:一个{1, ..., n}的子集S被称为JZP集,当且仅当对于任意S中的两个数x,y,若(x+y)/2为整数,那么(x+y)/2也属于S.例如,n=3,S={1,3}不是JZP集,因为(1+ ...

  2. HDU 4834 JZP Set(数论+递推)(2014年百度之星程序设计大赛 - 初赛(第二轮))

    Problem Description 一个{1, ..., n}的子集S被称为JZP集,当且仅当对于任意S中的两个数x,y,若(x+y)/2为整数,那么(x+y)/2也属于S.例如,n=3,S={1 ...

  3. HDU 4833 Best Financing(DP)(2014年百度之星程序设计大赛 - 初赛(第二轮))

    Problem Description 小A想通过合理投资银行理财产品达到收益最大化.已知小A在未来一段时间中的收入情况,描述为两个长度为n的整数数组dates和earnings,表示在第dates[ ...

  4. HDU 4832 Chess(DP+组合数学)(2014年百度之星程序设计大赛 - 初赛(第二轮))

    Problem Description 小度和小良最近又迷上了下棋.棋盘一共有N行M列,我们可以把左上角的格子定为(1,1),右下角的格子定为(N,M).在他们的规则中,“王”在棋盘上的走法遵循十字路 ...

  5. 2014年百度之星程序设计大赛 - 初赛(第一轮) hdu Grids (卡特兰数 大数除法取余 扩展gcd)

    题目链接 分析:打表以后就能发现时卡特兰数, 但是有除法取余. f[i] = f[i-1]*(4*i - 2)/(i+1); 看了一下网上的题解,照着题解写了下面的代码,不过还是不明白,为什么用扩展g ...

  6. 2014年百度之星程序设计大赛 - 资格赛 第二题 Disk Schedule

    双调欧几里得旅行商问题是一个经典动态规划问题.<算法导论(第二版)>思考题15-1和北京大学OJ2677都出现了这个题目. 旅行商问题描写叙述:平面上n个点,确定一条连接各点的最短闭合旅程 ...

  7. HDU6383 2018 “百度之星”程序设计大赛 - 初赛(B) 1004-p1m2 (二分)

    原题地址 p1m2 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 131072/131072 K (Java/Others)Total ...

  8. HDU6380 2018 “百度之星”程序设计大赛 - 初赛(B) A-degree (无环图=树)

    原题地址 degree Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 131072/131072 K (Java/Others)Tot ...

  9. HDU 6118 度度熊的交易计划 【最小费用最大流】 (2017"百度之星"程序设计大赛 - 初赛(B))

    度度熊的交易计划 Time Limit: 12000/6000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total S ...

随机推荐

  1. spring动态代理

    接下来我们来体会下动态代理带给我们的便利 package aop006; public interface Girl { public void KFC(String datetime); publi ...

  2. 递归一题总结(OJ P1117倒牛奶)

    题目:                    农民约翰有三个容量分别是A,B,C升的桶,A,B,C分别是三个从1到20的整数,最初,A和B桶都是空的,而C桶是装满牛奶的.有时,约翰把牛奶从一个桶倒到另 ...

  3. java的poi技术读取和导入Excel

    项目结构: http://www.cnblogs.com/hongten/gallery/image/111987.html  用到的Excel文件: http://www.cnblogs.com/h ...

  4. 初探SQL注入

    1.1注入语句(通过时间注入函数) 数据库名称 localhost:8080/ScriptTest/userServlet?username='union SELECT IF(SUBSTRING(cu ...

  5. [.net 面向对象程序设计进阶] (14) 缓存(Cache) (一) 认识缓存技术

    [.net 面向对象程序设计进阶] (14) 缓存(Cache)(一) 认识缓存技术 本节导读: 缓存(Cache)是一种用空间换时间的技术,在.NET程序设计中合理利用,可以极大的提高程序的运行效率 ...

  6. 使用VS+VisualGDB编译调试Linux程序

    Linux程序开发变得越来越多,越来越多的程序.产品需要跨平台,甚至有些开源项目只支持Linux平台,所以掌握Linux开发变得越来越重要. 但是对于习惯了Windows下的开发,使用了VS这个宇宙第 ...

  7. Log4net入门使用

    简介 几乎所有的大型应用都会有自己的用于跟踪调试的API.因为一旦程序被部署以后,就不太可能再利用专门的调试工具了.然而一个管理员可能需要有一套强大的日志系统来诊断和修复配置上的问题. 经验表明,日志 ...

  8. 《深入浅出Nodejs》—— 读后总结

    这一个月过去了三分之二,加上之前看过这本书三分之一,这才算是看完. 虽然看完一遍,但是这本书内容很深,以后肯定是还要继续翻阅的..... 什么是Nodejs Nodejs有几个特性:异步IO,事件驱动 ...

  9. Oralce 重建EM服务,OracleDBConsoleOracle

    今天打开Oracle ,想进em看看,结果发现OracleDBConsoleOracle服务老是报错: OracleDBConsoleOracle 服务因 2 (0x2) 服务性错误而停止. 搞不懂, ...

  10. iOS-APP发布应注意

    地区范围文件不用填 电话号码前缀是86