至今觉得这场 edu 的 G 比 EF 都要简单……

不知道为什么出题人要把 \(m=0\) 放进去,先特判掉。

要求至少一个 \(0\),至少一个 \(1\),至少一个 \(2\),容斥一波,变成总方案数-没有 \(0\)-没有 \(1\)-没有 \(2\)+没有 \(01\)+没有 \(02\)+没有 \(12\)+没有 \(012\)。

没有 \(0\) 和没有 \(2\) 比较难搞,放到最后讨论。

没有 \(1\),考虑一个联通块,这个联通块所有数都一样,方案数是 \(2^{cnt}\),其中 \(cnt\) 是联通块个数。

没有 \(01\),也就是只有 \(2\),如果一个联通块中没有边(单独一个点),那么当然可以随便放,否则这个联通块所有数都是 \(1\)。方案数 \(2^{cnt2}\),其中 \(cnt2\) 是单独一个点的联通块个数。

没有 \(02\),也就是只有 \(1\),等价于将这个图黑白染色的方案数。如果可以黑白染色,那么方案数是 \(2^{cnt}\),否则是 \(0\)。

没有 \(12\),和没有 \(01\) 一样。方案数是 \(2^{cnt2}\)。

没有 \(012\),因为 \(m\ne 0\),显然不可能。方案数为 \(0\)。

接下来就考虑没有 \(0\) 的方案数(没有 \(2\) 是一样的)。

这个数据范围很明显是让我们折半搜索。我们不妨先搜后半部分。

对于每个合法的后半部分(即没有两个是 \(0\) 的点相邻),前半部分有哪些点不能是 \(0\) 我们是知道的。

转变一下,变成当前半部分选取的 \(0\) 点集合为 \(S\) 时,后半部分有多少种方案 \(val_S\)。(我是这么写的)

满足条件的 \(S\) 就是不能选的点的补集的子集。

实际上,在补集的 \(val\) 加个 \(1\),搜完后再做高维后缀和就能得到真的 \(val_S\)。应该不难理解。

然后再搜前半部分,对每个合法方案都加上它的 \(val\) 就行了。

时间复杂度,如果前半部分有 \(T\) 个点,复杂度是 \(O(2^TT+2^{n-T})\)。

由于我比较懒,我就取了 \(T=\frac{n}{2}\)。实际上要是 \(T\) 控制得够好,应该可以跑过 \(n=50\)。(取 \(T=23\),大概是 3e8,3.5s+CF 神机应该没问题)

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
typedef pair<int,int> PII;
const int maxn=1048576;
#define MP make_pair
#define PB push_back
#define lson o<<1,l,mid
#define rson o<<1|1,mid+1,r
#define FOR(i,a,b) for(int i=(a);i<=(b);i++)
#define ROF(i,a,b) for(int i=(a);i>=(b);i--)
#define MEM(x,v) memset(x,v,sizeof(x))
inline ll read(){
char ch=getchar();ll x=0,f=0;
while(ch<'0' || ch>'9') f|=ch=='-',ch=getchar();
while(ch>='0' && ch<='9') x=x*10+ch-'0',ch=getchar();
return f?-x:x;
}
int n,m,cnt,cnt2,bar,lim;
ll e[40],ans,val[maxn];
bool ind[40],vis[40],col[40],flag=true;
void dfs(int u){
vis[u]=true;
FOR(v,0,n-1) if((e[u]>>v)&1){
if(!vis[v]) col[v]=col[u]^1,dfs(v);
else{
if(col[v]!=(col[u]^1)) flag=false;
}
}
}
void dfs1(int dep,ll st,ll used){
if(dep==bar) return void(val[(~st)&(lim-1)]++);
dfs1(dep-1,st,used);
if(!((st>>dep)&1)) dfs1(dep-1,st|e[dep],used|1<<dep);
}
void dfs2(int dep,ll st,ll used){
if(dep==bar+1) return void(ans-=2*val[used&(lim-1)]);
dfs2(dep+1,st,used);
if(!((st>>dep)&1)) dfs2(dep+1,st|e[dep],used|1<<dep);
}
int main(){
n=read();m=read();
if(!m) return puts("0"),0;
FOR(i,0,n-1) ind[i]=true;
FOR(i,1,m){
int u=read()-1,v=read()-1;
e[u]|=1ll<<v;e[v]|=1ll<<u;
ind[u]=ind[v]=false;
}
FOR(i,0,n-1) if(ind[i]) cnt2++;
FOR(i,0,n-1) if(!vis[i]) cnt++,dfs(i);
ans=(1ll<<n)-(1ll<<cnt)+(1ll<<cnt2)+(1ll<<cnt2)+(flag?1ll<<cnt:0);
bar=(n-1)/2;lim=1<<(bar+1);
dfs1(n-1,0,0);
for(int i=1;i<lim;i<<=1)
for(int j=0;j<lim;j+=i<<1)
FOR(k,0,i-1) val[j+k]+=val[i+j+k];
dfs2(0,0,0);
printf("%lld\n",ans);
}

CF1221G Graph And Number(容斥,搜索,FMT)的更多相关文章

  1. XTU 1242 Yada Number 容斥

    Yada Number Problem Description: Every positive integer can be expressed by multiplication of prime ...

  2. [BZOJ1853][Scoi2010]幸运数字 容斥+搜索剪枝

    1853: [Scoi2010]幸运数字 Time Limit: 2 Sec  Memory Limit: 64 MBSubmit: 3202  Solved: 1198[Submit][Status ...

  3. bzoj 1853 容斥 + 搜索

    思路:先把所有幸运数字找出来, 把没有用的去掉,然后爆搜容斥,因为最多只会搜十几个就超过限制了, 所以是可行的. #include<bits/stdc++.h> #define LL lo ...

  4. 2019.01.17 bzoj1853: [Scoi2010]幸运数字(容斥+dfs)

    传送门 搜索菜题,然而第一次没有注意然后爆longlonglong longlonglong了. 题意:称所有数位由6,86,86,8组成的数为幸运数字,问一个一个区间[l,r][l,r][l,r]中 ...

  5. 【LOJ#6072】苹果树(矩阵树定理,折半搜索,容斥)

    [LOJ#6072]苹果树(矩阵树定理,折半搜索,容斥) 题面 LOJ 题解 emmmm,这题似乎猫讲过一次... 显然先\(meet-in-the-middle\)搜索一下对于每个有用的苹果数量,满 ...

  6. [HAOI2015]按位或(min-max容斥,FWT,FMT)

    题目链接:洛谷 题目大意:给定正整数 $n$.一开始有一个数字 $0$,然后每一秒,都有 $p_i$ 的概率获得 $i$ 这个数 $(0\le i< 2^n)$.一秒恰好会获得一个数.每获得一个 ...

  7. 【BZOJ1853】幸运数字(搜索,容斥)

    [BZOJ1853]幸运数字(搜索,容斥) 题面 BZOJ 洛谷 题解 成功轰下洛谷rk1,甚至超越了一个打表选手 这题思路很明显吧,先搞出来所有范围内的合法数字,然后直接容斥, 容斥的话显然没有别的 ...

  8. bzoj 4036 按位或 —— min-max容斥+FMT

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4036 min-max容斥:https://blog.csdn.net/ez_2016gdgz ...

  9. HDU - 4059: The Boss on Mars (容斥 拉格朗日 小小的优化搜索)

    pro: T次询问,每次给出N(N<1e8),求所有Σi^4 (i<=N,且gcd(i,N)==1) ; sol:  因为N比较小,我们可以求出素因子,然后容斥.  主要问题就是求1到P的 ...

随机推荐

  1. 用vbs和ADSI管理Windows账户

    ADSI (Active Directory Services Interface)是Microsoft新推出的一项技术,它统一了许多底层服务的编程接口,程序员可以使用一致的对象技术来访问这些底层服务 ...

  2. [译]ABP v1.0终于发布了!

    ABP v1.0终于发布了! 今天是个大日子!经过约3年的不断开发,第一个稳定的ABP版本,1.0,已经发布了.感谢为该项目做出贡献或试用过的每个人. 立即开始使用新的ABP框架:abp.io/get ...

  3. [译]发布ABP v0.19包含Angular UI选项

    发布ABP v0.19包含Angular UI选项 ABP v0.19已发布,包含解决的~90个问题和600+次提交. 新功能 Angular UI 终于,ABP有了一个SPA UI选项,使用最新的A ...

  4. Mixin Messenger 源码解读 1 — — WCDB Swift

    Mixin Messenger 早期采用 FMDB 后来切换至 WCDB 沿用至今,一直比较可靠稳定,这里分享一下使用心得和功能扩展. 关于 Mixin Messenger Mixin Messeng ...

  5. LeetCode 350: 两个数组的交集 II Intersection of Two Arrays II

    题目: 给定两个数组,编写一个函数来计算它们的交集. Given two arrays, write a function to compute their intersection. 示例 1: 输 ...

  6. PageHelper使用以及PageInfo中分页对象的转化

    在使用Mybatis查询数据库展示到前端的过程中不可避免的要考虑到分页问题,这时就引入了Mybatis的PageHelper插件,这个插件对分页功能进行了强有力的封装,只需要将查询出来的数据List集 ...

  7. yolov3和ssd的区别

    版权声明:本文为博主原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接和本声明. 本文链接:https://blog.csdn.net/BlowfishKing/article/d ...

  8. RaiseException函数逆向

    书中内容: 代码逆向: 存在一个疑问:为什么在ExceptionAddress本来是错误产生代码的地址,但这里给存入一个_RaiseException的偏移地址. 答案在下个函数中:rtlRaiseE ...

  9. Web前端基础(5):CSS(二)

    1. 盒模型 在CSS中,"box model"这一术语是用来设计和布局时使用,然后在网页中基本上都会显示一些方方正正的盒子.我们称为这种盒子叫盒模型. 盒模型有两种:标准模型和I ...

  10. 反射DataTable转实体类

    using System; using System.Collections.Generic; using System.Data; using System.Reflection; namespac ...