Ponds HDU

Posted acmloser

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Ponds HDU相关的知识,希望对你有一定的参考价值。

原题链接
考察:拓扑排序+并查集
错误思路:
  离线处理,\\(d[i]\\)记录i的入度.如果\\(d[i]<=1\\)就不纳入并查集,否则就加入.
错误原因:
  删除一个点,可能使别的点\\(d[i]<=1\\)
思路:
  因为\\(d[i]\\)是会级联影响的,所以我们用拓扑排序求\\(d[i]<=1\\)的点.但是注意题目是无向边,我们要特判环防止TLE.

Code

#include <iostream>
#include <cstring>
#include <algorithm>
using namespace std;
typedef long long LL;
const int N = 10010,M = 100010;
struct Road{
	int to,ne;
}road[M<<1];
int n,m,d[N],idx,h[N],q[N],p[N],L[N],R[N];
LL w[N];
bool vis[N];
int sz[N];
void add(int a,int b)
{
	road[idx].to =b,road[idx].ne = h[a],h[a] = idx++;
}
int findf(int x)
{
	if(p[x]!=x) return p[x] = findf(p[x]);
	return p[x];
}
void merge(int a,int b)
{
	int pa = findf(a),pb = findf(b);
	if(pa==pb) return;
	sz[pb]+=sz[pa];
	w[pb]+=w[pa];
	p[pa] = pb;
}
void topsort()
{
	int hh = 0,tt=-1;
	for(int i=1;i<=n;i++)
	  if(d[i]<=1) q[++tt] = i;
	while(hh<=tt)
	{
		int u = q[hh++];
		vis[u] = 1;
		for(int i=h[u];~i;i=road[i].ne)
		{
			int v = road[i].to;
			if(vis[v]) continue;
			if(--d[v]<=1) q[++tt] = v;
		}
	}
}
int main()
{
	int T;
	scanf("%d",&T);
	while(T--)
	{
		memset(h,-1,sizeof h);
		memset(d,0,sizeof d);
		memset(vis,0,sizeof vis);
		scanf("%d%d",&n,&m);
		idx =0;
		for(int i=1;i<=n;i++) scanf("%lld",&w[i]),p[i] = i,sz[i] =1;
		for(int i=1;i<=m;i++)
		{
			int a,b;
			scanf("%d%d",&a,&b);
			d[a]++,d[b]++;
			add(a,b); add(b,a);
			L[i] = a,R[i] = b;
		}
		topsort();
		for(int i=1;i<=m;i++)
		{
			if(vis[L[i]]||vis[R[i]]) continue;
			merge(L[i],R[i]);
		}
		LL res = 0;
		for(int i=1;i<=n;i++)
		{
			int pa = findf(i);
			if(vis[pa]) continue;
			vis[pa] = 1;
			if(sz[pa]&1) res+=w[pa];
		}
		printf("%lld\\n",res);
	}
	return 0;
}

以上是关于Ponds HDU的主要内容,如果未能解决你的问题,请参考以下文章

HDU4057 Rescue the Rabbit(AC自动机+状压DP)

HDU3247 Resource Archiver(AC自动机+BFS+DP)

POJ - 2778 ~ HDU - 2243 AC自动机+矩阵快速幂

微信小程序代码片段

VSCode自定义代码片段——CSS选择器

谷歌浏览器调试jsp 引入代码片段,如何调试代码片段中的js