列表

详情


NC24405. [USACO 2013 Ope G]Yin and Yang

描述

Farmer John is planning his morning walk on the farm. The farm is structured like a tree: it has N barns (1 <= N <= 100,000) which are connected by N-1 edges such that he can reach any barn from any other. Farmer John wants to choose a path which starts and ends at two different barns, such that he does not traverse any edge twice. He worries that his path might be a little long, so he also wants to choose another "rest stop" barn located on this path (which is distinct from the start or the end). Along each edge is a herd of cows, either of the Charcolais (white hair) or the Angus (black hair) variety. Being the wise man that he is, Farmer John wants to balance the forces of yin and yang that weigh upon his walk. To do so, he wishes to choose a path such that he will pass by an equal number of Charcolais herds and Angus herds-- both on the way from the start to his rest stop, and on the way from the rest stop to the end. Farmer John is curious how many different paths he can choose that are "balanced" as described above. Two paths are different only if they consist of different sets of edges; a path should be counted only once even if there are multiple valid "rest stop" locations along the path that make it balanced. Please help determine the number of paths Farmer John can choose!

输入描述

* Line 1: The integer N.
* Lines 2..N: Three integers a_i, b_i and t_i, representing the two
barns that edge i connects. t_i is 0 if the herd along that
edge is Charcolais, and 1 if the herd is Angus.

输出描述

* Line 1: One integer, representing the number of possible paths
Farmer John can choose from.

示例1

输入:

7
1 2 0
3 1 1
2 4 0
5 2 0
6 3 1
5 7 1

输出:

1

说明:

INPUT DETAILS:
There are 7 barns and 6 edges. The edges from 1 to 2, 2 to 4 and 2 to 5 have
Charcolais herds along them.

OUTPUT DETAILS:
No path of length 2 can have a suitable rest stop on it, so we can only
consider paths of length 4. The only path that has a suitable rest stop is
3-1-2-5-7, with a rest stop at 2.

原站题解

上次编辑到这里,代码来自缓存 点击恢复默认模板

C++11(clang++ 3.9) 解法, 执行用时: 744ms, 内存消耗: 17912K, 提交时间: 2020-07-07 23:08:01

#include<bits/stdc++.h>
using namespace std;
const int maxn=100009;

int n;

int head[maxn],cntedge;
int nex[maxn<<1],to[maxn<<1],dist[maxn<<1];
void Addedge(int x,int y,int z){
	nex[++cntedge]=head[x];
	to[cntedge]=y;
	dist[cntedge]=z;
	head[x]=cntedge;
}

int vis[maxn],siz[maxn],mxsiz[maxn];
int getroot(int x,int fa,int tot){
	siz[x]=1;mxsiz[x]=0;
	int cen=0;
	for(int i=head[x];i;i=nex[i]){
		int v=to[i];
		if(vis[v]||v==fa)continue;
		int vcen=getroot(v,x,tot);
		if(!cen||mxsiz[vcen]<mxsiz[cen])cen=vcen;
		siz[x]+=siz[v];
		mxsiz[x]=max(mxsiz[x],siz[v]);
	}
	mxsiz[x]=max(mxsiz[x],tot-siz[x]);
	if(!cen||mxsiz[x]<mxsiz[cen])cen=x;
	return cen;
}


long long ans;
map<int,int>tong;
int q0[maxn],n0;
int q1[maxn],n1;
void dp(int x,int fa,int del){
	if(tong[del]!=0)q1[++n1]=del;
	else q0[++n0]=del;
	tong[del]++;
	for(int i=head[x];i;i=nex[i]){
		int v=to[i];
		if(vis[v]||v==fa)continue;
		if(dist[i])dp(v,x,del+1);
		else dp(v,x,del-1);
	}
	tong[del]--;
}

void calc(int x){
	map<int,int>S0,S1;
	for(int i=head[x];i;i=nex[i]){
		int v=to[i];
		if(vis[v])continue;
		n0=0;n1=0;tong.clear();
		if(dist[i])dp(v,x,1);
		else dp(v,x,-1);
		for(int j=1;j<=n0;++j){
			ans+=S1[-q0[j]];
			if(q0[j]==0)ans+=S0[0];
		}
		for(int j=1;j<=n1;++j){
			ans+=S0[-q1[j]]+S1[-q1[j]];
		}
		for(int j=1;j<=n0;++j)S0[q0[j]]++;
		for(int j=1;j<=n1;++j)S1[q1[j]]++;
	}
	ans+=S1[0];
}
void solve(int x,int tot){
	vis[x]=1;calc(x);
	for(int i=head[x];i;i=nex[i]){
		int v=to[i];
		if(vis[v])continue;
		int vsiz=(siz[v]<siz[x])?siz[v]:tot-siz[x];
		int vcen=getroot(v,x,vsiz);
		solve(vcen,vsiz);
	}
}

int main(){
	scanf("%d",&n);
	for(int i=1;i<=n-1;++i){
		int x,y,z;
		scanf("%d%d%d",&x,&y,&z);
		Addedge(x,y,z);
		Addedge(y,x,z);
	}
	
	int cen=getroot(1,0,n);
	solve(cen,n);
	
	printf("%lld\n",ans);
	return 0;
}

上一题