[BZOJ1494] [NOI2007]生成树计数

Description

最近,小栋在无向连通图的生成树个数计算方面有了惊人的进展,他发现:·n个结点的环的生成树个数为n。·n个结点的完全图的生成树个数为n^(n-2)。这两个发现让小栋欣喜若狂,由此更加坚定了

他继续计算生成树个数的想法,他要计算出各种各样图的生成树数目。一天,小栋和同学聚会,大家围坐在一张大圆桌周围。小栋看了看,马上想到了生成树问题。

如果把每个同学看成一个结点,邻座(结点间距离为1)的同学间连一条边,就变成了一个环。可是,小栋对环的计数已经十分娴熟且不再感兴趣。于是,小栋又把图变了一下:不仅把邻座的同学之间连

一条边,还把相隔一个座位(结点间距离为2)的同学之间也连一条边,将结点间有边直接相连的这两种情况统称为有边相连。



如图1所示。小栋以前没有计算过这类图的生成树个数,但是,他想起了老师讲过的计算任意图的生成树个数的一种通用方法:构造一个n×n的矩阵A={aij},其中其中di表示结点i的度数。与图1相应的A

矩阵如下所示。为了计算图1所对应的生成数的个数,只要去掉矩阵A的最后一行和最后一列,得到一个(n-1)×(n-1)的矩阵B,计算出矩阵B的行列式的值便可得到图1的生成树的个数所以生成树的个数为

|B|=3528。小栋发现利用通用方法,因计算过于复杂而很难算出来,而且用其他方法也难以找到更简便的公式进行计算。于是,他将图做了简化,从一个地方将圆桌断开,这样所有的同学形成了一条链

,连接距离为1和距离为2的点。例如八个点的情形如下:这样生成树的总数就减少了很多。小栋不停的思考,一直到聚会结束,终于找到了一种快捷的方法计算出这个图的生成树个数。可是,如果把距

离为3的点也连起来,小栋就不知道如何快捷计算了。现在,请你帮助小栋计算这类图的生成树的数目。

Input

包含两个整数k,n,由一个空格分隔。k表示要将所有距离不超过k(含k)的结点连接起来,n表示有n个结点。

Output

输出一个整数,表示生成树的个数。由于答案可能比较大,所以你 只要输出答案除65521 的余数即可。

Sample Input

3 5

Sample Output

75

试题分析

发现k很小,所以可以用最小表示法表示前k个点加上k+1个点的连通性,然后因为m比较大所以需要矩阵快速幂。

代码比较挫,在此理一下思路:

  • 把所有最小表示法预处理,并算出每个联通情况的总数。
  • 考虑每一个状态加上一个点枚举连边情况,并判断:无环,第一个点(会被删掉的)是否已经与后面还会考虑的某一个联通。
  • 加入连通性的时候转化为最小表示法。
  • 矩阵快速幂。
#include<iostream>
#include<cstring>
#include<cstdio>
#include<vector>
#include<algorithm> using namespace std;
#define LL long long inline LL read(){
LL x=0,f=1; char c=getchar();
for(;!isdigit(c);c=getchar()) if(c=='-') f=-1;
for(;isdigit(c);c=getchar()) x=x*10+c-'0';
return x*f;
}
const LL INF = 2147483600;
const LL MAXN = 100010;
const LL Mod = 65521; LL N,M; struct Mat{
LL x,y; LL a[101][101];
inline void init(LL rr,LL cc){x=rr; y=cc; memset(a,0,sizeof(a));}
}A,Ans; LL Pw[11]; LL sta[101],top;
Mat operator * (Mat A,Mat B){
Mat C; C.init(A.x,B.y);
for(LL i=1;i<=A.x;i++){
for(LL j=1;j<=B.y;j++){
for(LL k=1;k<=A.y;k++)
C.a[i][j]+=A.a[i][k]*B.a[k][j]%Mod,C.a[i][j]%=Mod;
}
} return C;
}
Mat operator ^ (Mat A,LL P){
Mat B; B.init(A.x,A.y); for(LL i=0;i<=A.x;i++) B.a[i][i]=1;
for(; P ; P>>=1,A=A*A) if(P&1) B=A*B; return B;
}
inline LL Pow(LL A,LL B){
LL res = 1; for(; B ; B>>=1,A=A*A%Mod) if(B&1) res=res*A%Mod; return res;
} LL Pos[MAXN+1];
inline void Get_state(LL k){
for(LL i=0;i<=top;i++) sta[i]=0;
top=0; while(k){
sta[++top]=k%10,k/=10;
} while(top<N) sta[top+1]=0,++top; reverse(sta+1,sta+N+1); return ;
} LL cnt; LL state[MAXN+1][11]; LL vis[13];
inline void Add(LL k){
Get_state(k); memset(vis,0,sizeof(vis)); LL now=0;
for(LL i=1;i<=N;i++){
if(!vis[sta[i]]) ++now; vis[sta[i]]++;
if(now-1<sta[i]) return ;
} ++cnt; LL res=1; Pos[k]=cnt;
for(LL i=0;i<=N;i++){
state[cnt][i]=sta[i];
if(vis[i]>1) res=res*Pow(vis[i],vis[i]-2)%Mod;
} Ans.a[1][cnt]=res;return ;
} bool ald[MAXN+1]; LL fa[MAXN+1];
LL find(LL x){
return (x==fa[x]?x:(fa[x]=find(fa[x])));
}
inline LL GS(){
for(LL i=0;i<10;i++) vis[i]=-1; LL now=0,id=0;
for(LL i=2;i<=N+1;i++){
LL x=find(i);
if(vis[x]==-1){
vis[x]=now; id=id*10+now; ++now;
} else id=id*10+vis[x];
}
return Pos[id];
}
inline void Add(LL k,LL add_state){
for(LL i=0;i<=N+1;i++) fa[i]=i;
for(LL i=1;i<=N;i++){
for(LL j=i+1;j<=N;j++){
if(state[k][i]==state[k][j]){
//cout<<"here\n";
LL xx=find(i),yy=find(j);
//cout<<xx<<" "<<yy<<endl;
if(xx!=yy) fa[yy]=xx;
}
}
}
for(LL i=1;i<=N;i++){
if(add_state&(1<<(i-1))){
LL xx=find(N+1),yy=find(i);
if(xx==yy) return ;
fa[yy]=xx;
}
} bool flag=false;
for(LL i=2;i<=N+1;i++){
if(find(1)==find(i)) {flag=true; break;}
} if(!flag) return ;
A.a[k][GS()]++;
}
inline void init_state(LL k){
for(LL i=0;i<(1<<N);i++) Add(k,i); return ;
} int main(){
//freopen(".in","r",stdin);
//freopen(".out","w",stdout);
N=read(),M=read(); Pw[0]=1;
for(LL i=1;i<=N;i++) Pw[i]=Pw[i-1]*10;
for(LL i=0;i<=43210;i++) if(!ald[i%Pw[N]]) Add(i),ald[i%Pw[N]]=true;
A.init(cnt,cnt); Ans.x=1; Ans.y=cnt;
for(LL i=1;i<=cnt;i++) init_state(i);
A=A^(M-N);
Ans=Ans*A; printf("%lld\n",Ans.a[1][1]%Mod);
return 0;
}

[BZOJ1494]生成树计数的更多相关文章

  1. bzoj1494 生成树计数 (dp+矩阵快速幂)

    题面欺诈系列... 因为一个点最多只能连到前k个点,所以只有当前的连续k个点的连通情况是对接下来的求解有用的 那么就可以计算k个点的所有连通情况,dfs以下发现k=5的时候有52种. 我们把它们用类似 ...

  2. 【BZOJ1494】【NOI2007】生成树计数(动态规划,矩阵快速幂)

    [BZOJ1494][NOI2007]生成树计数(动态规划,矩阵快速幂) 题面 Description 最近,小栋在无向连通图的生成树个数计算方面有了惊人的进展,他发现: ·n个结点的环的生成树个数为 ...

  3. BZOJ1494 [NOI2007]生成树计数

    题意 F.A.Qs Home Discuss ProblemSet Status Ranklist Contest 入门OJ ModifyUser  autoint Logout 捐赠本站 Probl ...

  4. [BZOJ1494][NOI2007]生成树计数 状压dp 并查集

    1494: [NOI2007]生成树计数 Time Limit: 5 Sec  Memory Limit: 64 MBSubmit: 793  Solved: 451[Submit][Status][ ...

  5. 【BZOJ1002】【FJOI2007】轮状病毒(生成树计数)

    1002: [FJOI2007]轮状病毒 Time Limit: 1 Sec  Memory Limit: 162 MBSubmit: 1766  Solved: 946[Submit][Status ...

  6. SPOJ 104 HIGH - Highways 生成树计数

    题目链接:https://vjudge.net/problem/SPOJ-HIGH 解法: 生成树计数 1.构造 基尔霍夫矩阵(又叫拉普拉斯矩阵) n阶矩阵 若u.v之间有边相连 C[u][v]=C[ ...

  7. Luogu P5296 [北京省选集训2019]生成树计数

    Luogu P5296 [北京省选集训2019]生成树计数 题目链接 题目大意:给定每条边的边权.一颗生成树的权值为边权和的\(k\)次方.求出所有生成树的权值和. 我们列出答案的式子: 设\(E\) ...

  8. Loj 2320.「清华集训 2017」生成树计数

    Loj 2320.「清华集训 2017」生成树计数 题目描述 在一个 \(s\) 个点的图中,存在 \(s-n\) 条边,使图中形成了 \(n\) 个连通块,第 \(i\) 个连通块中有 \(a_i\ ...

  9. 「UVA10766」Organising the Organisation(生成树计数)

    BUPT 2017 Summer Training (for 16) #6C 题意 n个点,完全图减去m条边,求生成树个数. 题解 注意可能会给重边. 然后就是生成树计数了. 代码 #include ...

随机推荐

  1. 基本控件文档-UISlider属性---iOS-Apple苹果官方文档翻译

    本系列所有开发文档翻译链接地址:iOS7开发-Apple苹果iPhone开发Xcode官方文档翻译PDF下载地址  //转载请注明出处--本文永久链接:http://www.cnblogs.com/C ...

  2. VSCode Web Developement for Javascript. Must have plugins.

    Es6 Javascript front-end web developemnt must have plugins Prettier - Code Formatter Javascript (ES6 ...

  3. OPENId是什么, OAUTH 是什么

    what is openId open id is said to be a protocol which uses url as username, so if a website supports ...

  4. POJ 3061 Subsequence ( 尺取法)

    题目链接 Description A sequence of N positive integers (10 < N < 100 000), each of them less than ...

  5. 解决ie9以下下不支持html5和媒体查询(Media Queries)

    ie9以下不支持媒体查询和html5,可以使用补丁完美兼容 1.html5shiv ie6~8不识别html5的新元素,可以通过使用html5shiv来解决 <!--[if lt IE 9]&g ...

  6. Spring Boot中配置文件application.properties使用

    一.配置文档配置项的调用 启动后在浏览器直接输入http://localhost:18080/user/test,就直接打印出配置文件中的配置内容. 二.绑定对象bean调用 有时候属性太多了,一个个 ...

  7. 动态替换Linux核心函数的原理和实现

    转载:https://www.ibm.com/developerworks/cn/linux/l-knldebug/ 动态替换Linux核心函数的原理和实现 在调试Linux核心模块时,有时需要能够实 ...

  8. centos_7.1.1503_src_6

    http://vault.centos.org/7.1.1503/os/Source/SPackages/ perl-Test-MockObject-1.20120301-3.el7.src.rpm ...

  9. Redis 3.0 编译安装

    Redis 3.0 编译安装 http://www.xuchanggang.cn/archives/991.html

  10. NFS+inotify实时同步

    Inotify简介 Inotify是一种文件系统事件通告机制,能够实时监控文件系统下文件的访问.修改.删除等各种变化情况并将其作为事件通告给用户态应用程序.Linux内核从2.6.13版本后已经集成了 ...