Codeforces Round #554 (Div. 2) F2. Neko Rules the Catniverse (Large Version) (矩阵快速幂 状压DP)
题意
有nnn个点,每个点只能走到编号在[1,min(n+m,1)][1,min(n+m,1)][1,min(n+m,1)]范围内的点。求路径长度恰好为kkk的简单路径(一个点最多走一次)数。
1≤n≤109,1≤m≤4,1≤k≤min(n,12)1\le n\le 10^9,1\le m\le 4,1\le k\le min(n,12)1≤n≤109,1≤m≤4,1≤k≤min(n,12)
分析
直接考虑走路径的话不能判有没有走过,然后就把路径转化为一个序列,每次往里面插入新的点(神了)。因为一个点可以走到比他小的所有点,那么我们把点从大到小插入。
假设现在已有序列为p1,p2,p3,...,pkp_1,p_2,p_3,...,p_kp1,p2,p3,...,pk。那么当前插入一个点iii。
假设插在pjp_jpj和pj+1p_{j+1}pj+1之间,必须满足pjp_jpj能走到iii并且iii能走到pj+1p_{j+1}pj+1。由于iii是最小的,那么所有pjp_jpj都能走到iii,所以只用考虑iii能走到哪些点。
- 一种情况是直接放在最后。
- 另一种情况是i+m≥pj+1i+m\ge p_{j+1}i+m≥pj+1。那么满足这个式子的pj+1p_{j+1}pj+1最多有m(≤4)m(\le 4)m(≤4)个。那么就把[i+1,i+m][i+1,i+m][i+1,i+m]这mmm个数有没有出现在序列中过状压成mmm位222进制数,记为SSS。当前方案就是bitcount(S)bitcount(S)bitcount(S)(SSS在222进制中有多少个111)。
那么一共就有bitcount(S)+1bitcount(S)+1bitcount(S)+1种方案。
另外还可以不插入。
所以DPDPDP状态设为f[i][j][S]f[i][j][S]f[i][j][S]表示到iii这个点,序列长度为jjj,上述状态为SSS的方案数,转移方程为:
f[i+1][j+1][(S<<1∣1)&(2m−1)]+=f[i][j][S]∗(bitcount(S)+1)f[i+1][j][(S<<1∣0)&(2m−1)]+=f[i][j][S]\begin{aligned}
f[i+1][j+1][(S<<1|1)\&(2^m-1)]&+=f[i][j][S]*(bitcount(S)+1)\\
f[i+1][j][(S<<1|0)\&(2^m-1)]&+=f[i][j][S]\end{aligned}f[i+1][j+1][(S<<1∣1)&(2m−1)]f[i+1][j][(S<<1∣0)&(2m−1)]+=f[i][j][S]∗(bitcount(S)+1)+=f[i][j][S]
由于nnn比较大,就矩阵加速就行了。这个矩阵快速幂还是挺好写的。。
CODE
#include <bits/stdc++.h>
using namespace std;
const int MAXN = 100005;
const int mod = 1e9 + 7;
int n, k, m, all;
inline void add(int &x, int y) { x += y; if(x >= mod) x -= mod; }
struct mat {
int a[210][210]; //最大状态数为(k+1)*(1<<m)<=(12+1)*(2^4)=208
mat() { memset(a, 0, sizeof a); }
inline mat operator *(const mat &o)const {
mat re;
for(int k = 0; k < all; ++k)
for(int i = 0; i < all; ++i) if(a[i][k])
for(int j = 0; j < all; ++j) if(o.a[k][j])
add(re.a[i][j], 1ll * a[i][k] * o.a[k][j] % mod);
return re;
}
inline mat operator ^(int b)const {
mat re, A = *this;
for(int i = 0; i < all; ++i) re.a[i][i] = 1;
while(b) {
if(b & 1) re = re * A;
A = A * A; b >>= 1;
}
return re;
}
};
inline int enc(int K, int S) { return K*(1<<m) + S; }
inline int nxt(int S, bool x) { return ((S<<1)|x) & ((1<<m)-1); }
int cnt[16];
int main () {
scanf("%d%d%d", &n, &k, &m);
all = (k+1)*(1<<m); //所有状态数
mat trans, ans;
ans.a[0][enc(0, 0)] = 1;
for(int s = 1; s < (1<<m); ++s) cnt[s] = cnt[s>>1] + (s&1); //预处理2进制下有多少个1
for(int i = 0; i <= k; ++i)
for(int s = 0; s < (1<<m); ++s) {
if(i < k) trans.a[enc(i, s)][enc(i+1, nxt(s, 1))] = cnt[s]+1;
trans.a[enc(i, s)][enc(i, nxt(s, 0))] = 1;
}
ans = ans * (trans ^ n);
int Ans = 0;
for(int s = 0; s < (1<<m); ++s)
add(Ans, ans.a[0][enc(k, s)]);
printf("%d\n", (Ans + mod) % mod);
}
Codeforces Round #554 (Div. 2) F2. Neko Rules the Catniverse (Large Version) (矩阵快速幂 状压DP)的更多相关文章
- CodeForces 1152F2 Neko Rules the Catniverse (Large Version)
题目链接:http://codeforces.com/problemset/problem/1152/F2 题目大意 见http://codeforces.com/problemset/problem ...
- Codeforces Round #554 (Div. 2) 1152B. Neko Performs Cat Furrier Transform
学了这么久,来打一次CF看看自己学的怎么样吧 too young too simple 1152B. Neko Performs Cat Furrier Transform 题目链接:"ht ...
- Codeforces Round #554 (Div. 2) 1152A - Neko Finds Grapes
学了这么久,来打一次CF看看自己学的怎么样吧 too young too simple 1152A - Neko Finds Grapes 题目链接:"https://codeforces. ...
- Codeforces Round #554 (Div. 2) C. Neko does Maths (简单推导)
题目:http://codeforces.com/contest/1152/problem/C 题意:给你a,b, 你可以找任意一个k 算出a+k,b+k的最小公倍数,让最小公倍数尽量小,求出 ...
- Codeforces Round #554 (Div. 2) C.Neko does Maths (gcd的运用)
题目链接:https://codeforces.com/contest/1152/problem/C 题目大意:给定两个正整数a,b,其中(1<=a,b<=1e9),求一个正整数k(0&l ...
- Codeforces Round #554 (Div. 2) C. Neko does Maths(数学+GCD)
传送门 题意: 给出两个整数a,b: 求解使得LCM(a+k,b+k)最小的k,如果有多个k使得LCM()最小,输出最小的k: 思路: 刚开始推了好半天公式,一顿xjb乱操作: 后来,看了一下题解,看 ...
- Codeforces Round #554 (Div. 2) B. Neko Performs Cat Furrier Transform(思维题+log2求解二进制位数的小技巧)
传送门 题意: 给出一个数x,有两个操作: ①:x ^= 2k-1; ②:x++; 每次操作都是从①开始,紧接着是② ①②操作循环进行,问经过多少步操作后,x可以变为2p-1的格式? 最多操作40次, ...
- Codeforces Round #554 (Div. 2) C. Neko does Maths (数论 GCD(a,b) = GCD(a,b-a))
传送门 •题意 给出两个正整数 a,b: 求解 k ,使得 LCM(a+k,b+k) 最小,如果有多个 k 使得 LCM() 最小,输出最小的k: •思路 时隔很久,又重新做这个题 温故果然可以知新❤ ...
- Codeforces Round #554 (Div. 2) E Neko and Flashback (欧拉路径 邻接表实现(当前弧优化..))
就是一欧拉路径 贴出邻接表欧拉路径 CODE #include <bits/stdc++.h> using namespace std; const int MAXN = 100005; ...
随机推荐
- Java基础---Java 开发工具IntelliJ IDEA 安装
1.1 开发工具概述IDEA是一个专门针对Java的集成开发工具(IDE),由Java语言编写.所以,需要有JRE运行环境并配置好环境变量.它可以极大地提升我们的开发效率.可以自动编译,检查错误.在公 ...
- java 中的容器(札记)
创建容器向上转型为接口的时候,有时候,并不是一定可行的,因为有的实现类,在接口的基础添加了自己的方法:比如:List 接口下面的 LinkedList 自己定义了一些方法 : Arrays.asLis ...
- nginx如何调用PHP(nginx+php运行原理)
采用nginx+php作为webserver的架构模式,在现如今运用相当广泛.然而第一步需要实现的是如何让nginx正确的调用php.由于nginx调用php并不是如同调用一个静态文件那么直接简单,是 ...
- Hibernate一对多自关联、多对多关联
今天分享hibernate框架的两个关联关系 多对多关系注意事项 一定要定义一个主控方 多对多删除 主控方直接删除 被控方先通过主控方解除多对多关系,再删除被控方 禁用级联删除 关联关系编辑,不 ...
- 小程序文案过长,‘收起/展开’文字,createSelectorQuery 获取节点成功,boundingClientRect 返回信息null
问题描述: wxml中id是动态生成的. 获取节点信息是在onReady生命周期函数内延时500ms执行的,select(id)可以获取全部节点信息,boundingClientRect (rect) ...
- Oracle VIP说明
本篇文档,描述说明VIP的作用 1.VIP全称 virtual ip 虚拟IP 2.Oracle为啥要搞个VIP 3.两节点RAC,集群单个节点故障关闭后,VIP漂移否继续对外提供服务 一.模拟RAC ...
- 音视频入门-02-RGB拼图
* 音视频入门文章目录 * 图片 & 像素点 & RGB 平时浏览的图片看不出像素点: 图片放大时,可以看出图片是一个个像素点组成的: 每个像素点的颜色可以用 RGB 表示: RGB ...
- java基础知识学习 内存相关
Java 内存分配策略 静态存储区(方法区):主要存放静态数据.全局 static 数据和常量.这块内存在程序编译时就已经分配好,并且在程序整个运行期间都存在. 栈区 :当方法被执行时,方法体内的局部 ...
- (十四)Activitivi5之个人任务分配
一.个人任务分配 1.1 方式一:直接流程图配置中写死: 1.2 方式二:使用流程变量 我们在启动流程的时候设置流程变量即可 /** * 启动流程实例 */ @Test public void sta ...
- springboot启动流程(四)application配置文件加载过程
所有文章 https://www.cnblogs.com/lay2017/p/11478237.html 触发监听器加载配置文件 在上一篇文章中,我们看到了Environment对象的创建方法.同时也 ...