似乎题解都是构造了最小生成树,可以换一种想法:
有这样一个结论,只要进行n次不重复的询问就可以确定所有的杯子底下是否有球。(可以自己随手画画,似乎很显然…
这里所指的不重复是指通过以前的询问推不出这一次询问的结果,这样才算不重复。随便举一个栗子
比如说已经询问了
(3,8)
和
(3,5)
这时再去询问
(6,8)
就算是重复了
然后就可以排个序贪心从小到大寻找不重复的n个询问,然后怎么维护这个不重复呢?并查集!!
#include<algorithm>
#include<iostream>
#include<cstdlib>
#include<cstring>
#include<cstdio>
#include<vector>
#include<queue>
#include<cmath>
#include<set>
#include<map>
#define ll long long
#define lowbit(x) (x&(-x))
#define N 2200040
using namespace std;
int sc()
{
int i=0,f=1; char c=getchar();
while(c<'0'||c>'9'){if(c=='-')f=-1;c=getchar();}
while(c>='0'&&c<='9')i=i*10+c-'0',c=getchar();
return i*f;
}
struct W{int l,r,v;}a[N];
int n,m,tot,fa[2222];
ll ans;
bool cmp(W a,W b){return a.v<b.v;}
int find(int x)
{
return (fa[x]==x||fa[x]==0)?fa[x]=x:fa[x]=find(fa[x]);
}
int main()
{
n=sc();
for(int i=1;i<=n;i++)
for(int j=i;j<=n;j++)
a[++m].l=i,a[m].r=j+1,a[m].v=sc();
sort(a+1,a+m+1,cmp);
for(int i=1;i<=m;i++)
{
int fx=find(a[i].l),fy=find(a[i].r);
if(fx!=fy)
{
fa[fy]=fx;
tot++;
ans+=a[i].v;
if(tot==n)break;
}
}
cout<<ans;
return 0;
}