题目

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

算法

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

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

有多少种添加边的方案,使得这个图是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. jquery日历签到控件的实现

    calendar.js var calUtil = { //当前日历显示的年份 showYear:2015, //当前日历显示的月份 showMonth:1, //当前日历显示的天数 showDays ...

  2. grawlew 编译

    http://somefuture.iteye.com/blog/2002052 Gradlew behind a proxy systemProp.http.proxyHost=www.someho ...

  3. 条码知识之九:EAN-128条码(上)

     EAN-128码,现称GS1-128码,是专用于GS1系统中的条码,可以标注商品的附加信息,在商品信息的标识.产品的跟踪与追溯中有广泛的用途. EAN-128码来自于CODE-128码,在字符集.条 ...

  4. Windows Phone 8初学者开发—第6部分:设置应用程序的样式

    原文 Windows Phone 8初学者开发—第6部分:设置应用程序的样式 Source Code: http://aka.ms/absbeginnerdevwp8  PDF Version: ht ...

  5. HDU 2087 剪花布条 KMP

    题目地址:http://acm.hdu.edu.cn/showproblem.php?pid=2087 KMP匹配数—— AC代码: #include <iostream> #includ ...

  6. Service初步了解

    1.Service什么 Service它是一个应用程序组件,Android其中的四个核心组件之间 Service没有图形界面 通过经常使用来处理一些比较长耗时的操作 可以使用Service更新Cont ...

  7. Linux下的定时器

    以下摘自linux下的man文件:(man  getitimer) #include  <sys/time.h> int  getitimer(int which,  struct iti ...

  8. javascript 学习随笔

    var carname="Volvo"; var carname; 重新声明变量后变量的值不变,在这里还是volvo var carname="Volvo"; ...

  9. jquery倒计时自动跳转

    刚开始我用下面这种方法一直报错,不知是什么原因,就是多加了页面加载时调用这个方法,还请高手看到后小小留言解惑

  10. Struts2-ActionContext

    官方解释: The ActionContext is the context in which an {@link Action} is executed. Each context is basic ...