清华集训2014 day1 task2 主旋律
题目
这可算是一道非常好的关于容斥原理的题了。
算法
好吧,这题我毫无思路,直接给正解。
首先,问题的正面不容易求,那么就求反面吧:
有多少种添加边的方案,使得这个图是DAG图(这里及以下所说的DAG图都是指这个图不是整个强连通的)。
利用容斥原理,DAG图的特征是有至少一个入度为\(0\)的点并且这个图不止一个点(这里及以下所说的点都是指求强连通后的点),就根据这个进行容斥。
设\(g(set)\)为集合里的点都是入度为\(0\)的方案数,注意,这个有点特别,比如这个:

它的值应该为\(0\)。虽然有两种方案,一种是两条边都添加,另一种是都不添加。但是,都不添加的话,由于两个点(偶数个点),系数应该是负的!!所以\(g(set) = (-1) + (1) = 0\)。
设\(f(set)\)为有多少种添加边的方案,使得整个图强连通。
好吧,我真的不知道怎样说清楚。-_-|||。不过看了代码应该就很清楚了。。。
这里的\(h(set)\)是这些点的诱导子图有多少条边。
至于时间复杂度,就是$$O(\sum_{i=0}{n}{Ci_n\cdot 2^i})$$
根据二次项定理可知时间复杂度为\(O(3^n)\)。
#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
#include <assert.h>
using namespace std;
#ifdef debug
#define ep(...) fprintf(stderr, __VA_ARGS__)
#else
#define ep(...) assert(true)
#endif
template <class T>
void relax(T &a, const T &b)
{
if (b > a) a = b;
}
template <class T>
void tension(T &a, const T &b)
{
if (b < a) a = b;
}
typedef long long i64;
const int MaxN = 15;
const int MaxSet = (1 << MaxN) + 3;
const int MOD = (int) 1e9 + 7;
int main()
{
#if defined(debug) || defined(local)
freopen("in", "r", stdin);
freopen("out", "w", stdout);
#endif
int n, m;
scanf("%d%d", &n, &m);
static int in[MaxSet], out[MaxSet];
for (int i = 0; i < m; i ++)
{
int u, v;
scanf("%d%d", &u, &v);
u --, v --;
in[1 << v] |= 1 << u;
out[1 << u] |= 1 << v;
}
static int cnt[MaxSet], power2[MaxSet];
power2[0] = 1;
for (int i = 1; i < 1 << n; i ++)
{
cnt[i] = cnt[i ^ (i & -i)] + 1;
power2[i] = power2[i - 1] << 1;
if (power2[i] >= MOD) power2[i] -= MOD;
}
static int f[MaxSet], g[MaxSet], h[MaxSet];
for (int i = 1; i < 1 << n; i ++)
{
if (cnt[i] == 1)
{
f[i] = g[i] = 1;
h[i] = 0;
continue;
}
int u = i & -i;
h[i] = h[i ^ u] + cnt[out[u] & i] + cnt[in[u] & i];
for (int sub = (i - 1) & i; sub; sub = (sub - 1) & i)
if (sub & u)
{
g[i] = ((i64) g[i] + MOD - (i64) g[i ^ sub] * f[sub]) % MOD;
}
static int w[MaxSet];
int rev_f = 0;
for (int sub = i; sub; sub = (sub - 1) & i)
{
if (sub == i)
{
w[sub] = 0;
}
else
{
int v = (i ^ sub) & -(i ^ sub);
w[sub] = w[sub ^ v] - cnt[out[v] & (i ^ sub)] + cnt[in[v] & sub];
}
rev_f = ((i64) rev_f + (i64) g[sub] * power2[h[i ^ sub] + w[sub]]) % MOD;
}
f[i] = (power2[h[i]] + MOD - rev_f) % MOD;
ep("tmp_g: %d\n", g[i]);
g[i] = (g[i] + f[i]) % MOD;
ep("%d: %d %d %d\n", i, f[i], g[i], h[i]);
}
cout << f[(1 << n) - 1] << endl;
ep("answer %d", f[(1 << n) - 1]);
return 0;
}
清华集训2014 day1 task2 主旋律的更多相关文章
- 清华集训2014 day1 task1 玛里苟斯
题目 这可算是描述很简单的一道题了!但是不简单. \(S\)是一个可重集合,\(S = \{a_1, a_2, \dots, a_n \}\). 等概率随机取\(S\)的一个子集\(A = \{a_{ ...
- 清华集训2014 day1 task3 奇数国
题目 题目看起来好像很难的样子!其实不然,这是最简单的一道题. 算法 首先要注意的是: \(number \cdot x + product \cdot y = 1\) ,那么我们称\(number\ ...
- uoj 41 【清华集训2014】矩阵变换 婚姻稳定问题
[清华集训2014]矩阵变换 Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://uoj.ac/problem/41 Description 给出 ...
- AC日记——【清华集训2014】奇数国 uoj 38
#38. [清华集训2014]奇数国 思路: 题目中的number与product不想冲: 即为number与product互素: 所以,求phi(product)即可: 除一个数等同于在模的意义下乘 ...
- UOJ#46. 【清华集训2014】玄学
传送门 分析 清华集训真的不是人做的啊嘤嘤嘤 我们可以考虑按操作时间把每个操作存进线段树里 如果现在点x正好使一个整块区间的右端点则更新代表这个区间的点 我们不难发现一个区间会因为不同的操作被分成若干 ...
- 清华集训2014 sum
清华集训2014sum 求\[∑_{i=1}^{n}(-1)^{⌊i√r⌋}\] 多组询问,\(n\leq 10^9,t\leq 10^4, r\leq 10^4\). 吼题解啊 具体已经讲得很详细了 ...
- UOJ#37. 【清华集训2014】主旋律
题目大意: 传送门 题解: 神题……Orz. 首先正难则反. 设$f_S$表示选取点集状态为s时,这部分图可以构成非强联通图的方案数. 设$p_{S,i}$表示点集s缩点后有i个入度为0点的方案数,保 ...
- [清华集训2015 Day1]主旋律-[状压dp+容斥]
Description Solution f[i]表示状态i所代表的点构成的强连通图方案数. g[i]表示状态i所代表的的点形成奇数个强连通图的方案数-偶数个强连通图的方案数. g是用来容斥的. 先用 ...
- 【uoj#37/bzoj3812】[清华集训2014]主旋律 状压dp+容斥原理
题目描述 求一张有向图的强连通生成子图的数目对 $10^9+7$ 取模的结果. 题解 状压dp+容斥原理 设 $f[i]$ 表示点集 $i$ 强连通生成子图的数目,容易想到使用总方案数 $2^{sum ...
随机推荐
- React Native-目前最火的前端技术?
做为一名产品经理,你是否遇到过这样的窘境,“帮我把字体调成 16号呗,颜色变成 #FFFF00FF,老大说这里最好改一下”,作为一名 app 的开发只能无奈但心里窃喜的告诉你,“只能等下个版本了,必须 ...
- Null指针
C++ Null 指针 C++ 指针 C++ 指针 在变量声明的时候,如果没有确切的地址可以赋值,为指针变量赋一个 NULL 值是一个良好的编程习惯.赋为 NULL 值的指针被称为空指针. NULL ...
- Python之路:Python简介
Python前世今生 python的创始人为吉多·范罗苏姆(Guido van Rossum).1989年的圣诞节期间他为了在阿姆斯特丹打发时间,决心开发一个新的脚本解释程序,作为ABC语言的一种继承 ...
- TCP/IP学习笔记
1. 2. >>> int socket(int af, int type,int protocol);//创建套接字,返回从文件描述表中取出新的索引号(int);AF_INET ...
- 英文:known good assembly(KGA) / 中文:确认好的组装件,已知好组装件
英文:known good assembly(KGA) / 中文:确认好的组装件,已知好组装件 正确地操作印制板装配,并可作为标准件与其它同类型装配件比较的组装.也称黄金组装.
- cmake 学习笔记(三)
转自:http://blog.csdn.net/dbzhang800/article/details/6329314 接前面的 Cmake学习笔记(一) 与 Cmake学习笔记(二) 继续学习 cma ...
- 如何修改Sublime 侧边栏Sidebar的颜色
参考自:http://blog.csdn.net/a497393102/article/details/10563791 首先要找到 Default.sublime-theme 文件, 点击 subl ...
- Uva 3226 Symmetry
题目给出一些点的坐标(横坐标,纵坐标),没有重叠的点,求是否存在一条竖线(平行于y轴的线),使线两边的点左右对称. 我的思路:对于相同的纵坐标的点,即y值相同的点,可以将x的总和计算出,然后除以点的数 ...
- Python类的继承演示样例
class Pet: __name = "" def __init__(self, name): self.__name = name def bark(self): return ...
- TCP/IP之TCP交互数据流、成块数据流
建立在TCP协议上的网络协议有telnet,ssh,ftp,http等等.这些协议根据数据吞吐量来分成两大类: (1)交互数据类型,例如telnet,ssh,这种类型的协议在大多数情况下只是做小流量的 ...