bzoj 2429 聪明的猴子

来源:互联网 发布:linux系统克隆教程 编辑:程序博客网 时间:2024/05/21 14:57

聪明的猴子

题目描述

在一个热带雨林中生存着一群猴子,它们以树上的果子为生。昨天下了一场大雨,现在雨过天晴,但整个雨林的地表还是被大水淹没着,部分植物的树冠露在水面上。猴子不会游泳,但跳跃能力比较强,它们仍然可以在露出水面的不同树冠上来回穿梭,以找到喜欢吃的果实。现在,在这个地区露出水面的有N棵树,假设每棵树本身的直径都很小,可以忽略不计。我们在这块区域上建立直角坐标系,则每一棵树的位置由其所对应的坐标表示(任意两棵树的坐标都不相同)。在这个地区住着的猴子有M个,下雨时,它们都躲到了茂密高大的树冠中,没有被大水冲走。由于各个猴子的年龄不同、身体素质不同,它们跳跃的能力不同。有的猴子跳跃的距离比较远(当然也可以跳到较近的树上),而有些猴子跳跃的距离就比较近。这些猴子非常聪明,它们通过目测就可以准确地判断出自己能否跳到
对面的树上。

【问题】 

现已知猴子的数量及每一个猴子的最大跳跃距离,还知道露出水面的每一棵树的坐标,你的任务是统计有多少个猴子可以在这个地区露出水面的所有树冠上觅食。

思路

每个猴子的跳跃距离是一样的,那么我们需要做的就是把图拿来求一个最长边的最小值,也就是最小生成树。

代码

#include<cstdio>#include<cstring>#include<iostream>#include<algorithm>#include<cmath>#include<vector>using namespace std;const int N=500+5;const int M=1000+5;int jump[N],father[N],m,flag=0,n,cnt=0,ans=0;double minc;struct node{    int x,y;}tree[M];struct edge{    int u,v;    double w;}e[1000001];bool cmp(edge a,edge b){    return a.w<b.w;}int find(int x){    if (father[x]==x) return x;    else return father[x]=find(father[x]);}int main(){    scanf("%d",&m);    for (int i=1;i<=m;i++)    scanf("%d",jump+i);    scanf("%d",&n);    for (int i=1;i<=n;i++)    {        scanf("%d%d",&tree[i].x,&tree[i].y);        father[i]=i;    }    for (int i=1;i<=n;i++)    for (int j=i+1;j<=n;j++)    {        e[++cnt].u=i;        e[cnt].v=j;        e[cnt].w=(tree[i].x-tree[j].x)*(tree[i].x-tree[j].x)+(tree[i].y-tree[j].y)*(tree[i].y-tree[j].y);    }    sort(e+1,e+cnt+1,cmp);    for (int i=1;i<=cnt;i++)    {        int t1=find(e[i].u),t2=find(e[i].v);        if (t1!=t2)        {            father[t2]=t1;            flag++;            if (flag==n-1)            {                minc=e[i].w;                break;            }        }    }    for (int i=1;i<=m;i++)    if (jump[i]*jump[i]>=minc)ans++;    printf("%d",ans);    return 0; }
原创粉丝点击