由于这是第一天去实现polya题,所以由易到难,先来个铺垫题(假设读者是看过课件的,不然可能会对有些“显然”的地方会看不懂):

一:POJ1286 Necklace of Beads :有三种颜色,问可以翻转,可以旋转的染色方案数,n<24。

1,n比较小,恶意的揣测出题人很有可能出超级多组数据,所以先打表。

2,考虑旋转:

for(i=;i<n;i++)  sum+=pow(n,gcd(n,i));  

3,考虑翻转:

if(n&)  sum+=n*pow(,n/+) ;
else {
sum+=n/*pow(,n/) ;
sum+=n/*pow(,n/+) ;
}

4,除以总置换数(2*n=n+n/2+n/2):

sum/=(*n);

5,终极代码:

#include<cmath>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<iostream>
using namespace std;
#define ll long long
ll ans[];
ll gcd(ll a,ll b) { if(b==) return a; return gcd(b,a%b); }
ll pow(ll a,ll x)
{
ll res=;
while(x){
if(x&) res*=a;
x>>=;
a*=a;
} return res;
}
int main()
{
ll n,i,sum;
for(n=;n<=;n++){
sum=;
for(i=;i<n;i++) sum+=pow(,gcd(n,i));
if(n&) sum+=n*pow(,n/+) ;
else {
sum+=n/*pow(,n/) ;
sum+=n/*pow(,n/+) ;
}
ans[n]=sum/n/;
}
while(~scanf("%lld",&n)){
if(n==-) return ;
printf("%lld\n",ans[n]);
}
return ;
}

二:HDU3923:Invoker :把3改成m即可。

1,注意有乘法的话担心超int还是用long long保险。

2,注意除置换群2*n的时候要逆元。。。比赛的时候就是没有加逆元,导致我牛客第四场排名15。。。不然连续5场前10挺好的。。。

#include<cmath>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<iostream>
using namespace std;
const int Mod=;
#define ll long long
int gcd(int a,int b) { if(b==) return a; return gcd(b,a%b); }
int qpow(int a,int x)
{
int res=;a%=Mod;
while(x){
if(x&) res=(ll)res*a%Mod;
x>>=;
a=(ll)a*a%Mod;
} return res;
}
int main()
{
int T,n,m,i,Case=,ans;
scanf("%d",&T); while(T--){
scanf("%d%d",&m,&n);
ans=;
for(i=;i<n;i++) ans=(ans+qpow(m,gcd(n,i)))%Mod;
if(n&) ans=(ans+n*qpow(m,n/+))%Mod;
else {
ans=(ans+n/*qpow(m,n/))%Mod;
ans=(ans+n/*qpow(m,n/+))%Mod;
}
ans=(ll)ans%Mod*qpow(*n,Mod-)%Mod;
printf("Case #%d: %d\n",++Case,ans);
}
return ;
}

----------------------------------------------------分界线----------------------------------------------------

下面的需要加优化了。

三:POJ2154 Color: 求可以旋转,不可以翻转的置换,n个珠子,n种颜色,答案mod K,n<1e9。

那么本题只考虑旋转,则:

for(i=;i<n;i++)  ans+=pow(n,gcd(n,i));
ans/=n;

之前莫比乌斯那里,我们常用技巧是合并。  这里考虑gcd相同的合并。(做多了还是有点灵感滴,yeah)。

得到:

解决这个公式需要:快速幂+欧拉函数+一些素数的常识。

1,快速幂不说了,注意先模运算。

2,欧拉函数,枚举L,即n的素因子,然后根据phi=L*(1-1/p1)*(1-1/p2)...得到欧拉函数。

3,素数常识,在这里指的是一个数n的素因子最多有一个大于根号n,所以一直除,把根号n前的素数都除完了,留下的一定是大于根号n的一个素数。

//可以线筛一部分欧拉函数出来,不够的再枚举素数;枚举的话不超过根号个,所以复杂度不会太高。
#include<cmath>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std;
const int maxn=;
int ans, Mod;
int pri[maxn+],cnt,vis[maxn+],phi[maxn+];
int qpow(int a,int x)
{
int res=;a%=Mod;//这里a一定要除一下,不然会超int,猪啊。
while(x){
if(x&) res=res*a%Mod;
x>>=; a=a*a%Mod;
} return res%Mod;
}
void prime()
{
phi[]=;
for(int i=;i<=maxn;i++){
if(!vis[i]) pri[++cnt]=i,phi[i]=i-;
for(int j=;j<=cnt&&pri[j]*i<=maxn;j++) {
vis[pri[j]*i]=;
if(i%pri[j]==) {
phi[i*pri[j]]=phi[i]*pri[j];
break;
}
else phi[i*pri[j]]=phi[i]*(pri[j]-);
}
}
}
int Phi(int x)
{
if(x<=maxn) return phi[x]%Mod;
int res=x;
for(int i=;pri[i]*pri[i]<=x;i++)
if(x%pri[i]==){
res=(res-res/pri[i]);
while(x%pri[i]==) x/=pri[i];
}
if(x!=) res=(res-res/x);
return res%Mod;
}
int main()
{
int n,T; prime();
scanf("%d",&T);
while(T--){
ans=; scanf("%d%d",&n,&Mod);
for(int i=;i*i<=n;i++){
if(n%i!=) continue;
ans=(ans+qpow(n,i-)*Phi(n/i))%Mod;
if(i*i!=n) ans=(ans+qpow(n,n/i-)*Phi(i))%Mod;
}
printf("%d\n",ans);
} return ;
}

--------------------【优化】-----------

--------------------【here】-------------

这些基本的都可以用欧拉函数来优化,比如上面的第二个例子。

上面的代码是500+ms。

优化后0ms。

#include<cmath>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std;
const int maxn=;
const int Mod=1e9+;
#define ll long long
ll ans;
ll pri[maxn+],cnt,vis[maxn+],phi[maxn+];
ll qpow(ll a,ll x)
{
ll res=;a%=Mod;//这里a一定要除一下,不然会超int,猪啊。
while(x){
if(x&) res=res*a%Mod;
x>>=; a=a*a%Mod;
} return res%Mod;
}
void prime()
{
phi[]=;
for(int i=;i<=maxn;i++){
if(!vis[i]) pri[++cnt]=i,phi[i]=i-;
for(int j=;j<=cnt&&pri[j]*i<=maxn;j++) {
vis[pri[j]*i]=;
if(i%pri[j]==) {
phi[i*pri[j]]=phi[i]*pri[j];
break;
}
else phi[i*pri[j]]=phi[i]*(pri[j]-);
}
}
}
int main()
{
ll n,m,T,Case=; prime();
scanf("%lld",&T);
while(T--){
ans=; scanf("%lld%lld",&m,&n);
for(ll i=;i*i<=n;i++){
if(n%i!=) continue;
ans=(ans+qpow(m,i)*phi[n/i])%Mod;
if(i*i!=n) ans=(ans+qpow(m,n/i)*phi[i])%Mod;
}
if(n&) ans=(ans+n*qpow(m,n/+))%Mod;
else {
ans=(ans+n/*qpow(m,n/))%Mod;
ans=(ans+n/*qpow(m,n/+))%Mod;
}
ans=ans%Mod*qpow(*n,Mod-)%Mod;
printf("Case #%lld: %lld\n",++Case,ans);
} return ;
}

其他类似的题目还有,HDOJ:2084、2647、1812、3411、2865、2481。POJ:1286、2409、2154、2888。

POJ2154 Color【 polya定理+欧拉函数优化】(三个例题)的更多相关文章

  1. 【poj2154】Color Polya定理+欧拉函数

    题目描述 $T$ 组询问,用 $n$ 种颜色去染 $n$ 个点的环,旋转后相同视为同构.求不同构的环的个数模 $p$ 的结果. $T\le 3500,n\le 10^9,p\le 30000$ . 题 ...

  2. poj2154Color polya定理+欧拉函数优化

    没想到贱贱的数据居然是错的..搞得我调了一中午+晚上一小时(哦不d飞LJH掉RP毕竟他是BUFF)结果重判就对了五次.. 回归正题,这题傻子都看得出是polya定理(如果你不是傻子就看这里),还没有翻 ...

  3. poj2409 & 2154 polya计数+欧拉函数优化

    这两个题都是项链珠子的染色问题 也是polya定理的最基本和最经典的应用之一 题目大意: 用m种颜色染n个珠子构成的项链,问最终形成的等价类有多少种 项链是一个环.通过旋转或者镜像对称都可以得到置换 ...

  4. poj 2154 Color(polya计数 + 欧拉函数优化)

    http://poj.org/problem?id=2154 大致题意:由n个珠子,n种颜色,组成一个项链.要求不同的项链数目.旋转后一样的属于同一种.结果模p. n个珠子应该有n种旋转置换.每种置换 ...

  5. POJ2154 Color 【Polya定理 + 欧拉函数】

    题目 Beads of N colors are connected together into a circular necklace of N beads (N<=1000000000). ...

  6. poj2154(polya定理+欧拉函数)

    题目链接:http://poj.org/problem?id=2154 题意:n 种颜色的珠子构成一个长为 n 的环,每种颜色珠子个数无限,也不一定要用上所有颜色,旋转可以得到状态只算一种,问有多少种 ...

  7. poj 2154 Color【polya定理+欧拉函数】

    根据polya定理,答案应该是 \[ \frac{1}{n}\sum_{i=1}^{n}n^{gcd(i,n)} \] 但是这个显然不能直接求,因为n是1e9级别的,所以推一波式子: \[ \frac ...

  8. 【POJ2154】Color Pólya定理+欧拉函数

    [POJ2154]Color 题意:求用$n$种颜色染$n$个珠子的项链的方案数.在旋转后相同的方案算作一种.答案对$P$取模. 询问次数$\le 3500$,$n\le 10^9,P\le 3000 ...

  9. Luogu4980 【模板】Polya定理(Polya定理+欧拉函数)

    对于置换0→i,1→i+1……,其中包含0的循环的元素个数显然是n/gcd(i,n),由对称性,循环节个数即为gcd(i,n). 那么要求的即为Σngcd(i,n)/n(i=0~n-1,也即1~n). ...

随机推荐

  1. 记录一下:关于mysql数据误删除恢复的问题

    大概看来几篇博客: 1.delete的可以通过回滚(rollback)的方式恢复;但是前提是,你的数据表引擎是InnoDB而不是MyISAM,而且操作不是自动提交的 但是这种方式不可以恢复trunca ...

  2. Android自己定义ViewGroup打造各种风格的SlidingMenu

    看鸿洋大大的QQ5.0側滑菜单的视频课程,对于側滑的时的动画效果的实现有了新的认识,似乎打通了任督二脉.眼下能够实现随意效果的側滑菜单了.感谢鸿洋大大!! 鸿洋大大用的是HorizontalScrol ...

  3. 使用cacheBuilder时捕获内部指定异常

    由于cacheBuilder是另起线程获取,对call方法中的抛出的异常进行了封装.所以我们在最外层捕获时是无法直接指定异常类型捕获的, 获取异常的原因判断实例类型 public static voi ...

  4. 设置安卓开机动画、开机logo

    我们要修改的是system>media文件夹下的bootanimation.zip(手机开机动画)这个文件 先来讲讲这个文件结构:该zip解压后得到两个文件, 第一个目录存放了开机时播放的图片( ...

  5. DataTable和DataRow利用反射直接转换为Model对象的扩展方法类

    DataTable和DataRow利用反射直接转换为Model对象的扩展方法类   /// <summary> /// 类 说 明:给DataTable和DataRow扩展方法,直接转换为 ...

  6. Windows socket I/O模型 之 select(2)

    在Windows socket I/O模型 之  select(1)中.我们仅仅是在console中简单的模拟了select的处理方法. 还有非常多特性不能改动.比方仅仅能写,不能读. 没使用线程.也 ...

  7. Linux的基本使用

    检测某个地址是否可以通信:ping xx.xx.xx.xx 检测某个端口是否开启:telnet xx.xx.xx.xx port 端口:用来区别不同服务 常用命令: 创建一个目录 /data mkdi ...

  8. 用算法求N(N&gt;=3)之内素数的个数

    首先.我们谈一下素数的定义.什么是素数?除了1和它本身外,不能被其它自然数整除(除0以外)的数 称之为素数(质数):否则称为合数. 依据素数的定义,在解决问题上,一開始我想到的方法是从3到N之间每一个 ...

  9. 【重磅干货】看了此文,Oracle SQL优化文章不必再看!

    目录 SQL优化的本质 SQL优化Road Map 2.1 制定SQL优化目标 2.2 检查执行计划 2.3 检查统计信息 2.4 检查高效访问结构 2.5 检查影响优化器的参数 2.6 SQL语句编 ...

  10. Android UI经常使用实例 怎样实现欢迎界面(Splash Screen)

    在Android平台下.下载一个应用后,首次打开映入眼帘的便是Splash Screen,暂且不说Android的设计原则提不提倡这样的Splash Screen.先来看看一般使用Splash Scr ...