3714: [PA2014]Kuglarz|贪心|思路题

似乎题解都是构造了最小生成树,可以换一种想法:
有这样一个结论,只要进行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;
}

你可能感兴趣的:(贪心)