bzoj 2429 聪明的猴子

聪明的猴子

题目描述

在一个热带雨林中生存着一群猴子,它们以树上的果子为生。昨天下了一场大雨,现在雨过天晴,但整个雨林的地表还是被大水淹没着,部分植物的树冠露在水面上。猴子不会游泳,但跳跃能力比较强,它们仍然可以在露出水面的不同树冠上来回穿梭,以找到喜欢吃的果实。现在,在这个地区露出水面的有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; 
}
  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值