思路:不断删除度数为1的点可以类似拓扑排序那样删除,然后并查集合并的时候统计一下顶点数和权值和就可以了
#include<bits/stdc++.h>
using namespace std;
#define LL long long
const int maxn = 10000+7;
vector<int>e[maxn];
int w[maxn],degree[maxn],fa[maxn],vis[maxn],cnt[maxn];
LL sum[maxn];
int n,m;
int Find(int x){return x==fa[x]?x:fa[x]=Find(fa[x]);}
void merge(int u,int v)
{
int uu = Find(u);
int vv = Find(v);
if(uu!=vv)
{
fa[uu]=vv;
sum[vv]+=sum[uu];
cnt[vv]+=cnt[uu];
}
}
void init()
{
for(int i = 0;i<=n;i++)
e[i].clear();
memset(degree,0,sizeof(degree));
memset(vis,0,sizeof(vis));
for(int i = 1;i<=n;i++)
{
fa[i]=i;
sum[i]=w[i];
cnt[i]=1;
}
}
void topo()
{
queue<int>q;
for(int i = 1;i<=n;i++)
if(degree[i]<=1)
{
vis[i]=1;
q.push(i);
}
while(!q.empty())
{
int u = q.front();
q.pop();
for(int i = 0;i<e[u].size();i++)
{
int v = e[u][i];
if(--degree[v]<=1 && !vis[v])
{
q.push(v);
vis[v]=1;
}
}
}
}
int main()
{
int T;
scanf("%d",&T);
while(T--)
{
scanf("%d%d",&n,&m);
for(int i = 1;i<=n;i++)
scanf("%d",&w[i]);
init();
for(int i = 1;i<=m;i++)
{
int u,v;
scanf("%d%d",&u,&v);
e[u].push_back(v);
e[v].push_back(u);
degree[u]++,degree[v]++;
}
topo();
for(int u = 1;u<=n;u++)
{
if(!vis[u])
{
for(int i = 0;i<e[u].size();i++)
{
int v = e[u][i];
if(!vis[v])
merge(u,v);
}
}
}
LL ans = 0;
for(int u = 1;u<=n;u++)
{
if(Find(u)==u && (cnt[u]&1) && !vis[u])
ans+=sum[u];
}
printf("%lld\n",ans);
}
}