生成子群问题
orz 群论。
之前看到 grass8cow 博客中关于 Grupa Permutacji 这道题做法的简略记述后一直感觉这个题是究极神秘题。APIO 听了 Kubic 的讲课(当时讲的是 LOJ177 生成子群阶数)后终于算是懂了一点了。
众所周知,所有有限群同构于一个置换群的子群。当我们拿着一堆置换,然后再复合来复合去,所有可能得到的结果自然生成了一个群。grupa 这个题研究了这个群的平均逆序对个数,而 LOJ177 研究了这个群的大小。以前我一直认为这种东西很难做,出乎意料的是这两个问题都可以做到
解决这两个问题的第一个共同关键是在于生成子群的“均匀性”。我们这样描述这个性质:对于所有置换的第
比如说,对于 grupa 这个题我们研究逆序对,所以对于一个下标二元组
如何说明这种“均匀性”呢?我们考虑建图分析。对于每个生成元置换
由于置换群不断的复合自己总会走回自己,所以说我们同时可以连一条反向边,再写上
这样,你从任意元组
我们不妨取一个以
解决这两个问题第二个共同的关键在于如何削减生成元集合的大小,也就是说找到另一组生成元集合使得它们跟原生成元集合生成的群同构。如果你无聊中翻过 OI-wiki 中一些较为冷门的页面,你可能会知道 schreier-sims 算法 给出了一个这个问题的确定性做法。我没有认真看过这个算法 QwQ,只大致看过 Kubic APIO 讲义里的东西,但是更适合 OI 宝宝体质的做法是:随机选取一个生成元的子集,以任意顺序复合起来,重复足够多次之后将有大概率得到一些能够生成原来的群的置换。
所以对于 grupa 这个题我们选取
// ...
int n,k;
int a[N][N];
int p[N],q[N];
inline int id(int x,int y){return (x-1)*n+y;}
int f[N*N],inv[N*N];
int ca[N*N],cb[N*N];
int rt(int x){
if(f[x]==x) return x;
return f[x]=rt(f[x]);
}
int main(){
n=read();k=read();
for(int i=1;i<=k;++i)
for(int j=1;j<=n;++j) a[i][j]=read();
for(int i=1;i<=n*n;++i) f[i]=i;
for(int it=0;it<L;++it){
for(int t=1;t<=n;++t) p[t]=t;
for(int i=1;i<=k;++i)
if(rng()&1){
for(int t=1;t<=n;++t) q[p[t]]=a[i][t];
for(int t=1;t<=n;++t) p[t]=q[t];
}
for(int i=1;i<=n;++i)
for(int j=1;j<=n;++j){
if(i==j) continue;
f[rt(id(i,j))]=rt(id(p[i],p[j]));
}
}
for(int i=1;i<=n;++i)
for(int j=1;j<=n;++j){
if(i==j) continue;
if(i<j) ++ca[rt(id(i,j))];
else ++cb[rt(id(i,j))];
}
inv[1]=1;
for(int i=2;i<=n*n;++i) inv[i]=(ll)inv[P%i]*(P-P/i)%P;
int res=0;
for(int i=1;i<=n*n;++i)
if(f[i]==i&&ca[i]&&cb[i]) res=(res+(ll)ca[i]*cb[i]%P*inv[ca[i]+cb[i]])%P;
printf("%d\n",res);
return 0;
}
再考虑 LOJ177 这个题。我们统计有多少种置换,不妨先统计
前者方法我们已经熟知,后者可以这样考虑:之前我们建出来的每一个经过
// ... const int B >= O(n);
struct perm{
int p[N];
void init(){
for(int i=1;i<=n;++i) p[i]=i;
}
friend perm operator+(const perm a,const perm b){
perm c;
for(int i=1;i<=n;++i) c.p[i]=b.p[a.p[i]];
return c;
}
perm inv(){
perm x;
for(int i=1;i<=n;++i) x.p[p[i]]=i;
return x;
}
}a[N],b[N],path[N<<1];
vector<pii> vec[N];
ll res[10];int len;
bool vis[N];
void dfs(int u,perm cur){
path[u]=cur;
vis[u]=1;++num;
for(auto [v,w]:vec[u]){
if(vis[v]) continue;
dfs(v,cur+a[w]);
}
}
void iter(){
for(int i=1;i<=m;++i)
for(int j=1;j<=n;++j) vec[j].emplace_back(a[i].p[j],i);
perm Init;
Init.init();
dfs(n,Init);
for(int i=1;i<=n;++i) path[i+n]=path[i].inv(),vec[i].clear();
for(int it=1;it<=B;++it){
b[it].init();
for(int i=1;i<=m;++i)
for(int j=1;j<=n;++j)
if(vis[j]&&(rng()&1)) b[it]=b[it]+path[j]+a[i]+path[a[i].p[j]+n];
}
for(int i=1;i<=n;++i) vis[i]=0;
m=B;
for(int i=1;i<=m;++i) a[i]=b[i];
--n;
ll carry=0;
for(int i=0;i<=len;++i){
res[i]*=num;res[i]+=carry;
carry=res[i]/T;res[i]%=T;
}
if(carry) res[++len]=carry;
num=0;
}
int main(){
n=read();m=read();
for(int i=1;i<=m;++i)
for(int j=1;j<=n;++j) a[i].p[j]=read();
res[len=0]=1;
while(n) iter();
printf("%lld",res[len]);
for(int i=len-1;~i;--i) printf("%016lld",res[i]);
putchar('\n');
return 0;
}
upd:
拜谢环一周!在与环一周讨论后我们弄明白了这样一个问题:
对于一个群,我们求其最少生成元个数,称为这个群的秩。
一个群的秩是可以达到
所以说求群阶数时一定要保留
但 grupa 那个题只需要你把生成元连成的图连出来,这时候不需要遍历到所有生成元就可以建出这张图了,所以只需要保留