问题描述

  C村住着n户村民,由于交通闭塞,C村的村民只能通过信件与外界交流。为了方便村民们发信,C村打算在C村建设k个邮局,这样每户村民可以去离自己家最近的邮局发信。

现在给出了m个备选的邮局,请从中选出k个来,使得村民到自己家最近的邮局的距离和最小。其中两点之间的距离定义为两点之间的直线距离。

输入格式

  输入的第一行包含三个整数n, m, k,分别表示村民的户数、备选的邮局数和要建的邮局数。

  接下来n行,每行两个整数x, y,依次表示每户村民家的坐标。

  接下来m行,每行包含两个整数x, y,依次表示每个备选邮局的坐标。

  在输入中,村民和村民、村民和邮局、邮局和邮局的坐标可能相同,但你应把它们看成不同的村民或邮局。

输出格式

  输出一行,包含k个整数,从小到大依次表示你选择的备选邮局编号。(备选邮局按输入顺序由1到m编号)

样例输入

5 4 2

0 0

2 0

3 1

3 3

1 1

0 1

1 0

2 1

3 2

样例输出

2 4

数据规模和约定

  对于30%的数据,1<=n<=10,1<=m<=10,1<=k<=5;

  对于60%的数据,1<=m<=20;

  对于100%的数据,1<=n<=50,1<=m<=25,1<=k<=10。

import java.util.Scanner;

public class Main {
public static int n, m, k;
public static double[][] distance; //distanc[i][j]表示第i个村庄到第j个邮局的直线距离
public static int[] answer; //存放最终选中的k个邮局编号
public static double minSum; //表示选中k个邮局的最小距离和 public void init() {
distance = new double[n + 1][m + 1];
answer = new int[k + 1];
minSum = Double.MAX_VALUE;
} /*
* 参数count:表示已经选中的邮局个数
* 参数current:表示DFS目前遍历到的邮局编号
* 参数minDis:minDis[i]表示村庄i到已选中邮局中的最短距离
* 参数tempAns:存放目前遍历过程中已经选中的邮局编号
*/
public void dfs(int count, int current, double[] minDis, int[] tempAns) {
if(count == k) { //当已经选中k个邮局时
double tempSum = 0;
for(int i = 1;i <= n;i++)
tempSum += minDis[i];
if(tempSum < minSum) {
minSum = tempSum;
for(int i = 1;i <= k;i++)
answer[i] = tempAns[i];
}
return;
} else if(k - count <= m - current && current <= m) { //表示在剩下未选中的邮局个数多于还需要选中的邮局个数
boolean judge = false; //用于判断当前选中的邮局current+1是否符合要求
double[] minDis1 = new double[n + 1]; //用于保存当前minDis数组中值
for(int i = 1;i <= n;i++)
minDis1[i] = minDis[i];
for(int i = 1;i <= n;i++) {
if(minDis[i] > distance[i][current + 1]) {
minDis[i] = distance[i][current + 1];
judge = true;
}
}
tempAns[count + 1] = current + 1;
if(judge == true) //邮局current + 1符合要求,选中个数加1继续搜索
dfs(count + 1, current + 1, minDis, tempAns);
dfs(count, current + 1, minDis1, tempAns); //直接舍弃邮局current+1,进行下一次搜索
}
} public static void main(String[] args) {
Main test = new Main();
Scanner in = new Scanner(System.in);
n = in.nextInt();
m = in.nextInt();
k = in.nextInt();
test.init();
double[][] point = new double[n + 1][2]; //存放n个村庄的位置坐标
for(int i = 1;i <= n;i++) {
point[i][0] = in.nextDouble(); //村庄横坐标
point[i][1] = in.nextDouble(); //村庄纵坐标
}
for(int j = 1;j <= m;j++) {
double x = in.nextDouble(); //邮局横坐标
double y = in.nextDouble(); //邮局纵坐标
for(int i = 0;i <= n;i++) //计算1~n村庄到邮局j的直线距离
distance[i][j] = Math.sqrt((point[i][0]-x)*(point[i][0]-x) +
(point[i][1]-y)*(point[i][1]-y));
}
int[] tempAns = new int[k + 1];
double[] minDis = new double[n + 1]; //minDis[i]表示第i个村庄到已选中的邮局中的最小距离
for(int i = 1;i <= n;i++)
minDis[i] = Double.MAX_VALUE;//初始化第i个村庄到选中邮局中最小距离为最大值 test.dfs(0, 0, minDis, tempAns);
for(int i = 1;i <= k;i++)
System.out.print(answer[i]+" ");
}
}

最新文章

  1. Hibernate5.2关联关系之单向多对一(二)
  2. Linux GDB Debugging
  3. python运算符和表达式
  4. viewport 详解
  5. CentOS下一键安装Openstack
  6. Uva 725 除法
  7. uva 11324 The Largest Clique
  8. C#无边框窗体移动 将事件绑定到想实现的控件上
  9. const的用法,特别是用在函数前面与后面的区别!
  10. ListView展示不同布局需要注意的地方
  11. 用系统默认mail服务实现邮件发送
  12. 一起学习造轮子(一):从零开始写一个符合Promises/A+规范的promise
  13. 不失一般性和快捷性地判定决策单调(洛谷P1912 [NOI2009]诗人小G)(动态规划,决策单调性,单调队列)
  14. 装了as之后提示NO JVM installation found.....
  15. git版本超前了N个版本且落后了N个版本的解决办法
  16. 大数据入门第二十三天——SparkSQL(二)结合hive
  17. [转载]ValidationExpression验证规则
  18. [转][Chrome]浏览器粘贴行为
  19. 造一个轮子然后安装到pypi上
  20. window7下 cmd命令行 Mysql导出表结构 + 表数据

热门文章

  1. Java元注解@Retention规则
  2. 初识spring boot maven管理--使用spring-boot-starter-parent
  3. Azure B2C登录,react-web端实现,自定义登录页面ui
  4. sublime text 3 关联鼠标右击
  5. 案例(一) 利用机器算法RFM模型做用户价值分析
  6. Vue element-ui date-picker 结束时间大于开始时间且开始时间小于此刻(转)
  7. linux常用命令---中英文设置
  8. day05:数组与字典常识(20170217)
  9. 使用PHP得到所有的HTTP请求头
  10. 从无到有Springboot整合Spring-data-jpa实现简单应用