题目

这可算是一道非常好的关于容斥原理的题了。

算法

好吧,这题我毫无思路,直接给正解。

首先,问题的正面不容易求,那么就求反面吧:

有多少种添加边的方案,使得这个图是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 主旋律的更多相关文章

  1. 清华集训2014 day1 task1 玛里苟斯

    题目 这可算是描述很简单的一道题了!但是不简单. \(S\)是一个可重集合,\(S = \{a_1, a_2, \dots, a_n \}\). 等概率随机取\(S\)的一个子集\(A = \{a_{ ...

  2. 清华集训2014 day1 task3 奇数国

    题目 题目看起来好像很难的样子!其实不然,这是最简单的一道题. 算法 首先要注意的是: \(number \cdot x + product \cdot y = 1\) ,那么我们称\(number\ ...

  3. uoj 41 【清华集训2014】矩阵变换 婚姻稳定问题

    [清华集训2014]矩阵变换 Time Limit: 20 Sec  Memory Limit: 256 MB 题目连接 http://uoj.ac/problem/41 Description 给出 ...

  4. AC日记——【清华集训2014】奇数国 uoj 38

    #38. [清华集训2014]奇数国 思路: 题目中的number与product不想冲: 即为number与product互素: 所以,求phi(product)即可: 除一个数等同于在模的意义下乘 ...

  5. UOJ#46. 【清华集训2014】玄学

    传送门 分析 清华集训真的不是人做的啊嘤嘤嘤 我们可以考虑按操作时间把每个操作存进线段树里 如果现在点x正好使一个整块区间的右端点则更新代表这个区间的点 我们不难发现一个区间会因为不同的操作被分成若干 ...

  6. 清华集训2014 sum

    清华集训2014sum 求\[∑_{i=1}^{n}(-1)^{⌊i√r⌋}\] 多组询问,\(n\leq 10^9,t\leq 10^4, r\leq 10^4\). 吼题解啊 具体已经讲得很详细了 ...

  7. UOJ#37. 【清华集训2014】主旋律

    题目大意: 传送门 题解: 神题……Orz. 首先正难则反. 设$f_S$表示选取点集状态为s时,这部分图可以构成非强联通图的方案数. 设$p_{S,i}$表示点集s缩点后有i个入度为0点的方案数,保 ...

  8. [清华集训2015 Day1]主旋律-[状压dp+容斥]

    Description Solution f[i]表示状态i所代表的点构成的强连通图方案数. g[i]表示状态i所代表的的点形成奇数个强连通图的方案数-偶数个强连通图的方案数. g是用来容斥的. 先用 ...

  9. 【uoj#37/bzoj3812】[清华集训2014]主旋律 状压dp+容斥原理

    题目描述 求一张有向图的强连通生成子图的数目对 $10^9+7$ 取模的结果. 题解 状压dp+容斥原理 设 $f[i]$ 表示点集 $i$ 强连通生成子图的数目,容易想到使用总方案数 $2^{sum ...

随机推荐

  1. [译]SSIS 通过环境变量配置数据源连接参数

    场景 希望在包执行的时候可以随意选择参数,这时候我们可以用环境变量 . 另外所有包都可以用环境变量,有大量包的时候就比较方便. Step 1: 创建SSIS包 在Data Flow Task里面创建 ...

  2. UVa---------10935(Throwing cards away I)

    题目: Problem B: Throwing cards away I Given is an ordered deck of n cards numbered 1 to n with card 1 ...

  3. centos下卸载jdk

    链接地址:http://blog.csdn.net/shuixin536/article/details/8954011 http://sunqiusong.email.blog.163.com/bl ...

  4. 「C」 数组、字符串、指针

    一.数组 (一)数组 概念:用来存储一组数据的构造数据类型 特点:只能存放一种类型的数据,如全部是int型或者全部是char型,数组里的数据成为元素. (二)数组的定义 格式: 类型 数组名[元素个数 ...

  5. Java Web学习笔记(1)

    1.项目名称用小写,类名用大小写骆驼式,对象名用骆驼式但是第一个字母是小写: 2.写对象属性时要空行,第一个方法也要空行,一般要加注释: 3.new 新的对象时等号左右要空格,if语句左右摇有空格: ...

  6. Log4J logger图片

  7. js 时间戳转为日期格式

    原文:js 时间戳转为日期格式 js 时间戳转为日期格式 什么是Unix时间戳(Unix timestamp): Unix时间戳(Unix timestamp),或称Unix时间(Unix time) ...

  8. 编程算法 - 最长公共子序列(LCS) 代码(C)

    最长公共子序列(LCS) 代码(C) 本文地址: http://blog.csdn.net/caroline_wendy 题目: 给定两个字符串s,t, 求出这两个字符串最长的公共子序列的长度. 字符 ...

  9. ASP.NET事务存储过程

    --修改存储过程 alter proc proc_get_student as select * from student; asp.net 的事务就是针对数据层来处理的呀! 没有数据处理不能使用事务 ...

  10. iOS开发进阶之 UIWebView

    刚接触IOS开发1年多,现在对于混合式移动端开发越来越流行,因为开发成本上.速度上都比传统的APP开发要好,混合式开发是传统模式与PC网页端相结合的模式.那么提到了 APP的混合模式开发,在Andro ...