题目描述

加里敦星球的人们特别喜欢喝可乐。因而,他们的敌对星球研发出了一个可乐机器人,并且放在了加里敦星球的\(1\)号城市上。这个可乐机器人有三种行为:停在原地,去下一个相邻的城市,自爆。它每一秒都会随机触发一种行为。现在给出加里敦星球城市图,在第\(0\)秒时可乐机器人在\(1\)号城市,问经过了\(t\)秒,可乐机器人的行为方案数是多少?

输入格式

第一行输入两个正整数\(N,M\),\(N\)表示城市个数,\(M\)表示道路个数。(\(1≤N≤30\),\(0≤M≤100\))

接下来M行输入\(u,v\),表示\(u,v\)之间有一条道路。(\(1≤u,v≤n\))保证两座城市之间只有一条路相连。

最后输入时间\(t\)。

输出格式

输出可乐机器人的行为方案数,答案可能很大,请输出对\(2017\)取模后的结果。

数据范围

对于20%的数据,有\(1<t≤1000\)

对于100%的数据,有\(1<t≤10^6\)。


一开始看着数据范围还以为是一道卡常数题啊......

这个保证\(t≤10^6\),直觉不会想到\(logt\)啊QAQ


显然,如果这道题目是卡常数题的话,那就是一道很简单的小学生dp了

\(f[i][j][0]\)表示第\(i\)时刻,在城市\(j\)爆炸的方案数,\(f[i][j][1]\)表示第\(i\)时刻,在城市\(j\)不爆炸的方案数

\(f[i][j][0]=f[i-1][j][0]+f[i-1][j][1]\),\(f[i][j][1]=f[i][j][1]+f[i][k][1]\)(\(k\)是\(j\)的相邻结点)

这样是要MLE的,显然状态的答案只与前一状态相关,直接滚动即可

事实证明,这样只有20分


看起来这是一阶递推,考虑用矩乘来优化

先考虑\(f[i][j][1]=f[i][j][1]+f[i][k][1]\),可以发现,如果我们假设\(i\)结点和自己本身相邻的话,可以将前一项并到后一项当中

我们可以用一个\(n*n\)的矩阵\(A\)和\(f[0][?][1]\)构成的矩阵\(B\)相乘得到,而\(n*n\)的矩阵是表示是否直接相邻的邻接矩阵,特别的\(A[i][i]=1\)

这样的话,我们对\(A\)矩阵快速幂即可

继续考虑\(f[i][j][0]=f[i-1][j][0]+f[i-1][j][1]\),我们可以发现,这其实是\(\sum f[i-1][j][1]\),按照前一个幂的处理,可以发现,这其实是一个矩阵的等比数列

直接按照等比数列求个和就好了咯

时间复杂度\(O(nlogn)\)

#include<cstdio>
#include<iostream>
#include<algorithm>
#include<cstdlib>
#include<cstring>
#include<string>
#include<climits>
#include<vector>
#include<cmath>
#include<map>
#define LL long long using namespace std; inline char nc(){
static char buf[100000],*p1=buf,*p2=buf;
if (p1==p2) { p2=(p1=buf)+fread(buf,1,100000,stdin); if (p1==p2) return EOF; }
return *p1++;
} inline void read(int &x){
char c=nc();int b=1;
for (;!(c>='0' && c<='9');c=nc()) if (c=='-') b=-1;
for (x=0;c>='0' && c<='9';x=x*10+c-'0',c=nc()); x*=b;
} inline void read(LL &x){
char c=nc();LL b=1;
for (;!(c>='0' && c<='9');c=nc()) if (c=='-') b=-1;
for (x=0;c>='0' && c<='9';x=x*10+c-'0',c=nc()); x*=b;
} inline int read(char *s)
{
char c=nc();int len=0;
for(;!((c>='A' && c<='Z')||(c>='a' && c<='z'));c=nc()) if (c==EOF) return 0;
for(;((c>='A' && c<='Z')||(c>='a' && c<='z'));s[len++]=c,c=nc());
s[len++]='\0';
return len;
} inline void read(char &x){
for (x=nc();!(x=='?' || x=='+' || x=='-');x=nc());
} int wt,ss[19];
inline void print(int x){
if (x<0) x=-x,putchar('-');
if (!x) putchar(48); else {
for (wt=0;x;ss[++wt]=x%10,x/=10);
for (;wt;putchar(ss[wt]+48),wt--);}
}
inline void print(LL x){
if (x<0) x=-x,putchar('-');
if (!x) putchar(48); else {for (wt=0;x;ss[++wt]=x%10,x/=10);for (;wt;putchar(ss[wt]+48),wt--);}
} int n,m,T,f[40],g[40];
const int mo=2017;
struct data
{
int f[40][40];
}a,b; void add(data &x,data y)
{
for (int i=1;i<=n;i++)
for (int j=1;j<=n;j++)
x.f[i][j]=(x.f[i][j]+y.f[i][j])%mo;
} void mul(data &x,data y)
{
data z;
memset(z.f,0,sizeof(z.f));
for (int i=1;i<=n;i++)
for (int j=1;j<=n;j++)
for (int k=1;k<=n;k++)
z.f[i][j]=((x.f[i][k]*y.f[k][j])%mo+z.f[i][j])%mo;
x=z;
} data Power(data x,int y)
{
data res;
memset(res.f,0,sizeof(res.f));
for (int i=1;i<=n;i++)
res.f[i][i]=1;
for (;y;y>>=1)
{
if (y&1) mul(res,x);
mul(x,x);
}
return res;
} data Sum(data x,int y)
{
data res;
memset(res.f,0,sizeof(res.f));
if (y==0) return res;
if (y==1) return x;
for (int i=1;i<=n;i++)
res.f[i][i]=1;
add(res,Power(x,y>>1));mul(res,Sum(x,y>>1));
if (y&1) add(res,Power(x,y));
return res;
} int main()
{
read(n);read(m);
int x,y;
memset(a.f,0,sizeof(a.f));
for (int i=1;i<=m;i++)
read(x),read(y),a.f[x][y]=1,a.f[y][x]=1;
for (int i=1;i<=n;i++)
a.f[i][i]=1;
read(T);
b=a;
a=Power(a,T);
memset(g,0,sizeof(g));
memset(f,0,sizeof(f));
g[1]=1;
for (int i=1;i<=n;i++)
for (int j=1;j<=n;j++)
f[i]=((g[j]*a.f[i][j])%mo+f[i])%mo;
int ans=0;
for (int i=1;i<=n;i++)
ans=(ans+f[i])%mo;
b=Sum(b,T-1);
for(int i=1;i<=n;i++)
b.f[i][i]=(b.f[i][i]+1)%mo;
memset(g,0,sizeof(g));
memset(f,0,sizeof(f));
g[1]=1;
for (int i=1;i<=n;i++)
for (int j=1;j<=n;j++)
f[i]=((g[j]*b.f[i][j])%mo+f[i])%mo;
for (int i=1;i<=n;i++)
ans=(ans+f[i])%mo;
print(ans),puts("");
return 0;
}

【TJOI2017】可乐的更多相关文章

  1. 【BZOJ4887】[TJOI2017]可乐(矩阵快速幂)

    [BZOJ4887][TJOI2017]可乐(矩阵快速幂) 题面 BZOJ 洛谷 题解 模板题??? #include<iostream> #include<cstdio> # ...

  2. 【bzoj4887】:[Tjoi2017]可乐 矩阵乘法,快速幂

    [bzoj4887]:[Tjoi2017]可乐 题目大意:一张无相连通图(n<=30),从1号点开始走,每秒可以走到相邻的点也可以自爆,求第t秒(t<=1e6)后所有的方案数是多少对201 ...

  3. 洛谷P3758 - [TJOI2017]可乐

    Portal Description 给出一张\(n(n\leq30)\)个点\(m(m\leq100)\)条边的无向图.初始时有一个可乐机器人在点\(1\),这个机器人每秒会做出以下三种行为之一:原 ...

  4. [TJOI2017]可乐

    题目描述 加里敦星球的人们特别喜欢喝可乐.因而,他们的敌对星球研发出了一个可乐机器人,并且放在了加里敦星球的1号城市上.这个可乐机器人有三种行为: 停在原地,去下一个相邻的城市,自爆.它每一秒都会随机 ...

  5. Luogu 3758 [TJOI2017]可乐(有向图邻接矩阵幂的意义 矩阵快速幂)

    题目描述 加里敦星球的人们特别喜欢喝可乐.因而,他们的敌对星球研发出了一个可乐机器人,并且放在了加里敦星球的1号城市上.这个可乐机器人有三种行为: 停在原地,去下一个相邻的城市,自爆.它每一秒都会随机 ...

  6. BZOJ4887: [Tjoi2017]可乐 矩阵快速幂

    Description 加里敦星球的人们特别喜欢喝可乐.因而,他们的敌对星球研发出了一个可乐机器人,并且 放在了加里敦星球的1号城市上.这个可乐机器人有三种行为:停在原地,去下一个相邻的 城市,自爆. ...

  7. P3758 [TJOI2017]可乐

    题目描述 加里敦星球的人们特别喜欢喝可乐.因而,他们的敌对星球研发出了一个可乐机器人,并且放在了加里敦星球的1号城市上.这个可乐机器人有三种行为: 停在原地,去下一个相邻的城市,自爆.它每一秒都会随机 ...

  8. 洛谷P3758/BZOJ4887 [TJOI2017] 可乐 [矩阵快速幂]

    洛谷传送门,BZOJ传送门 可乐 Time Limit: 10 Sec  Memory Limit: 64 MBSubmit: 299  Solved: 207 Description 加里敦星球的人 ...

  9. [BZOJ4887][TJOI2017]可乐(DP+矩阵快速幂)

    题目描述 加里敦星球的人们特别喜欢喝可乐.因而,他们的敌对星球研发出了一个可乐机器人,并且放在了加里敦星球的1号城市上.这个可乐机器人有三种行为: 停在原地,去下一个相邻的城市,自爆.它每一秒都会随机 ...

  10. BZOJ4887:[TJOI2017]可乐(矩阵乘法)

    Description 加里敦星球的人们特别喜欢喝可乐.因而,他们的敌对星球研发出了一个可乐机器人,并且 放在了加里敦星球的1号城市上.这个可乐机器人有三种行为:停在原地,去下一个相邻的 城市,自爆. ...

随机推荐

  1. python-函数基础、函数参数

    目录 函数的基础 什么是函数 为何用函数 如何调用函数 定义函数的三种形式 无参函数 有参函数 空函数 函数的返回值 什么是返回值 为什么要有返回值 函数的调用 函数参数的应用 形参和实参 位置参数 ...

  2. STM32的IAP方案

    from:   http://bbs.eeworld.com.cn/thread-294115-1-1.html 几乎所有的同类书籍都介绍综合性的应用示例如“万年历 + 温度显示 + 闹钟响铃 + 计 ...

  3. Linux学习-SRPM 的使用 : rpmbuild (Optional)

    新版的 rpm 已经 将 RPM 与 SRPM 的指令分开了,SRPM 使用的是 rpmbuild 这个指令,而不是 rpm 喔! 利用默认值安装 SRPM 文件 (--rebuid/--recomp ...

  4. Monkeyrunner 简介及其环境搭建

    Monkeyrunner是通过坐标.控件ID和控件上的文字操作应用的界面元素,其测试用例是用python写的,这样就弥补了monkey只有简单命令无法执行复杂用例的缺陷.Monkeyrunner采用的 ...

  5. 光学字符识别OCR-2

    灰度聚类 接着我们就对图像的色彩进行聚类.聚类的有两个事实依据:         1.灰度分辨率   肉眼的灰度分辨率大概为40,因此对于像素值254和255,在我们肉眼看来都 只是白色:       ...

  6. day05_02 IDE介绍及设置

    notepad++比较麻烦,使用IDE工具进行程序开发 集成开发环境(IDE,Integrated Development Environment) VIM #经典的linux下的文本编辑器 Emac ...

  7. Leetcode 467.环绕字符串中的唯一子字符串

    环绕字符串中的唯一子字符串 把字符串 s 看作是"abcdefghijklmnopqrstuvwxyz"的无限环绕字符串,所以 s 看起来是这样的:"...zabcdef ...

  8. install cinnamon on ubuntu 14.04

    emotion: I feel not comfortable with ubuntu 14.04 default desktop unity,i still look for a alternati ...

  9. Welcome-to-Swift-24高级运算符(Advanced Operators)

    除了基本操作符中所讲的运算符,Swift还有许多复杂的高级运算符,包括了C语和Objective-C中的位运算符和移位运算. 不同于C语言中的数值计算,Swift的数值计算默认是不可溢出的.溢出行为会 ...

  10. [luoguP2754] 星际转移问题(最大流)

    传送门 不同的时间每个飞船所在的地点不同,给我们启示按照时间构建分层图. 同一个地点 x <x, dayi - 1> -> <x, dayi> 连一条容量为 INF 的边 ...