题目大意

  有一个 \(n\) 个点的环,你要用 \(m\) 中颜色染这 \(n\) 个点。

  要求连续 \(m\) 个点的颜色不能是 $1 \sim m $ 的排列。

  两种环相同当且仅当这两个环可以在旋转之后变得一模一样。

  求方案数对 \({10}^9+7\) 取模的结果。

  \(n\leq {10}^9,m\leq 7\)

题解

  考虑 polya 定理,记 \(f(n)\) 为 \(n\) 个点的答案,\(g(n)\) 为 \(n\) 个点不考虑旋转的答案。那么就有

\[\begin{align}
f(n)&=\frac{1}{n}\sum_{i=1}^ng(\gcd(n,i))\\
&=\frac{1}{n}\sum_{i\mid n}\varphi(\frac{n}{i})g(i)
\end{align}
\]

  \(g(i)\) 可以 DP 计算。

  记 \(h_{i,j,k}\) 为长度为 \(i\) 的链,前 \(m\) 个点的状态(颜色)为 \(j\),最后 \(m\) 个点的状态为 \(k\) 的方案数。

  还可以记录前 \(m\) 个颜色的前多少个是互不相同的,还有最后 \(m\) 个点的颜色。就前 \(j\) 个的颜色是互不相同的,第 \(j+1\) 个点颜色和前面某个点颜色相同。

  显然 \(g(i)\) 有一个长度不超过 \(m^m\) 的递推式。

  暴力打出前面的项然后 BM 求出递推式即可。

  开 O2 只用了 48s 就打出来了。

  \(m=7\) 时递推式长度为 \(409\)。

  表在这:https://github.com/ywwywwyww/THUSC2017/tree/master/yww/farben

代码

const int N=1010;
const ll p=1000000007;
ll fp(ll a,ll b)
{
ll s=1;
for(;b;b>>=1,a=a*a%p)
if(b&1)
s=s*a%p;
return s;
} ll b[10][2010]=表;
ll a[10][2010]=表; int c[N],d[N],t;
ll pw[N][N];
ll ans;
int n,m;
int s[N];
int len;
void add(int &a,ll b)
{
a=(a+b)%p;
} void mul()
{
static int f[N];
for(int i=0;i<=2*len;i++)
f[i]=0;
for(int i=0;i<=len;i++)
if(s[i])
for(int j=0;j<=len;j++)
add(f[i+j],(ll)s[i]*s[j]);
for(int i=0;i<=2*len;i++)
s[i]=f[i];
}
void module()
{
for(int i=2*len;i>=len;i--)
if(s[i])
{
for(int j=1;j<=len;j++)
add(s[i-j],(ll)s[i]*b[m][j]);
s[i]=0;
}
} void fp(int x)
{
if(!x)
{
for(int i=0;i<=2*len;i++)
s[i]=0;
s[0]=1;
return;
}
fp(x>>1);
mul();
if(x&1)
{
for(int i=2*len;i>=1;i--)
s[i]=s[i-1];
s[0]=0;
}
module();
} ll calc(int x)
{
if(x<=500)
return a[m][x];
fp(x-1);
ll res=0;
for(int i=1;i<=len;i++)
res=(res+(ll)a[m][i]*s[i-1])%p;
return res;
} void dfs(int x,int y,ll phi)
{
if(x>t)
{
ans=(ans+calc(y)*phi)%p;
return;
}
for(int i=0;i<d[x];i++)
dfs(x+1,y*pw[x][i],phi*(c[x]-1)%p*pw[x][d[x]-i-1]%p);
dfs(x+1,y*pw[x][d[x]],phi);
}
int main()
{
open("farben");
scanf("%d%d",&n,&m);
int _n=n;
for(int i=2;i*i<=_n;i++)
if(_n%i==0)
{
c[++t]=i;
while(_n%i==0)
{
d[t]++;
_n/=i;
}
}
if(_n>1)
{
c[++t]=_n;
d[t]=1;
}
for(int i=1;i<=t;i++)
{
pw[i][0]=1;
for(int j=1;j<=d[i];j++)
pw[i][j]=pw[i][j-1]*c[i]%p;
}
len=b[m][0];
dfs(1,1,1);
ans=ans*fp(n,p-2)%p;
ans=(ans%p+p)%p;
printf("%lld\n",ans);
return 0;
}

打表程序

const ll p=1000000007;
const int N=1010;
const int n=1000;
//const int m=5;
int m; ll fp(ll a,ll b)
{
ll s=1;
for(;b;b>>=1,a=a*a%p)
if(b&1)
s=s*a%p;
return s;
} const int MA=2100000; int e[MA];
int ban[MA];
vector<int> g[N];
int pw[N];
int ma; namespace gao1
{
int a[N],b[N],c[N];
void dfs(int x)
{
if(x>m)
{
for(int i=1;i<=m;i++)
b[i]=0;
int tot=0;
int s=0;
int s2=0;
int first=0;
for(int i=1;i<=m;i++)
{
if(!b[a[i]])
b[a[i]]=++tot;
c[i]=b[a[i]];
s+=c[i]*pw[i-1];
s2+=a[i]*pw[i-1];
if(c[i]<=c[i-1]&&!first)
first=i-1;
}
if(tot==m)
ban[s2]=1;
else
g[first].push_back(s);
return;
}
for(int i=1;i<=m;i++)
{
a[x]=i;
dfs(x+1);
}
}
void gao()
{
dfs(1);
}
} namespace gao2
{
int a[N],b[N],c[N];
void dfs(int x)
{
if(x>m)
{
for(int i=1;i<=m;i++)
b[i]=0;
for(int i=m;i>=1;i--)
if(!a[i]||!b[a[i]])
{
c[i]=a[i];
b[a[i]]=1;
}
else
c[i]=0;
int s=0,s1=0;
for(int i=1;i<=m;i++)
s+=c[i]*pw[i-1];
for(int i=1;i<=m;i++)
s1+=a[i]*pw[i-1];
e[s1]=s;
return;
}
for(int i=0;i<=m;i++)
{
a[x]=i;
dfs(x+1);
}
}
void gao()
{
dfs(1);
}
} int f[N];
int h[2][MA];
int d[MA]; void add(int &a,int b)
{
a=(a+b)%p;
} int append(int a,int b)
{
return a/(m+1)+b*pw[m-1];
} namespace gao3
{
void gao(int x)
{
memset(d,0,sizeof d); for(int i=0;i<=ma;i++)
{
int flag=1;
for(int y=i,j=1;j<=x;j++)
{
y=append(y,j);
if(ban[y])
{
flag=0;
break;
}
}
d[i]=flag;
} memset(h,0,sizeof h);
int cur=0;
for(auto v:g[x])
h[cur][e[v]]++;
for(int i=m;i<=n;i++)
{
fprintf(stderr,"%d %d %d\n",m,x,i);
memset(h[cur^1],0,sizeof h[cur^1]);
for(int j=0;j<=ma;j++)
if(h[cur][j]&&!ban[j])
{
add(f[i],d[j]*h[cur][j]);
for(int k=1;k<=m;k++)
add(h[cur^1][e[append(j,k)]],h[cur][j]);
}
cur^=1;
}
}
} int main(int argc,char **argv)
{
// freopen("farben2.txt","w",stdout);
sscanf(argv[1],"%d",&m);
ma=fp(m+1,m);
pw[0]=1;
for(int i=1;i<=m;i++)
pw[i]=pw[i-1]*(m+1);
gao1::gao();
gao2::gao(); for(int i=1;i<m;i++)
f[i]=fp(m,i); for(int i=1;i<m;i++)
gao3::gao(i); for(int i=1;i<=n;i++)
printf("%d\n",f[i]);
return 0;
}

【THUSC2017】【LOJ2981】如果奇迹有颜色 DP BM 打表 线性递推的更多相关文章

  1. 【模板】BM + CH(线性递推式的求解,常系数齐次线性递推)

    这里所有的内容都将有关于一个线性递推: $f_{n} = \sum\limits_{i = 1}^{k} a_{i} * f_{n - i}$,其中$f_{0}, f_{1}, ... , f_{k ...

  2. HDU 5863 cjj's string game ( 16年多校10 G 题、矩阵快速幂优化线性递推DP )

    题目链接 题意 : 有种不同的字符,每种字符有无限个,要求用这k种字符构造两个长度为n的字符串a和b,使得a串和b串的最长公共部分长度恰为m,问方案数 分析 : 直觉是DP 不过当时看到 n 很大.但 ...

  3. HDU - 6172:Array Challenge (BM线性递推)

    题意:给出,三个函数,h,b,a,然后T次询问,每次给出n,求sqrt(an); 思路:不会推,但是感觉a应该是线性的,这个时候我们就可以用BM线性递推,自己求出前几项,然后放到模板里,就可以求了. ...

  4. BM求线性递推模板(杜教版)

    BM求线性递推模板(杜教版) BM求线性递推是最近了解到的一个黑科技 如果一个数列.其能够通过线性递推而来 例如使用矩阵快速幂优化的 DP 大概都可以丢进去 则使用 BM 即可得到任意 N 项的数列元 ...

  5. [NOI2017]泳池——概率DP+线性递推

    [NOI2017]泳池 实在没有思路啊~~~ luogu题解 1.差分,转化成至多k的概率减去至多k-1的概率.这样就不用记录“有没有出现k”这个信息了 2.n是1e9,感觉要递推然后利用数列的加速技 ...

  6. LG5487 【模板】线性递推+BM算法

    [模板]线性递推+BM算法 给出一个数列 \(P\) 从 \(0\) 开始的前 \(n\) 项,求序列 \(P\) 在\(\bmod~998244353\) 下的最短线性递推式,并在 \(\bmod~ ...

  7. 2018 焦作网络赛 L Poor God Water ( AC自动机构造矩阵、BM求线性递推、手动构造矩阵、矩阵快速幂 )

    题目链接 题意 : 实际上可以转化一下题意 要求求出用三个不同元素的字符集例如 { 'A' .'B' .'C' } 构造出长度为 n 且不包含 AAA.BBB CCC.ACB BCA.CAC CBC ...

  8. 牛客多校第九场 A The power of Fibonacci 杜教bm解线性递推

    题意:计算斐波那契数列前n项和的m次方模1e9 题解: $F[i] – F[i-1] – F[i-2] = 0$ $F[i]^2 – 2 F[i-1]^2 – 2 F[i-2]^2 + F[i-3] ...

  9. 【CF607B】Zuma——区间dp(记忆化搜索/递推)

    以下是从中文翻译成人话的题面: 给定一个长度小于等于500的序列,每个数字代表一个颜色,每次可以消掉一个回文串,问最多消几次可以消完? (7.16) 这个题从洛谷pend回来以后显示有103个测试点( ...

随机推荐

  1. Java开发笔记(八十八)文件字节I/O流

    前面介绍了如何使用字符流读写文件,并指出字符流工具的处理局限,进而给出随机文件工具加以改进.随机文件工具除了支持访问文件内部的任意位置,更关键的一点是通过字节数组读写文件数据,采取字节方式比起字符方式 ...

  2. Myeclipse10.7添加本地插件方法

    -

  3. dubbo-源码分析Consumer

    counsumer使用服务的时候会在xml中配置<dubbo:reference> dubbo在spring.handles里的NamespaceHandle又有如下配置: registe ...

  4. Cannot read property 'validate' of undefined

    在使用element-UI表单验证中一直报错,'Error in event handler for “click”: “TypeError: Cannot read property ‘valida ...

  5. 查看Eclipse版本号及各个版本区别

    1. 找到eclipse安装目录 2. 进入readme文件夹,打开readme_eclipse.html 3. readme_eclipse.html呈现的第二行即数字版本号,如: Eclipse ...

  6. 【学习记录】第一章 数据库设计-《SQL Server数据库设计和开发基础篇视频课程》

    一.课程笔记 1.1  软件开发周期 (1)需求分析阶段 分析客户的业务和数据处理需求. (2)概要设计阶段 设计数据库的E-R模型图,确认需求信息的正确和完整. /* E-R图:实体-关系图(Ent ...

  7. eclipse导入maven项目,但无法编译的问题

    同事今天从git 导入项目到eclipse 后,发现项目所依赖的包找不到依赖,初步判定是maven的依赖没有导入项目中. 最终发现,在项目中的.classpath 文件加入以下代码即可解决问题. &l ...

  8. Linux命令之常用篇

    一.文件和目录 1. cd命令 它用于切换当前目录,它的参数是要切换到的目录的路径,可以是绝对路径,也可以是相对路径. 指令 说明 cd /home 进入‘home’目录 cd .. 返回上一级目录 ...

  9. R语言学习——数据合并及绘制密度分布曲线图

    setwd("E:/08_cooperation/07_X-lab/06-Crosstalk/Aadapter_primer")# 读取lane01.txt,并对其按列进行相加处理 ...

  10. Element-UI动态更换主题

    参考:vue-基于elementui换肤[自定义主题] 实践: 需求1.后期维护主题色不更换:  直接在线主题生成工具下载,在APP.VUE引入:(注意Element UI 版本1.3?2.0) 需求 ...