题意:长度为 nnn 的环染 mmm 种颜色,要求任意相邻 mmm 个元素不能包含全部的颜色。求方案数 模 109+710^9+7109+7,循环同构。

n≤109,m≤7n\leq 10^9,m\leq7n≤109,m≤7

为啥我现在天天都在打表啊

先上 polya,对于移动 iii 位的置换不动点个数为 gcd⁡(i,n)\gcd(i,n)gcd(i,n)。而因为这个也是循环的,所以可以把每相邻的 n/gcd⁡(i,n)n/\gcd(i,n)n/gcd(i,n) 个看成一个环。设 f(n)f(n)f(n) 表示循环不同构的方案,显然答案就是

∑d∣nf(d)φ(nd)\sum_{d\mid n}f(d)\varphi(\frac nd)d∣n∑​f(d)φ(dn​)

这样就把循环同构搞掉了。

然后有一个暴力的dp:f(i,S)f(i,S)f(i,S) 表示填了 iii 个数,最后 mmm 个的颜色状态是 SSS,枚举开头 mmm 个转移到最后再看合不合法。

发现这个 dp 是个矩阵的形式。有个很强的结论:存在 nnn 阶矩阵递推等价于存在 nnn 阶线性递推。这个矩阵递推可以有很多含义,包括任意元素、所有元素的和,甚至矩阵本身。

也就是说,尽管这个限制很多,我们也有(xia)理(j)由(b)相(cai)信(xiang)这个 fff 一定存在不超过 mmm^mmm 次的线性递推式。然后你就可以打表BM线性递推一条龙服务了。

但是这个暴力 dp 实在太慢了,你光枚举开始状态都有 mmm^mmm ,后面再怎么都至少有个 mmm^mmm,写得不好可能还要多个 mmm^mmm,可能不能在可预见的时间内跑出来。考虑做点优化。

上面已经看出来了,复杂度瓶颈其实在枚举开头,考虑这部分怎么优化。注意到如果开头枚举的 mmm 个数中有相同的,那么结尾的时候检查就不用到这里了,也就是开头剩下的怎么填都不影响合法性。而之前的还可以重标号搞掉。

具体而言,我们枚举一个 www 表示开头有多少个连续的不同的数,即对于 i∈[1,w],ai=ii\in[1,w],a_i=ii∈[1,w],ai​=i。然后后面一个数钦定与前面的相同,即 ai+1∈[1,w]a_{i+1}\in [1,w]ai+1​∈[1,w]。之后就可以正常转移了。最后再乘上个 mw‾m^{\underline w}mw​ 加起来就是答案。

这样复杂度到了O(N⋅mm⋅poly⁡(m))O(N\cdot m^m\cdot \operatorname{poly}(m))O(N⋅mm⋅poly(m)),其中 NNN 为打前多少个数的表,考场上根据你的梦想和人品来决定,这里打了 100010001000 个。因为懒得进一步优化,就直接跑 55min⁡55\min55min 跑出来了,但我看其他人只跑了几十秒……

然后就是套板子的事情了。

暴力求 fff:

#include <iostream>
#include <cstdio>
#include <cstring>
#include <cctype>
#define MAXN 1000005
using namespace std;
const int MOD=1e9+7,N=1000;
inline int add(const int& x,const int& y){return x+y>=MOD? x+y-MOD:x+y;}
typedef long long ll;
inline int qpow(int a,int p)
{int ans=1;while (p){if (p&1) ans=(ll)ans*a%MOD;a=(ll)a*a%MOD,p>>=1;}return ans;
}
int m,a[10],MAX,vis[MAXN];
inline bool check(int x)
{if (~vis[x]) return vis[x];for (int i=0;i<m;i++) a[i]=0;int t=x;for (int i=0;i<m;i++){a[x%m]=1;x/=m;}for (int i=0;i<m;i++) if (!a[i]) return (vis[t]=1);return (vis[t]=0);
}
int cur[MAXN],t[MAXN],ans[MAXN];
inline void solve()
{for (int w=1;w<m;w++){int st=0;for (int i=1;i<=w;i++) st=st*m+i;for (int k=1;k<=w;k++){for (int i=0;i<MAX;i++) cur[i]=0;cur[st*m+k]=1;int tmp=0;for (int tt=st*m+k,j=1;j<=w+1;tt=(tt*m+(j>w? k:j))%MAX,j++)if (!check(tt))goto end1;tmp=add(tmp,1);end1:;for (int i=m;i>=m-w+1;i--) tmp=(ll)tmp*i%MOD;ans[w+1]=add(ans[w+1],tmp);for (int T=1;T<=N-w-1;T++){for (int i=0;i<MAX;i++) t[i]=0;for (int i=0;i<MAX;i++)if (check(i))for (int v=0;v<m;v++){int j=(i*m+v)%MAX;if (check(j)) t[j]=add(t[j],cur[i]);}for (int i=0;i<MAX;i++) cur[i]=t[i];int tmp=0;for (int i=0;i<MAX;i++){for (int t=i,j=1;j<=w+1;t=(t*m+(j>w? k:j))%MAX,j++)if (!check(t))goto end;tmp=add(tmp,cur[i]);end:;}for (int i=m;i>=m-w+1;i--) tmp=(ll)tmp*i%MOD;ans[T+w+1]=add(ans[T+w+1],tmp);
//              if (T+w+1==m+1) ans[m]=add(ans[m],tmp);}}}
}
int main()
{memset(vis,-1,sizeof(vis));cin>>m;MAX=qpow(m,m);printf("%d\n",N);solve();for (int i=1;i<m;i++) ans[i]=qpow(m,i);for (int i=1;i<=N;i++) printf("%d ",ans[i]); return 0;
}

找递推式:

#include <iostream>
#include <cstdio>
#include <cstring>
#include <cctype>
#define MAXN 2005
using namespace std;
inline int read()
{int ans=0;char c=getchar();while (!isdigit(c)) c=getchar();while (isdigit(c)) ans=(ans<<3)+(ans<<1)+(c^48),c=getchar();return ans;
}
const int MOD=1e9+7;
typedef long long ll;
inline int add(const int& x,const int& y){return x+y>=MOD? x+y-MOD:x+y;}
inline int dec(const int& x,const int& y){return x<y? x-y+MOD:x-y;}
inline int qpow(int a,int p)
{int ans=1;while (p){if (p&1) ans=(ll)ans*a%MOD;a=(ll)a*a%MOD,p>>=1;}return ans;
}
int n,m,F[MAXN],R[MAXN],las[MAXN],p,delta,tmp[MAXN];
int main()
{n=read();for (int i=1;i<=n;i++) F[i]=read();for (int k=1;k<=n;k++){int res=0;for (int i=1;i<=R[0];i++) res=(res+(ll)F[k-i]*R[i])%MOD;if (res==F[k]) continue;if (!R[0]){R[0]=p=k,delta=F[k];continue;}memcpy(tmp,R,sizeof(tmp));int x=(ll)dec(F[k],res)*qpow(delta,MOD-2)%MOD;R[k-p]=add(R[k-p],x);for (int i=1;i<=las[0];i++) R[k-p+i]=dec(R[k-p+i],(ll)las[i]*x%MOD);R[0]=max(R[0],k+las[0]-p);memcpy(las,tmp,sizeof(las)),delta=dec(F[k],res),p=k;}printf("%d\n",R[0]);for (int i=1;i<=R[0];i++) printf("%d%c",F[i],",\n"[i==R[0]]);for (int i=1;i<=R[0];i++) printf("%d%c",R[i],",\n"[i==R[0]]);return 0;
}

交上去的程序:

#include <iostream>
#include <cstdio>
#include <cstring>
#include <cctype>
using namespace std;
const int len[]={0,0,1,7,18,47,134,413};
const int S[][500]=表;
const int R[][500]=表;
const int MOD=1e9+7;
typedef long long ll;
inline int add(const int& x,const int& y){return x+y>=MOD? x+y-MOD:x+y;}
inline int dec(const int& x,const int& y){return x<y? x-y+MOD:x-y;}
inline int qpow(int a,int p)
{int ans=1;while (p){if (p&1) ans=(ll)ans*a%MOD;a=(ll)a*a%MOD;p>>=1;}return ans;
}
int m,F[1000],ans[1000],cur[1000],t[1000];
inline void mul(int* F,int* G)
{for (int i=0;i<=(len[m]<<1);i++) t[i]=0;for (int i=0;i<=len[m];i++)for (int j=0;j<=len[m];j++)t[i+j]=(t[i+j]+(ll)F[i]*G[j])%MOD;for (int i=0;i<=(len[m]<<1);i++) F[i]=t[i];for (int i=(len[m]<<1);i>len[m];i--)if (F[i]){for (int j=1;j<=len[m];j++) F[i-j]=(F[i-j]+(ll)F[i]*R[m][j])%MOD;F[i]=0; }
}
inline void qpow(int* F,int p)
{ans[0]=1;for (int i=1;i<=len[m];i++) ans[i]=0;while (p){if (p&1) mul(ans,F);mul(F,F),p>>=1;}for (int i=0;i<=len[m];i++) F[i]=ans[i];
}
inline int calc(int n)
{for (int i=0;i<=len[m];i++) cur[i]=0;cur[1]=1;qpow(cur,n);int res=0;for (int i=1;i<=len[m];i++) res=(res+(ll)S[m][i]*ans[i])%MOD;return res;
}
inline int phi(int x)
{int ans=1;for (int i=2;i*i<=x;i++)if (x%i==0){ans*=i-1,x/=i;while (x%i==0) ans*=i,x/=i;}if (x>1) ans*=x-1;return ans;
}
int main()
{int n;cin>>n>>m;int ans=0;for (int i=1;i*i<=n;i++)if (n%i==0){ans=(ans+(ll)calc(i)*phi(n/i))%MOD;if (i*i<n) ans=(ans+(ll)calc(n/i)*phi(i))%MOD;}cout<<(ll)ans*qpow(n,MOD-2)%MOD;return 0;
}

【THUSC 2017】如果奇迹有颜色【polya引理】【矩阵】【计数dp】【BM打表+线性递推】相关推荐

  1. 【hdu2481】Toy,burnside引理+矩阵乘法

    传送门 思路: 快把我做哭了TAT 从昨天上午开始想,搞了一下午有一个点没有想明白 TA爷看过题后想了5min貌似就爆正解了TAT 下面我就来讲一讲~ 一开始先想没有置换情况下的方案数 手玩无果后打了 ...

  2. HTML前端页面颜色的四种方法,色号表

    HTML前端页面颜色的四种方法,色号表 颜色的三种表示方式: (1)单词:red green black-用法: <font color="pink" size=" ...

  3. THUSC 2017 游记

    Day0 早上在家里整理东西. 下午坐飞机去北京.(怎么又去北京,上周刚去的北京) 一开始飞机爬升的时候太无聊就睡着了.醒了以后就开始吃东西.吐槽一句:厦航的飞机就是好啊.上面的点心也比上次海航的好吃 ...

  4. 洛谷P4727:图的同构计数(Polya引理)(dfs)

    解析 <关于我想了半天 dp 结果看题解 dfs 就行这回事> 我就说 gcd⁡\gcdgcd 这玩意 dp 个锤子啊- 拆分数的增长速度远没有想像中那么大,事实上,n=60n=60n=6 ...

  5. 2017.9.18 数颜色 思考记录

    这个题暴力是不是可以过啊,感觉卡卡常数问题不大. 做这个题需要主席树套树状数组或带改莫队,,处于经济考虑,于是学了带改莫队 其实差不多,主体和普通莫队基本一样,就是多了对修改的暴力处理.. 把修改次数 ...

  6. hdu 6035:Colorful Tree (2017 多校第一场 1003) 【树形dp】

    题目链接 单独考虑每一种颜色,答案就是对于每种颜色至少经过一次这种的路径条数之和.反过来思考只需要求有多少条路径没有经过这种颜色即可. 具体实现过程比较复杂,很神奇的一个树形dp,下面给出一个含较详细 ...

  7. 【SDOI2013】项链【莫比乌斯反演】【Polya定理】【递推式求通项】【数论】

    题意:TTT 组数据,每组给定 n,an,an,a,求满足下列条件的项链数量: 有 nnn 个珠子. 每个珠子上有三个 [1,a]∩Z[1,a]\cap \Z[1,a]∩Z 的数,且三个数 gcd⁡\ ...

  8. RBG三颜色填格子(非DP实现)

    **RGB三颜色的一种简单实现,,,,其实要把三种颜色填在5个格子里面或者10个格子里面,这首先明确就不是一种组合,而是一种排列,那么排列的话,更加确定的是,不止是顺序的排列,也可以重复出现数字,所以 ...

  9. MATLAB画图——用数字表示颜色以及删除矩阵行

    本篇文章用来记录本人在书写电离层程序时候的想法和设计. 在设计画图函数时,如果曲线的颜色类别较多,又想显示不同类别的数据表示,可以尝试直接用数字代表颜色,省略指定曲线颜色的步骤,这里以GPS数据为例, ...

最新文章

  1. 如何寻找属于自己的高富帅和大公司!
  2. 如何成为一名软件架构师?
  3. 32位数型计算机什么意思,展示32位是什么意思
  4. python中 str.strip()用法
  5. 底部居中_中文编程:安卓的底部菜单设计
  6. python写文字方法_Transcrypt: 用Python写js的方法
  7. 排名前15位的Kubernetes监控和安全工具
  8. P3275-[SCOI2011]糖果【差分约束,负环】
  9. matlab中的rng函数
  10. 手机越“清理”越卡顿,清理软件“坑”过你吗?
  11. 远程访问dmz和虚拟服务器的设置
  12. C#winform两个或多个panel重叠布局时如何显示一个
  13. Linux C语言中gotoxy函数
  14. 易点易动助力企业年中固定资产盘点
  15. css vue 内联_Vue绑定内联样式问题
  16. Mysql如何按照指定间隔时间查询数据
  17. 加息对银行股影响|加息是对银行股的利好
  18. python数据分析002—python基础语法
  19. CDbCriteria CArrayDataProvider zii.widgets.grid (2)
  20. Xilinx原语的用法

热门文章

  1. 量子力学到底神奇在哪里?看完这个,我的认知彻底坍塌了
  2. 相亲对象能有多油腻......
  3. “Python简直万能!”拜托快醒醒!
  4. 如何解读决策树和随机森林的内部工作机制?
  5. 学习总结之数据挖掘三大类六分项
  6. linux+4.4+android,Ubuntu 14.04 x64配置Android 4.4 kitkat编译环境的方法
  7. oracle unpivot 空值,sql – 处理UNPIVOT中的NULL值
  8. 联想电脑如何添加无线网络连接服务器,安装英特尔MYWIFI的操作步骤
  9. mysql connector c编程_MySQL数据库之MySQL Connector 编程
  10. php如何将页面嵌入在另一盒子里,如何将一个盒子在显示在浏览器的正中间_html/css_WEB-ITnose...