loj#2509. 「AHOI / HNOI2018」排列(思维题 set)

题意

题目链接

Sol

神仙题Orz

首先不难看出如果我们从(a_i)(i)连一条边,我们会得到以(0)为根的树(因为每个点一定都有一个入度,出现环说明无解),同时在进行排列的时候需要保证父亲节点一定在孩子节点之前出现

接下来考虑直接贪心。对于某些权值很小的点,我们需要让其尽早出现,同时又要满足选择的条件。

那么我们可以从小的点开始,依次向他的父亲合并,并删除该点(也就是如果父亲一但被删除,那么这个点立马被删除)

下面的内容抄袭摘抄自这里

然后直接用set搞一搞

复杂度:(O(nlog n))

#include<bits/stdc++.h>
#define LL long long 
using namespace std;
const int MAXN = 5e5 + 10, SS = 1e7 + 10;
template<typename A, typename B> inline void chmax(A &x, B y) {
	x = x > y ? x : y;
}
inline int read() {
	char c = getchar(); int x = 0, f = 1;
	while(c < '0' || c > '9') {if(c == '-') f = -1; c = getchar();}
	while(c >= '0' && c <= '9') x = x * 10 + c - '0', c = getchar();
	return x * f;
}
int N, a[MAXN], fa[MAXN], vis[MAXN], ufa[MAXN];
LL w[MAXN], siz[MAXN];
vector<int> v[MAXN];
struct comp {
	bool operator ()(int x, int y) {
		return w[x] * siz[y] == w[y] * siz[x] ? x < y : w[x] * siz[y] < w[y] * siz[x];
	}
};
int find(int x) {
	return ufa[x] == x ? ufa[x] : ufa[x] = find(ufa[x]);
}
set<int, comp> s;
int dfs(int x) {
	vis[x] = 1;
	for(auto &to : v[x]) {
		if(vis[to] == 1) return 1;
		if(dfs(to)) return 1;
	}
	vis[x] = 2;
	return 0;
}
int main() {
	N = read();
	for(int i = 1; i <= N; i++) {
		fa[i] = read(); 
		v[fa[i]].push_back(i);
	}
	for(int i = 1; i <= N; i++) 
		if(!vis[i]) 
			if(dfs(i)) {puts("-1"); return 0;}
	for(int i = 1; i <= N; i++) w[i] = read(), ufa[i] = i, siz[i] = 1, s.insert(i);
	siz[0] = 1; ufa[0] = 0;
	LL ans = 0;
	for(int i = 1; i <= N; i++) {
		int x = *s.begin(); s.erase(s.begin());
		int f = find(fa[find(x)]);
		if(f) s.erase(f);
		ans += siz[f] * w[x];
		siz[f] += siz[x]; w[f] += w[x]; 
		ufa[x] = f;
		if(f) s.insert(f);
	}
	cout << ans;
	return 0;
}
/*
3
0 1 1
5 7 3
*/
原文地址:https://www.cnblogs.com/zwfymqz/p/10415475.html