https://vijos.org/p/1067

守望者-warden,长期在暗夜精灵的的首都艾萨琳内担任视察监狱的任务,监狱是成长条行的,守望者warden拥有一个技能名叫“闪烁”,这个技能可以把她传送到后面的监狱内查看,她比较懒,一般不查看完所有的监狱,只是从入口进入,然后再从出口出来就算完成任务了。

描述

头脑并不发达的warden最近在思考一个问题,她的闪烁技能是可以升级的,k级的闪烁技能最多可以向前移动k个监狱,一共有n个监狱要视察,她从入口进去,一路上有n个监狱,而且不会往回走,当然她并不用每个监狱都视察,但是她最后一定要到第n个监狱里去,因为监狱的出口在那里,但是她并不一定要到第1个监狱。

守望者warden现在想知道,她在拥有k级闪烁技能时视察n个监狱一共有多少种方案?

格式

输入格式

第一行是闪烁技能的等级k(1<=k<=10)
第二行是监狱的个数n(1<=n<=2^31-1)

输出格式

由于方案个数会很多,所以输出它 mod 7777777后的结果就行了

样例1

样例输入1[复制]

 
  1. 2
  2. 4

样例输出1[复制]

 
  1. 5

限制

各个测试点1s

提示

把监狱编号1 2 3 4,闪烁技能为2级,
一共有5种方案
→1→2→3→4
→2→3→4
→2→4
→1→3→4
→1→2→4

小提示:建议用int64,否则可能会溢出

分析:快被自己蠢哭了,下午5点开始做这道题一直到现在,9点半。
首先这道题要递推,得出递推式 f[n] = f[n-k] + f[n-k+1] + ...+f[n-1];怎么退出的呢?到n点分为起点从1开始,从2开始...从n-k(一个k就到n点了)开始,从1开始的方案数就是f[n-1],从2开始的就相当于从1到n-1,所以可以写成f[n-2],而从n-k点开始的呢,就直接是f[n-k];
求这个递推式用矩阵来构造举例构造:f[n] = f[n-1] + f[n-2]+f[n-3]
可以写成| 0 1 0 |    | f[n-3] |     | f[n-2] |
           | 0 0 1 | * | f[n-2] | =  | f[n-1] |
           | 1 1 1 |    | f[n-1] |     | f[n-3] + f[n-2] + f[n-1] |
同理原式可以写成 k * K的矩阵,右上角的(k - 1)*(k - 1)主对角线都是一,第k行全为1;
然后就可以写最原始的那个式子了,即:f[1]...f[k],
最后根据n与k的关系分两种情况,k>=n,直接输出,n>k,利用矩阵快速幂求最后的变化矩阵,然后在用他左乘最原始的式子
弱,弱爆了。。。
  1. #include <iostream>
  2. #include <cstring>
  3. #include <algorithm>
  4. #include <cstdio>
  5. using namespace std;
  6.  
  7. int n,k;
  8. const int mod = ;
  9. struct Mat
  10. {
  11. __int64 mat[][];
  12. };
  13. Mat operator * (Mat a, Mat b)
  14. {
  15. Mat c;
  16. memset(c.mat, , sizeof(c.mat));
  17. for(int t = ; t <= k; t++)
  18. {
  19. for(int i = ; i <= k; i++)
  20. {
  21. for(int j = ; j <= k; j++)
  22. c.mat[i][j] = ( c.mat[i][j] + a.mat[i][t] % mod * ( b.mat[t][j] % mod ) ) % mod; //就是这里乘法忘了+c.mat[i][j]查了半个小时错,弱爆了
  23. }
  24. }
  25. return c;
  26. }
  27. Mat operator ^ (Mat a, int t)
  28. {
  29. Mat c;
  30. for(int i = ; i <= k; i++)
  31. for(int j = ; j <= k; j++)
  32. c.mat[i][j] = (i == j);
  33. while(t)
  34. {
  35. if(t & )
  36. c = a * c;
  37. a = a * a;
  38. t >>= ;
  39. }
  40. return c;
  41. }
  42. int main()
  43. {
  44. scanf("%d%d", &k,&n);
  45. Mat ori,temp,res;
  46. memset(ori.mat, , sizeof(ori.mat));
  47. for(int i = ; i <= k; i++)
  48. {
  49. for(int j = ; j < i; j++) //这里是处理最原始的那个式子;假设求f[4] ,那么起点可以前3点作为起点,这个好像也能从自己作为起点,所以有个自加,就像f[1] = 1;
  50. {
  51. ori.mat[i][] = (ori.mat[j][] + ori.mat[i][]) % mod;
  52. }
  53. ori.mat[i][]++;
  54. }
  55. if(n <= k)
  56. {
  57. printf("%I64d\n",ori.mat[n][] % mod);
  58. }
  59. else
  60. {
  61. memset(temp.mat, , sizeof(temp.mat));
  62. for(int i = ; i < k; i++)
  63. {
  64. temp.mat[i][i + ] = ;
  65. }
  66. for(int i = ; i <= k; i++)
  67. temp.mat[k][i] = ;
  68. res = temp ^ (n - k);
  69. ori = res * ori;
  70. printf("%I64d\n", ori.mat[k][] % mod);
  71. }
  72. return ;
  73. }

P1067Warcraft III 守望者的烦恼(十大矩阵问题之七求递推式)的更多相关文章

  1. Vijos P1067Warcraft III 守望者的烦恼

    题目 背景 守望者-warden,长期在暗夜精灵的的首都艾萨琳内担任视察监狱的任务,监狱是成长条行的,守望者warden拥有一个技能名叫“闪烁”,这个技能可以把她传送到后面的监狱内查看,她比较懒,一般 ...

  2. [Vijos1067]Warcraft III 守望者的烦恼(DP + 矩阵优化)

    传送门 可知 f[i] = f[i - 1] + f[i - 2] + ... + f[i - k] 直接矩阵优化就好了 #include <cstdio> #include <cs ...

  3. vijosP1067Warcraft III 守望者的烦恼

    vijosP1067Warcraft III 守望者的烦恼 链接:https://vijos.org/p/1067 [思路] 矩阵乘法. 可以得出递推式:      f[i]=sum{ f[n-1], ...

  4. C++矩阵加速经典题目:Warcraft III 守望者的烦恼 [vijos 1067]

    Warcraft III 守望者的烦恼 背景 守望者-warden,长期在暗夜精灵的的首都艾萨琳内担任视察监狱的任务,监狱是成长条行的,守望者warden拥有一个技能名叫"闪烁", ...

  5. 矩阵经典题目七:Warcraft III 守望者的烦恼(矩阵加速递推)

    https://www.vijos.org/p/1067 非常easy推出递推式f[n] = f[n-1]+f[n-2]+......+f[n-k]. 构造矩阵的方法:构造一个k*k的矩阵.当中右上角 ...

  6. vijos Warcraft III 守望者的烦恼

    题解 转移方程好写吧 一个一维递推式 然后我们可以构造矩阵优化 嗯,最近学一下递推优化 代码 #include<cstdio> #include<cstring> #inclu ...

  7. HDU5950 Recursive sequence (矩阵快速幂加速递推) (2016ACM/ICPC亚洲赛区沈阳站 Problem C)

    题目链接:传送门 题目: Recursive sequence Time Limit: / MS (Java/Others) Memory Limit: / K (Java/Others) Total ...

  8. HDU5950 Recursive sequence 非线性递推式 矩阵快速幂

    题目传送门 题目描述:给出一个数列的第一项和第二项,计算第n项. 递推式是 f(n)=f(n-1)+2*f(n-2)+n^4. 由于n很大,所以肯定是矩阵快速幂的题目,但是矩阵快速幂只能解决线性的问题 ...

  9. Ipad,IPhone(矩阵求递推项+欧拉定理)

    Ipad,IPhone Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Total S ...

随机推荐

  1. 【ArcGIS二次开发】CreateFeature报错(HRESULT E_FAIL)

    在VS2010下基于ArcGIS 10.1做二次开发时遇到个奇怪的问题,对于MXD工程文档中已经存在的图层,获取其FeatureClass可以调用CreateFeature()函数生成要素,但是对于通 ...

  2. 使用c#创建php可以调用的dll

    1. 创建一个 C# Class Library ,命名为:ClassLibraryDemo 2. 打开项目的属性,在点选左边的 “Application”(就是第一个tab) , 然后点击 Asse ...

  3. MySQL学习指引

    mysql指引 1,mysql基本安装 2,mysql多实例安装与维护 3,备份恢复 备份数据库 分备数据库 分备表 恢复数据库

  4. C语言 const与指针

    //const与指针 #include<stdio.h> #include<stdlib.h> void main(){ ; ; const int *p; p = & ...

  5. oracle学习之表空间

    一.oracle当中的dual表 注意:sql语句一定要有一个 : 结尾,不然会报错. Oracle数据库内种特殊表DualDual表Oracle实际存表任何用户均读取用没目标表SelectDual表 ...

  6. ASP.NET中进行消息处理(MSMQ) 二

    在我上一篇文章<ASP.NET中进行消息处理(MSMQ)一>里对MSMQ做了个通俗的介绍,最后以发送普通文本消息和复杂的对象消息为例介绍了消息队列的使用. 本文在此基础上继续介绍MSMQ的 ...

  7. SSM三大框架整合详细教程(Spring+SpringMVC+MyBatis)

    使用 SSM ( Spring . SpringMVC 和 Mybatis )已经有三个多月了,项目在技术上已经没有什么难点了,基于现有的技术就可以实现想要的功能,当然肯定有很多可以改进的地方.之前没 ...

  8. Java中sleep()和wait()的区别

    1. 这两个方法来自不同的类,sleep方法属于Thread,wait方法属于Object. 2. 最主要是sleep方法没有释放锁,而wait方法释放了锁,使得其他线程可以使用同步控制块或者方法. ...

  9. (转)shell实例手册

    原文地址:http://hi.baidu.com/quanzhou722/item/f4a4f3c9eb37f02d46d5c0d9 实在是太好的资料了,不得不转 shell实例手册 0说明{ 手册制 ...

  10. 性能指标TP99之我解

    首先给出Google到的答案: The tp90 is a minimum time under which 90% of requests have been served. tp90 = top ...