HDU 2242 双连通分量 考研路茫茫——空调教室

思路就是求边双连通分量,然后缩点,再用树形DP搞一下。

代码和求强连通很类似,有点神奇,=_=,慢慢消化吧

  1 #include <cstdio>
  2 #include <cstring>
  3 #include <algorithm>
  4 #include <vector>
  5 #include <stack>
  6 using namespace std;
  7
  8 const int maxn = 10000 + 10;
  9 const int maxm = 20000 + 10;
 10
 11 int n, m;
 12 int a[maxn];
 13
 14 struct Edge
 15 {
 16     int v, nxt;
 17 }edges[maxm * 2];
 18 int ecnt;
 19 int head[maxn];
 20
 21 bool vis[maxn];
 22
 23 void AddEdge(int u, int v)
 24 {
 25     edges[ecnt].v = v;
 26     edges[ecnt].nxt = head[u];
 27     head[u] = ecnt++;
 28 }
 29
 30 stack<int> S;
 31 int dfs_clock, scc_cnt;
 32 int low[maxn], pre[maxn], sccno[maxn], w[maxn];
 33
 34 void dfs(int u, int fa)
 35 {
 36     low[u] = pre[u] = ++dfs_clock;
 37     S.push(u);
 38
 39     for(int i = head[u]; ~i; i = edges[i].nxt)
 40     {
 41         if(i == (fa ^ 1)) continue;
 42         int v = edges[i].v;
 43         if(!pre[v])
 44         {
 45             dfs(v, i);
 46             low[u] = min(low[u], low[v]);
 47         }
 48         else if(!sccno[v]) low[u] = min(low[u], pre[v]);
 49     }
 50
 51     if(pre[u] == low[u])
 52     {
 53         scc_cnt++;
 54         for(;;)
 55         {
 56             int x = S.top(); S.pop();
 57             sccno[x] = scc_cnt;
 58             w[scc_cnt] += a[x];
 59             if(x == u) break;
 60         }
 61     }
 62 }
 63
 64 void find_scc()
 65 {
 66     scc_cnt = dfs_clock = 0;
 67     memset(w, 0, sizeof(w));
 68     memset(pre, 0, sizeof(pre));
 69     memset(sccno, 0, sizeof(sccno));
 70     for(int i = 0; i < n; i++) if(!pre[i]) dfs(i, -1);
 71 }
 72
 73 vector<int> G[maxn];
 74
 75 void dfs2(int u)
 76 {
 77     vis[u] = true;
 78     for(int i = 0; i < G[u].size(); i++)
 79     {
 80         int v = G[u][i];
 81         if(vis[v]) continue;
 82         dfs2(v);
 83         w[u] += w[v];
 84     }
 85 }
 86
 87 int main()
 88 {
 89     while(scanf("%d%d", &n, &m) == 2 && n)
 90     {
 91         int sum = 0;
 92         for(int i = 0; i < n; i++) { scanf("%d", a + i); sum += a[i]; }
 93
 94         memset(head, -1, sizeof(head));
 95         ecnt = 0;
 96
 97         while(m--)
 98         {
 99             int u, v; scanf("%d%d", &u, &v);
100             AddEdge(u, v); AddEdge(v, u);
101         }
102
103         find_scc();
104
105         if(scc_cnt == 1) { puts("impossible"); continue; }
106
107         for(int i = 1; i <= scc_cnt; i++) G[i].clear();
108         for(int u = 0; u < n; u++)
109             for(int i = head[u]; ~i; i = edges[i].nxt)
110             {
111                 int v = edges[i].v;
112                 if(sccno[u] != sccno[v]) G[sccno[u]].push_back(sccno[v]);
113             }
114
115         memset(vis, false, sizeof(vis));
116         dfs2(1);
117
118         int ans = 1000000000;
119         for(int i = 1; i <= scc_cnt; i++) ans = min(ans, abs(sum - w[i] * 2));
120         printf("%d\n", ans);
121     }
122
123     return 0;
124 }

代码君

时间: 2024-10-07 11:57:57

HDU 2242 双连通分量 考研路茫茫——空调教室的相关文章

HDU 2242 考研路茫茫——空调教室(边双连通)

HDU 2242 考研路茫茫--空调教室 题目链接 思路:求边双连通分量,然后进行缩点,点权为双连通分支的点权之和,缩点完变成一棵树,然后在树上dfs一遍就能得出答案 代码: #include <cstdio> #include <cstring> #include <cstdlib> #include <algorithm> #include <vector> using namespace std; const int N = 10005;

hdu 2242 考研路茫茫――空调教室 (双连通入门题)

1 /********************************************************** 2 题目: 考研路茫茫——空调教室(hdu 2242) 3 链接: http://acm.hdu.edu.cn/showproblem.php?pid=2242 4 算法: 双联通 5 思路: 找到桥,然后计算桥两边的差,在把桥一边 6 的值压缩到和另一边相邻的数上. 7 8 *************************************************

hdoj 2242 考研路茫茫——空调教室 【无向图求边双联通 缩点 + 树形dp】

考研路茫茫--空调教室 Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Submission(s): 2447    Accepted Submission(s): 721 Problem Description 众所周知,HDU的考研教室是没有空调的,于是就苦了不少不去图书馆的考研仔们.Lele也是其中一个.而某教室旁边又摆着两个未装上的空调,更是引起人们无限YY

HDU 2242 考研路茫茫----空调教室

传送门 考研路茫茫——空调教室 Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submission(s): 2445    Accepted Submission(s): 720 Problem Description 众所周知,HDU的考研教室是没有空调的,于是就苦了不少不去图书馆的考研仔们.Lele也是其中一个.而某教室旁边又摆着两个未装上的空调,更是引起人们无

HDU 2242 考研路茫茫——空调教室 无向图缩环+树形DP

考研路茫茫——空调教室 Problem Description 众所周知,HDU的考研教室是没有空调的,于是就苦了不少不去图书馆的考研仔们.Lele也是其中一个.而某教室旁边又摆着两个未装上的空调,更是引起人们无限YY. 一个炎热的下午,Lele照例在教室睡觉的时候,竟然做起了空调教室的美梦. Lele梦到学校某天终于大发慈悲给某个教室安上了一个空调.而且建造了了M条通气管道,让整个教学楼的全部教室都直接或间接和空调教室连通上,构成了教室群,于是,全部教室都能吹到空调了. 不仅仅这样,学校发现教

HDU 2242 考研路茫茫——空调教室 (双连通分量+树形DP)

题目地址:HDU 2242 先用双连通分量缩点,然后形成一棵树,然后在树上做树形DP,求出每个点的子树和.然后找最小值即可.需要注意一下重边的问题,第一次返回父节点时可以忽略,因为这是反向边,然后之后再返回的时候就不是反向边了.不能忽略了. 代码如下: #include <iostream> #include <string.h> #include <math.h> #include <queue> #include <algorithm> #i

HDU - 2242 考研路茫茫――空调教室(树形DP+强连通分量)

题目大意:众所周知,HDU的考研教室是没有空调的,于是就苦了不少不去图书馆的考研仔们.Lele也是其中一个.而某教室旁边又摆着两个未装上的空调,更是引起人们无限YY. 一个炎热的下午,Lele照例在教室睡觉的时候,竟然做起了空调教室的美梦. Lele梦到学校某天终于大发慈悲给某个教室安上了一个空调.而且建造了了M条通气管道,让整个教学楼的全部教室都直接或间接和空调教室连通上,构成了教室群,于是,全部教室都能吹到空调了. 不仅仅这样,学校发现教室人数越来越多,单单一个空调已经不能满足大家的需求.于

考研路茫茫——空调教室HDU2242(Tarjan缩点)

题意:http://acm.hdu.edu.cn/showproblem.php?pid=2242 给你一个图,问你缩完点树上割边的做小绝对值差. 思路: 这题核算起来整整做了我一天(即24个小时)!!!一开始是MLE了近20发,然后TLE5.6发,再WA了一个晚上加一个下午. 有一种自闭是你突然对任何事物都失去追求:期中考?0分算了,这题再错我不学了还不行吗. 再加上HDU的评测机本来就很迷,该RE的判MLE,我N开1还MLE.服了嗷 ! 算法里要注意的就是:Tarjan里因为你补了一条反向边

hdu2422考研路茫茫——空调教室 tarjan+树形dp

//给一个无向图,其每个顶点都有权值,求去掉一条边,将这个图分为两部分 //问这两部分的所有顶点和的绝对值的最小值 //用tarjan缩点 , 缩点后为一棵树 //然后用树形dp求出其最小的绝对值 // ans = min(ans , (int)(abs((double)(sum - 2*dp[v])))) ; //其中dp[u] 表示以u点为根节点的子树的学生数 #include<cstdio> #include<cstring> #include<iostream>