10684 我要摇出一个妹子

10684 我要摇出一个妹子

该题有题解

时间限制:1000MS  内存限制:65535K
提交次数:184 通过次数:27

题型: 编程题   语言: C++;C

Description

    微信里面的有一个功能叫摇一摇,听说能摇到妹纸。顿时,众教主们看到都马上拿起手机摇啊摇,希望能摇到一个附近的妹纸@_@。不过由于现在微信摇出来的妹纸的距离
具有不确定性,因此白教主想用更好的办法。就是用程序手机了方圆几十公里的所有有玩微信摇一摇的妹纸的坐标,然后尝试找出离自己最近的一个妹纸,并约之@_@ 但是
由于没看到过妹纸本人,因此想先了解一下,所以想找一个离自己至少有一定距离的妹纸,防止过近。
    但由于白教主手头上面的工作很多,他只做了手机妹纸坐标这一步,剩下的想拜托你来完成。希望你能帮助白教主,如果成功,白教主肯定有赏赐的^_^

    离白教主最近的点就是所有点中,和白教主自己那个点的两点间距离最小,且大过一定距离的那个妹纸。
    如果有很多,那么输出x坐标最小的,如果x坐标一样,则输出x坐标一样的点中y坐标最小的。


    注意,为计算方便,这里的两点间坐标使用哈密顿距离。
    即dis(<x1,y1>,<x2,y2>) = |x1-x2|+|y1-y2|



输入格式

样例输入:
    样例输入有多个,每一个case之间用空行分割
    第一行是白教主当前的坐标,两个整数(x,y)
    第二行是收集到的妹子的个数n,不大于10000
    下面是n个妹子的坐标,每一个一行,两个整数(x,y)
    所有坐标的x和y都不超过1000
    接下来一个数m(m<=10000),表示有m个距离,每一个数表示想找对应的至少mi距离的妹纸
    下面是m个整数,每一个整数表示至少的距离mi (mi不超过1000)


输出格式

样例输出:
    对于每一个case
   输出有m行
   每一行输出一个离白教主坐标最近且大于等于mi的坐标,如果不存在,输出-1



输入样例

   0 0
   6
   1 1
   2 2
   3 4
   4 3
   3 3
   4 4
   5
   1 2 3 4 7


输出样例

1 1
1 1
2 2
2 2
3 4


提示

请用
while(scanf("%d",&x)>0)
{
。。。。。。。。
}
之类的输入来读取多个case



作者

a470086609

思路

题目考查对快排的运用,比较简单。因为要对多个mi(最小距离)进行求解,为了避免多次查找,我们还需要定义距离数组disArray,记录每个距离下最合适的妹子在妹子坐标数组ptArray中存储的位置。当查找距离最小为mi的妹子时,我们找到disArray[mi]的值作为ptArray的下标找出妹子坐标。

代码

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
typedef struct
{
    int x,y;
    int dis;
}point;
int cmp(const void *a,const void *b)
{
     point *c=(point *)a;
     point *d=(point *)b;
     return c->dis!=d->dis?c->dis-d->dis:c->x!=d->x?c->x-d->x:c->y-d->y;//三级排序,主dis排序,次x排序,从y排序
}
int main()
{
    int x,y,girlNum,m,index,index1;
    int n,disArray[1005];
    point ptArray[10005];
    while(scanf("%d%d",&x,&y)>0)
    {
        memset(disArray,-1,sizeof(disArray));
        scanf("%d",&girlNum);
        for(index=0; index<girlNum; index++)
        {
            scanf("%d%d",&ptArray[index].x,&ptArray[index].y);
            ptArray[index].dis=abs(x-ptArray[index].x)+abs(y-ptArray[index].y);
        }
        qsort(ptArray,girlNum,sizeof(ptArray[0]),cmp);
        scanf("%d",&m);
        index=0;index1=0;
        while(index1<girlNum){//从距离0~1000选出相对应距离的第一个妹子(这个距离下应该选择的妹子)。
            if(index<=ptArray[index1].dis)
            {
                disArray[index]=index1;
                index++;
            }
            else index1++;
        }
        while(m--)
        {
            scanf("%d",&n);
            if(n>ptArray[girlNum-1].dis)
            {
                printf("%d\n",-1);
            }
            else
            {
                printf("%d %d\n",ptArray[disArray[n]].x,ptArray[disArray[n]].y);
            }
        }
    }
    return 0;
}


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值