AtCoder NIKKEI Programming Contest 2019 E. Weights on Vertices and Edges (并查集)

时间:2021-10-28 15:10:56

题目链接:https://atcoder.jp/contests/nikkei2019-qual/tasks/nikkei2019_qual_e

题意:给出一个 n 个点 m 条边的无向图,每个点和每条边都有权值,让你删除一些边,使得最终的图中满足条件:一条边存在当且仅当包含该边的连通块的点权值和大于等于该边权值,问最少要移走多少条边。

题解:删边不好做,考虑加边,对于每条边,判断加入是否合法。按边权从小到大排序进行加边,要加入一条边之前,若之前两点不连通,则合并起来变成一个连通块,可以用并查集来维护。存在一种情况,加入一些新的边之后,连通块权值增加,之前不能加入的边现在可以加入了,所以要记录之前连通块中已经加入的边的数量和全部边的数量以及最大的边权,便可以计算可以加入的边的数量。

 #include <bits/stdc++.h>
using namespace std;
#define ll long long
#define ull unsigned long long
#define mst(a,b) memset((a),(b),sizeof(a))
#define mp(a,b) make_pair(a,b)
#define pi acos(-1)
#define pii pair<int,int>
#define pb push_back
#define lowbit(x) ((x)&(-x))
const int INF = 0x3f3f3f3f;
const double eps = 1e-;
const int maxn = 1e5 + ;
const int maxm = 1e6 + ;
const ll mod = ; int a[maxn],fa[maxn]; struct edge {
int u,v,w;
bool operator < (const edge &x)const {
return w < x.w;
}
}e[maxn]; int findd(int x) {
return x == fa[x] ? x : fa[x] = findd(fa[x]);
} struct node {
int num,mxedge,ok;
ll sum;
}blo[maxn]; int main() {
#ifdef local
freopen("data.txt", "r", stdin);
// freopen("data.txt", "w", stdout);
#endif
int n,m;
scanf("%d%d",&n,&m);
for(int i = ; i <= n; i++) scanf("%d",&a[i]);
for(int i = ; i <= m; i++) scanf("%d%d%d",&e[i].u,&e[i].v,&e[i].w);
sort(e + , e + + m);
for(int i = ; i <= n; i++) {
fa[i] = i;
blo[i].num = blo[i].mxedge = blo[i].ok = ;
blo[i].sum = a[i];
}
int ans = ;
for(int i = ; i <= m; i++) {
int u = e[i].u, v = e[i].v, w = e[i].w;
int fu = findd(u), fv = findd(v);
if(fu == fv) {
blo[fu].num++;
blo[fu].mxedge = max(blo[fu].mxedge, w);
} else {
fa[fv] = fu;
blo[fu].num += blo[fv].num + ;
blo[fu].mxedge = max(max(blo[fu].mxedge, blo[fv].mxedge), w);
blo[fu].sum += blo[fv].sum;
blo[fu].ok += blo[fv].ok;
} if(blo[fu].sum >= blo[fu].mxedge) {
ans += blo[fu].num - blo[fu].ok;
blo[fu].ok = blo[fu].num;
}
}
printf("%d\n",m - ans);
return ;
}