HDU 5739 Fantasia 双连通分量 树形DP
题意:
给出一个无向图,每个顶点有一个权值\(w\),一个连通分量的权值为各个顶点的权值的乘积,一个图的权值为所有连通分量权值之和。
设删除顶点\(i\)后的图\(G_i\)的权值为\(z_i\),求\(\sum i \times z_i \; mod \; 10^9 + 7\)。
分析:
这里找到了一篇论文,借其中的图说明一下:
左图为原图,首先找出所有的点双连通分量(一种颜色对应一个双连通),然后重新建一个图(右图):
对每个双连通加一个点,然后对这个新点和双连通中所有的点连边(正方形对应原图中的点,圆形对应新加的点)。
这样就得到了一个森林(block forest),因为一个双连通建了一颗树,而不同双连通之间通过割点连通。
所以,原图中的割点都是非叶子节点,其他点都是叶子节点。
将每个新点的权值设为\(1\),这样就不影响原来连通分量的权值。
删除点\(u\)后,\(G_u\)的权值为\(\sum f(v_i)\),其中点\(v_i\)与点\(u\)相邻,\(f(v_i)\)为所在连通分量的权值。
对每棵树任选一个新点作为根,计算\(dp(u)\)为以\(u\)为根的子树的权值,然后再一遍\(DFS\)就可以\(O(n)\)计算删除每个点后的权值。
注意:只有一个点的树需要单独处理。
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <vector>
#include <stack>
#include <map>
using std::pair;
using std::vector;
using std::stack;
const int maxn = 200000 + 10;
const int MOD = 1000000007;
typedef pair<int, int> Edge;
typedef long long LL;
LL mul(LL a, LL b) { return a * b % MOD; }
void add(LL& a, LL b) { a += b; if(a >= MOD) a -= MOD; }
LL sub(LL a, LL b) { b = MOD - b; a += b; if(a >= MOD) a -= MOD; return a; }
LL pow_mod(LL a, int p) {
LL ans = 1;
while(p) {
if(p & 1) ans = mul(ans, a);
a = mul(a, a);
p >>= 1;
}
return ans;
}
LL div(LL a, LL b) { return mul(a, pow_mod(b, MOD - 2)); }
int n, m, cc_cnt;
int w[maxn * 2];
bool vis[maxn * 2];
vector<int> G[maxn * 2], bcc[maxn], single;
int dfs_clock, bcc_cnt;
int pre[maxn], bccno[maxn];
stack<Edge> S;
int dfs(int u, int fa) {
int lowu = pre[u] = ++dfs_clock;
int child = 0;
for(int v : G[u]) {
Edge e(u, v);
if(!pre[v]) {
S.push(e);
child++;
int lowv = dfs(v, u);
lowu = std::min(lowu, lowv);
if(lowv >= pre[u]) {
bcc_cnt++; bcc[bcc_cnt].clear();
for(;;) {
Edge t = S.top(); S.pop();
int &x = t.first, &y = t.second;
if(bccno[x] != bcc_cnt) { bcc[bcc_cnt].push_back(x); bccno[x] = bcc_cnt; }
if(bccno[y] != bcc_cnt) { bcc[bcc_cnt].push_back(y); bccno[y] = bcc_cnt; }
if(x == u && y == v) break;
}
}
} else if(pre[v] < pre[u] && v != fa) {
S.push(e);
lowu = std::min(lowu, pre[v]);
}
}
if(fa < 0 && !child) single.push_back(u);
return lowu;
}
void find_bcc() {
memset(pre, 0, sizeof(pre));
memset(bccno, 0, sizeof(bccno));
single.clear();
dfs_clock = bcc_cnt = 0;
for(int i = 0; i < n; i++) if(!pre[i]) {
dfs(i, -1);
cc_cnt++;
}
}
void rebuild() {
for(int i = 0; i < n + bcc_cnt; i++) G[i].clear();
for(int i = 0; i < bcc_cnt; i++) {
int u = n + i; G[u].clear();
w[u] = 1;
for(int v : bcc[i + 1]) {
G[u].push_back(v);
G[v].push_back(u);
}
}
}
vector<int> root;
LL d[maxn * 2];
void dp(int u) {
d[u] = w[u];
vis[u] = true;
for(int v : G[u]) if(!vis[v]) {
dp(v);
d[u] = mul(d[u], d[v]);
}
}
LL ans[maxn], sum;
void solve(int u, int fa, LL prod) {
ans[u] = 0;
for(int v : G[u]) if(v != fa) {
solve(v, u, prod);
if(u < n) add(ans[u], d[v]);
}
if(u < n) add(ans[u], div(prod, d[u]));
add(ans[u], sub(sum, prod));
}
int main()
{
int T; scanf("%d", &T);
while(T--) {
scanf("%d%d", &n, &m);
for(int i = 0; i < n; i++) {
scanf("%d", w + i);
G[i].clear();
}
while(m--) {
int u, v; scanf("%d%d", &u, &v);
u--; v--;
G[u].push_back(v);
G[v].push_back(u);
}
find_bcc();
rebuild();
memset(vis, false, sizeof(vis));
root.clear();
sum = 0;
for(int u : single) add(sum, w[u]);
for(int i = 0; i < bcc_cnt; i++) {
int r = n + i; if(!vis[r]) {
root.push_back(r);
dp(r);
add(sum, d[r]);
}
}
memset(ans, 0, sizeof(ans));
for(int r : root) solve(r, -1, d[r]);
LL res = 0;
for(int u : single)
ans[u] = sub(sum, w[u]);
for(int i = 0; i < n; i++) add(res, mul(ans[i], i + 1));
printf("%lld\n", res);
}
return 0;
}
HDU 5739 Fantasia 双连通分量 树形DP的更多相关文章
- HDU 2242 考研路茫茫——空调教室(边双连通分量+树形dp+重边标号)
http://acm.hdu.edu.cn/showproblem.php?pid=2242 题意: 思路:首先求一下双连通分量,如果只有一个双连通分量,那么无论断哪根管子,图还是连通的. 最后只需要 ...
- hdu 4514 并查集+树形dp
湫湫系列故事——设计风景线 Time Limit: 6000/3000 MS (Java/Others) Memory Limit: 65535/32768 K (Java/Others)Tot ...
- [HDU 5293]Tree chain problem(树形dp+树链剖分)
[HDU 5293]Tree chain problem(树形dp+树链剖分) 题面 在一棵树中,给出若干条链和链的权值,求选取不相交的链使得权值和最大. 分析 考虑树形dp,dp[x]表示以x为子树 ...
- hdu 4612 Warm up 双连通+树形dp思想
Warm up Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 65535/65535 K (Java/Others) Total S ...
- HDU 2242 考研路茫茫—空调教室 (边双连通+树形DP)
<题目链接> 题目大意: 给定一个连通图,每个点有点权,现在需要删除一条边,使得整张图分成两个连通块,问你删除这条边后,两联通块点权值和差值最小是多少. 解题分析: 删除一条边,使原连通图 ...
- HDU 5739 Fantasia
可以将这个图转换成森林来进行树形dp求解.看了这篇具体教学才会的:http://www.cnblogs.com/WABoss/p/5696926.html 大致思路:求解一下点双连通分量(Tarjan ...
- HDU 5293 Tree chain problem 树形dp+dfs序+树状数组+LCA
题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=5293 题意: 给你一些链,每条链都有自己的价值,求不相交不重合的链能够组成的最大价值. 题解: 树形 ...
- hdu 4003 Find Metal Mineral 树形DP
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4003 Humans have discovered a kind of new metal miner ...
- HDU 5758 Explorer Bo(树形DP)
[题目链接] http://acm.hdu.edu.cn/showproblem.php?pid=5758 [题目大意] 给出一棵树,每条路长度为1,允许从一个节点传送到任意一个节点,现在要求在传送次 ...
随机推荐
- Java 编码规范有感
应小组要求,开发测试都需要考阿里编码规范,因此,相当于是突击了一下关于编码规范方面的知识,目前做的项目后期需要进行项目迁移,数据迁移,功能迁移... 各种迁移... 阿里巴巴编码规范(Java)考试地 ...
- 零基础逆向工程27_Win32_01_宽字符_MessageBox_win32调试输出
1 多字节字符 ASCII码表:0 ~ 2^7-1 扩展ASCII码表:2^7 ~ 2^8-1 什么是GB2312:1980年,两个字节存储一个汉字:不通用,别国会有乱码. UCICODE:只有一个字 ...
- 怎么旋转PDF文件的方向并保存成功
http://jingyan.baidu.com/article/59a015e39d7802f79488651e.html PDF格式的文档是非常普遍的一种阅读电子书格式,基本上非常好用了,不过有时 ...
- mybatis-动态sql2
mybatis的动态sql中常用的有 if where foreach set 项目沿用之前的. 1.dao层添加接口: package com.java1234.map ...
- cesium加载shp格式数据
方法一: shp格式转换为GeoJson格式并加载 首先注意shp的坐标系,要转换为WGS84,使用arcgis或QGIS 工具:http://mapshaper.org/: 注意:export时,输 ...
- 这些年,在wp平台打拼的日子
最近经常胃疼,在当地的镇医院看了几次都没有改善,只好去市医院照胃镜检查,发现有胃炎,虽然是很普通和常见的毛病,但这种毛病一但沾上,就很难根治,一喝酒或者吃饭不定时.熬夜.吃酸辣冷冻等食物都容易引起复发 ...
- IOS UIActionSheet(底部 弹出框的使用)
UIActionSheet *sheet = [[UIActionSheet alloc] initWithTitle:@"确定要注销?" delegate:self cancel ...
- POI对EXCEL的操作【重点:如何设置CELL格式为文本格式】
实际开发过程中通常用到的就是从数据库导出EXCEL表格了,JXL可以这样做,其实POI也可以(关于JXL与POI的异同可访问我之前总结的文章),之前写过POI对七种文档(当然也包括EXCEL)的内容读 ...
- UVA 11404 Plalidromic Subsquence (回文子序列,LCS)
最长回文子序列可以用求解原串s和反转串rv的LCS来得到,因为要求回文串分奇偶,dp[i][j]保存长度, 要求字典序最小,dp[i][j]应该表示回文子序列的端点,所以边界为单个字符,即i+j=le ...
- 2018.6.13 Java语言基础复习总结
Java语言基础与面向对象编程实践 第一章 初识Java 1.1机器语言 机器语言是指一台计算机全部的指令集合.机器语言室友0和1组成的二进制数,是一串串由0和1组成的指令序列,可将这些指令序列交给计 ...