题链:

http://acm.hdu.edu.cn/showproblem.php?pid=6021

题解:

题意:
对于一个长度为 N的由小写英文字母构成的随机字符串,
当它进行一次变换,所有字符 i 都会变成a[i]。
同时变换数组:a[i]是26个字母组成的排列。
现在需要知道这个随机串变换到自身的期望变换次数。
请你输出期望答案乘上26^N以后模 1000000007的结果。


容斥,LCM
其实题目要求的就是每种串(共有 26^N种串)回到自身的变化次数之和。

暴力求法就是:
看每种串的每个字符在循环长度为多少的循环里(设第i个位置的字符的循环长度为 Di),
则该串的变化次数为 LCM(D1,D2,D3...,DN)(最小公倍数)

不难发现,对于给定的变化数组 a[ ],循环长度不同的循环节的种类个数不超过 6 个(1+2+3+4+5+6+7>26)
6很小,所以就可以在这个 "6" 上面搞事情。

枚举循环节的集合 S,表示串里的字符只能是这些循环节的字母集合里的字母
就是把 N 个字符放到那些循环节的字母集合中去。
(假设这个集合的循环节只包含了 W 个字母)
但是我们要使得串的变化次数为 LCM(Di, ${i}\epsilon{S}$ ),(即答案为这些循环节长度的 LCM)
那么每个循环节里面都至少要有一个字符被在里面。
所以问题转化为:N个东西分到 M 个盒子,每个盒子都至少有一个东西。
求法如下:
设 f[S] 表示在 S集合的循环节里随便放的方案数,即每个字符可以随便选择 W 个字母里面的任意一个。
显然 f[S] = $W^N$。但是这个 f[S] 有非法方案,即存在某些循环节里没有放字符。
所以容斥如下:
ANS = 没有涵盖至少 0个循环节的方案数(即f[S])
          -没有涵盖至少 1个循环节的方案数
          +没有涵盖至少 2个循环节的方案数
          -...+... 
用于容斥的方案数就直接枚举 S 的子集 _S,(方案数即为 f[_S])。

代码:

#include<cstdio>
#include<cstring>
#include<iostream>
#define _ % mod
#define filein(x) freopen(#x".in","r",stdin);
#define fileout(x) freopen(#x".out","w",stdout);
using namespace std;
const int mod=1000000007;
int f[100],vis[100],d[100],g[100],h[100],tot;;
char S[100];
int T,N,ANS;
int gcd(int a,int b){
while(b^=a^=b^=a%=b);
return a;
}
int pow(int a,int b){
int now=1;
while(b){
if(b&1) now=(1ll*now*a)_;
a=(1ll*a*a)_; b>>=1;
}
return now;
}
void precircle(){
tot=0;
scanf("%d",&N);
scanf("%s",S+1);
memset(d,0,sizeof(d));
memset(h,0,sizeof(h));
memset(vis,0,sizeof(vis));
for(int i=1;i<=26;i++)
if(!vis[S[i]-'a'+1]){
int tmp=0,p=S[i]-'a'+1;
while(!vis[p]) tmp++,vis[p]=1,p=S[p]-'a'+1;
d[tmp]++;
}
for(int i=1;i<=26;i++)
if(d[i]) tot++,g[tot]=d[i],d[tot]=i;
for(int s=1,cnt;cnt=0,s<(1<<tot);s++){
for(int i=1;i<=tot;i++)
if(s&(1<<(i-1)))
cnt+=g[i]*d[i],h[s]++;
f[s]=pow(cnt,N);
}
/*for(int s=1,tmp;s<(1<<tot);s++){
for(int _s=s;_s;_s=(_s-1)&s){
if(s==_s) continue;
tmp=-f[_s];
tmp=(1ll*tmp+mod)_;
f[s]=(1ll*f[s]+tmp+mod)_;
}
}正推就不用容斥了*/
for(int s=1,tmp;s<(1<<tot);s++){
for(int _s=s;_s;_s=(_s-1)&s){
if(s==_s) continue;
tmp=-f[_s];
tmp=(1ll*tmp+mod)_;
f[s]=(1ll*f[s]+tmp+mod)_;
}
}//逆推需要容斥
}
void dfs(int p,int s,int lcm){
if(p==tot+1){
ANS=(1ll*ANS+(1ll*f[s]*lcm)_)_;
return;
}
dfs(p+1,s,lcm);
dfs(p+1,s|(1<<(p-1)),1ll*lcm/gcd(lcm,d[p])*d[p]);
}
int main()
{
scanf("%d",&T);
while(T--){
ANS=0;
precircle();
dfs(1,0,1);
printf("%d\n",ANS);
}
return 0;
}

●HDU 6021 MG loves string的更多相关文章

  1. hdu 6021 MG loves string

    MG loves string Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 262144/262144 K (Java/Others ...

  2. hdu 6021 MG loves string (一道容斥原理神题)(转)

    MG loves string    Accepts: 30    Submissions: 67  Time Limit: 2000/1000 MS (Java/Others)    Memory ...

  3. 【HDU 6021】 MG loves string (枚举+容斥原理)

    MG loves string  Accepts: 30  Submissions: 67  Time Limit: 2000/1000 MS (Java/Others)  Memory Limit: ...

  4. MG loves string

    MG loves string Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 262144/262144 K (Java/Others ...

  5. hdu 6020 MG loves apple 恶心模拟

    题目链接:点击传送 MG loves apple Time Limit: 3000/1500 MS (Java/Others)    Memory Limit: 262144/262144 K (Ja ...

  6. hdu6021[BestCoder #93] MG loves string

    这场BC实在是有趣啊,T2是个没有什么算法但是细节坑的贪心+分类讨论乱搞,T3反而码起来很顺. 然后出现了T2过的人没有T3多的现象(T2:20人,T3:30人),而且T2的AC率是惨烈的不到3% ( ...

  7. 【HDU 6020】 MG loves apple (乱搞?)

    MG loves apple  Accepts: 20  Submissions: 693  Time Limit: 3000/1500 MS (Java/Others)  Memory Limit: ...

  8. best corder MG loves gold

    MG loves gold  Accepts: 451  Submissions: 1382  Time Limit: 3000/1500 MS (Java/Others)  Memory Limit ...

  9. Hdu 5806 NanoApe Loves Sequence Ⅱ(双指针) (C++,Java)

    Hdu 5806 NanoApe Loves Sequence Ⅱ(双指针) Hdu 5806 题意:给出一个数组,求区间第k大的数大于等于m的区间个数 #include<queue> # ...

随机推荐

  1. 个人作业2:QQ音乐APP案例分析

    APP案例分析 QQ音乐 选择理由:毕竟作为QQ音乐九年的资深老用户以及音乐爱好者 第一部分 调研 1.第一次上手的体验    我算是很早期的QQ音乐的用户,用QQ音乐七八年,除了体验各方面还不错之外 ...

  2. django搭建web (三) admin.py -- 待续

    demo 关于模型myQuestion,myAnswer将在后述博客提及 # -*- coding: utf-8 -*- from __future__ import unicode_literals ...

  3. bzoj千题计划220:bzoj3938: Robot

    http://www.lydsy.com/JudgeOnline/problem.php?id=3938 以时间为x轴,以距离为y轴,那么每个机器人的行走路径就是一条折线 把折线分段加入线段树里,然后 ...

  4. python 面向对象之继承与派生

    一:初识继承 1,什么是继承? 继承指的是类与类之间的关系,是一种什么"是"什么的关系,继承的功能之一就是用来解决代码重用问题 继承是一种创建新类的方式,在python中,新建的类 ...

  5. day-3 python多线程编程知识点汇总

    python语言以容易入门,适合应用开发,编程简洁,第三方库多等等诸多优点,并吸引广大编程爱好者.但是也存在一个被熟知的性能瓶颈:python解释器引入GIL锁以后,多CPU场景下,也不再是并行方式运 ...

  6. JAVA_SE基础——35.static修饰成员函数

    在Java中适用static关键字修饰的方法称为静态方法. 声明静态方法的语法格式如下: 权限修饰符 static 数据类型 方法名(){ 方法体 } 静态方法 可以使用类名直接调用     类名.方 ...

  7. php的数组的函数

    1.可以将一个二位数组转化成两个一维数组,没有指定键就是默认的索引 注意二位数组有几种类型,其中最常见的一种是外层循环是一个索引数组,然后内层是一个关联数组.这种通过便利第一层,然后第二层指定关联词就 ...

  8. FTP传输文件被破坏的问题(Linux、Busybox)

    在网络设备上抓包后,通过FTP传输到本机,发现抓包文件破坏.更换tftp后文件正常,定位问题在FTP上. FTP的传输模式有两种:①ASCII  ②二进制 ①ASCII: 以ASCII编码的方式传输文 ...

  9. XPath编写规则学习

    XPath编写规则学习   辅助工具:firefox安装findbugs,view Xpath firefox :Xpath验证方式:$x("xpath"); 粘贴xpath语句回 ...

  10. 新概念英语(1-41)Penny's bag

    新概念英语(1-41)Penny's bag Who is the tin of tobacco for? A:Is that bag heavy, Penny? B:Not very. A:Here ...