CCF CSP 2020-09-1 称检测点查询 解题思路及满分代码(C++11)

本文介绍了如何利用C++ STL中的sort函数解决一个实际问题——根据市民位置找出最近的三个核酸检测点。题目要求在考虑距离的同时,确保在距离相等情况下编号较小的点优先。解题思路涉及结构体定义、排序逻辑和完整代码实现,适用于初学者掌握排序和结构体的应用。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

问题描述

题目背景

2020 年 6 月 8 日,国务院联防联控机制发布《关于加快推进新冠病毒核酸检测的实施意见》,提出对“密切接触者”等八类重点人群“应检尽检”,其他人群“愿检尽检”。

问题描述

某市设有 n 个核酸检测点,编号从 1 到 n,其中 i 号检测点的位置可以表示为一个平面整数坐标 ( x i , y i ) (x_i,y_i) (xi,yi)

为方便预约核酸检测,请根据市民所在位置 ( X , Y ) (X,Y) (X,Y),查询距其最近的三个检测点。
多个检测点距离相同时,编号较小的视为更近。

输入格式

输入共 n+1 行。

第一行包含用空格分隔的三个整数 n、X 和 Y,表示检测点总数和市民所在位置。

第二行到第 n+1 行依次输入 n 个检测点的坐标。第 i+1 行 ( 1 ≤ i ≤ n ) (1≤i≤n) 1in包含用空格分隔的两个整数 x i x_i xi y i y_i yi,表示 i 号检测点所在位置。

输出格式

输出共三行,按距离从近到远,依次输出距离该市民最近的三个检测点编号。

样例输入1

3 2 2
2 2
2 3
2 4

样例输出1

1
2
3

样例输入2

5 0 1
-1 0
0 0
1 0
0 2
-1 2

样例输出2

2
4
1

样例2解释
在这里插入图片描述
评测用例规模与约定

全部的测试点满足, 3 ≤ n ≤ 200 3≤n≤200 3n200,所有坐标均为整数且绝对值不超过 1000。

提示

市民到第 i 号检测点的距离 Di 可由如下公式算出:

D i = ( X − x i ) 2 + ( Y − y i ) 2 D_i = (X-x_i)^2 + (Y-y_i)^2 Di=(Xxi)2+(Yyi)2

解题思路

题目本质上就是一个简单的排序题,只要会熟练运用STL中的sort排序就可以很快解出。

但有一个要点需要注意:各个检测点是有编号的,在距离相同的情况下,编号靠前的要排在前面。

下面是解题步骤:

  1. 定义一个结构体来存储检测点的坐标、距离、编号信息,然后所有坐标存储在一个结构体数组内。
typedef struct point{
	int num;
	long x, y;
	long d;
}point;
point p[210];

这里距离d我直接用长整型来表示,计算的时候就不开平方了,如果要开平方用double就行。

  1. 编写sort排序要用到的cmp函数,定义一下对结构体排序的逻辑
//优先按距离值递增排序,距离相等时编号小的排前面
bool cmp(point a, point b) {
	if(a.d != b.d) {
		return a.d < b.d;
	}else return a.num < b.num;
}
  1. 最后只需要排序,输出前三个坐标的编号即可。

完整的满分代码贴在最后

满分代码

#include<cstdio>
#include<algorithm>
using namespace std;
typedef struct point{
	int num;
	long x, y;
	long d;
}point;
point p[210];

bool cmp(point a, point b) {
	if(a.d != b.d) {
		return a.d < b.d;
	}else return a.num < b.num;
}

int main() {
	int n;
	long X, Y;
	scanf("%d %ld %ld", &n, &X, &Y);
	for(int i=0; i<n; i++) {
		scanf("%ld %ld", &p[i].x, &p[i].y);
		p[i].num = i+1;
	}
	for(int i=0; i<n; i++) {
		long dx = p[i].x-X;
		long dy = p[i].y-Y;
		p[i].d = dx*dx + dy*dy;
	}
	sort(p, p+n, cmp);
	for(int i=0; i<3; i++) {
		printf("%d\n", p[i].num);
	}
	
	return 0;
}

在这里插入图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

BlueMoon2000

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值