[BZOJ4671]异或图 - xjr01 - 博客园

考虑先算一些限制少的情况

gi表示把n个点的图,划分成i个连通块的方案数

连通块之间不连通很好处理(怎么处理看下边),但是内部必须连通,就很难办了

所以再降低条件,fi表示,把n个点的图,划分成i个"连通块",保证连通块之间不会有边相连,但是内部可以不连通的方案数

fi计算方法如下:

用Bell(n)的复杂度枚举集合划分,然后相邻集合之间不能连边,

然后考虑凑出符合这个集合划分的图有多少个,异或高斯消元,xi表示第i个图选择与否,如果必须不选,等号右边就是0,否则不管。

求自由元个数

fi和gi的关系:

就是枚举到底是有几个连通块

然后斯特林反演:

虽然往上枚举,但是式子证明思路是一样的,(-1)项的指数只要保证在所谓[n=m]时候是偶数就好了

高斯消元也可以换成线性基

每个图一个元素。每个线性基的位表示这个图这个边有没有,并且再和这次必要的边取&

要求多少个子集xor为全0

线性基之后,2^(s-sz)即可。

代码:

#include<bits/stdc++.h>
#define reg register int
#define il inline
#define numb (ch^'0')
#define int long long
using namespace std;
typedef long long ll;
il void rd(int &x){
char ch;x=;bool fl=false;
while(!isdigit(ch=getchar()))(ch=='-')&&(fl=true);
for(x=numb;isdigit(ch=getchar());x=x*+numb);
(fl==true)&&(x=-x);
}
namespace Miracle{
const int M=;
const int N=;
int n,m;
int edge[M][][];
ll f[*];
ll t[N];
int ans[M];
char s[N*N];
ll id[N];
int vis[*];
ll Guass(int n){
memset(ans,-,sizeof ans);
memset(vis,,sizeof vis);
// for(reg i=1;i<=n;++i){
// for(reg j=1;j<=m;++j){
// cout<<f[i][j]<<" ";
// }cout<<" = "<<f[i][m+1]<<endl;
// }
// cout<<endl;
int free=;
for(reg i=;i<=m;++i){
int id=;
for(reg j=;j<=n;++j){
if((!vis[j])&&(f[j]&(1LL*<<(i-)))) id=j;
}
if(!id){
++free;continue;
}
vis[i]=;
if(id!=i) swap(f[i],f[id]);
for(reg j=;j<=n;++j){
if(j==i) continue;
if(!vis[j]&&(f[j]&(1LL*<<(i-)))){
f[j]^=f[i];
}
}
}
return (1LL*<<free);
}
void dfs(int x,int sz){
if(x==n+){
//cout<<" x "<<x<<" sz "<<sz<<endl;
memset(f,,sizeof f);
int cnt=;
for(reg i=;i<=n;++i){
for(reg j=i+;j<=n;++j){
if(id[i]!=id[j]){
++cnt;
for(reg k=;k<=m;++k){
if(edge[k][i][j]) f[cnt]|=(1LL*<<(k-));
}
}
}
}
// cout<<" cnt "<<cnt<<endl;
t[sz]+=Guass(max(cnt,m));
return;
}
for(reg i=;i<=sz;++i){
id[x]=i;
dfs(x+,sz);
id[x]=;
}
id[x]=sz+;
dfs(x+,sz+);
id[x]=;
}
void calc(int l){
for(n=;n<=;++n){
if(n*(n-)/==l) break;
}
}
int main(){
rd(m);
for(reg i=;i<=m;++i){
scanf("%s",s+);
int l=strlen(s+);
if(!n) calc(l);
int t=;
for(reg j=;j<=n;++j){
for(reg k=j+;k<=n;++k){
++t;
edge[i][j][k]=edge[i][k][j]=s[t]-'';
}
}
}
dfs(,);
ll ans=,jie=;
for(reg i=;i<=n;++i){
if(i&){
ans+=jie*t[i];
}else{
ans-=jie*t[i];
}
jie*=i;
}
printf("%lld",ans);
return ;
} }
signed main(){
Miracle::main();
return ;
} /*
Author: *Miracle*
Date: 2019/2/16 21:40:44
*/

总结:

找到两个数组f,g

f范围宽松好统计,g范围严格难统计但是和答案有直接关系,

这样,只要得到f和g的关系,就可以找到答案!

异或下线性方程组的自由元个数:

先变成n*(n+1)的矩阵

然后高斯消元,如果某一个id找不到,那么一定是自由元了,计数器++

注意,每次找i和消除必须在全局位置,并且用一个vis标记表示是否还能动

最后削成的上三角矩阵,除了无解情况,剩下的一定有唯一解

最新文章

  1. 百度app测试服务
  2. adb无法使用,提示error: unknown host service的解决办法
  3. require和include的区别
  4. SVN版本回退
  5. Bootstrap系列 -- 37. 基础导航样式
  6. Windows 下的.NET+ Memcached安装
  7. JS框架比较
  8. SqlServer数据文件增长也很快,到底是哪些表增长造成的呢?
  9. 取得inputStream的长度
  10. 将Sublime Text3添加到右键菜单中
  11. 注册表与盘符(转victor888文章 )
  12. 移动端图片放大滑动查看-插件photoswipe的使用
  13. PostgreSQL9.6.2的WINDOWS下安装
  14. Vue全家桶(Vue-cli、Vue-route、vuex)
  15. 命令行BASH的基本操作
  16. JavaMail在Windows平台下正常发送邮件,部署到Linux后则发送失败
  17. windows与linux换行规则
  18. ElasicSearch(1)
  19. 小姐姐手把手教你JS数组中的对象去重
  20. GUI_事件监听机制与ActionListener演示

热门文章

  1. 扩展1000!(n!)的尾数零的个数
  2. C# 不用递归,获取无限层级数据
  3. leaflet动态路径
  4. phpstorm ftp主动模式能连接上,但获取不到目录;
  5. 前端学习-基础部分-css(二)
  6. Oracle 执行计划(三)-------表连接方式
  7. redis 初步认识三(设置登录密码)
  8. 如何解决一个从SkylineGlobe5版本升级到7版本遇到的小问题
  9. 搭建suse11.4内网源服务器
  10. 写个.net开发者的Linux迁移指南