目录

1 问题描述

2 解决方案

 


1 问题描述

问题描述
  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。

2 解决方案

本题主要考查DFS搜索和剪枝,具体解释请看注释。

具体代码如下:

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.蓝桥杯历届试题 邮局(DFS)

算法笔记_178:历届试题 邮局(Java)的更多相关文章

  1. 算法笔记_197:历届试题 带分数(Java)

    目录 1 问题描述 2 解决方案   1 问题描述 问题描述 100 可以表示为带分数的形式:100 = 3 + 69258 / 714. 还可以表示为:100 = 82 + 3546 / 197. ...

  2. 算法笔记_177:历届试题 城市建设(Java)

    目录 1 问题描述 2 解决方案   1 问题描述 问题描述 栋栋居住在一个繁华的C市中,然而,这个城市的道路大都年久失修.市长准备重新修一些路以方便市民,于是找到了栋栋,希望栋栋能帮助他. C市中有 ...

  3. 算法笔记_189:历届试题 横向打印二叉树(Java)

    目录 1 问题描述 2 解决方案   1 问题描述 问题描述 二叉树可以用于排序.其原理很简单:对于一个排序二叉树添加新节点时,先与根节点比较,若小则交给左子树继续处理,否则交给右子树. 当遇到空子树 ...

  4. 算法笔记_186:历届试题 高僧斗法(Java)

    目录 1 问题描述 2 解决方案   1 问题描述 问题描述 古时丧葬活动中经常请高僧做法事.仪式结束后,有时会有“高僧斗法”的趣味节目,以舒缓压抑的气氛. 节目大略步骤为:先用粮食(一般是稻米)在地 ...

  5. 算法笔记_184:历届试题 约数倍数选卡片(Java)

    目录 1 问题描述 2 解决方案   1 问题描述 问题描述 闲暇时,福尔摩斯和华生玩一个游戏: 在N张卡片上写有N个整数.两人轮流拿走一张卡片.要求下一个人拿的数字一定是前一个人拿的数字的约数或倍数 ...

  6. 算法笔记_183:历届试题 九宫重排(Java)

    目录 1 问题描述 2 解决方案   1 问题描述 问题描述 如下面第一个图的九宫格中,放着 1~8 的数字卡片,还有一个格子空着.与空格子相邻的格子中的卡片可以移动到空格中.经过若干次移动,可以形成 ...

  7. 算法笔记_176:历届试题 最大子阵(Java)

    目录 1 问题描述 2 解决方案   1 问题描述 问题描述 给定一个n*m的矩阵A,求A中的一个非空子矩阵,使这个子矩阵中的元素和最大. 其中,A的子矩阵指在A中行和列均连续的一块. 输入格式 输入 ...

  8. 算法笔记_174:历届试题 地宫取宝(Java)

    目录 1 问题描述 2 解决方案   1 问题描述 问题描述 X 国王有一个地宫宝库.是 n x m 个格子的矩阵.每个格子放一件宝贝.每个宝贝贴着价值标签. 地宫的入口在左上角,出口在右下角. 小明 ...

  9. 算法笔记_172:历届试题 波动数列(Java)

    目录 1 问题描述 2 解决方案   1 问题描述 问题描述 观察这个数列: 1 3 0 2 -1 1 -2 ... 这个数列中后一项总是比前一项增加2或者减少3. 栋栋对这种数列很好奇,他想知道长度 ...

随机推荐

  1. bio、nio、aio及select、poll、epoll

    BIO与NIO.AIO的区别http://blog.csdn.net/skiof007/article/details/52873421 select.poll.epoll之间的区别总结http:// ...

  2. 如何在ubuntu安装phpstorm

    第一步:使用组合键ctrl+alt+t 打开Terminal,cd /home/xxx(当前登录用户名)/downloads(下载目录) 第二步:下载 phpstorm 附截止发文最新版本链接:htt ...

  3. HDU 1890 Robotic Sort (splay tree)

    Robotic Sort Time Limit: 6000/2000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Tota ...

  4. oracle存储过程获取异常信息码和异常信息

    oracle存储过程,可以通过sqlcode 获取异常编码.通过sqlerrm获取异常信息. 例子: create or replace procedure write2blob(p_id in nu ...

  5. systemtap 用户态调试3

    [root@localhost ~]# cat test.c #include <stdio.h> int main( void) { int a=0; a=fun(10,20); pri ...

  6. 手动为Android 4.x 手机加入�自己的根证书(CA 证书)

    首先看Android 4.x 系统的证书存放位置: AOSP Android系统中CA证书文件的位置在:/ system/etc/security/cacerts/一系列的以数字命名的.0文件 方法一 ...

  7. TSC条码打印机C#例程(tsclib.dll调用)

    TSC条码打印机C#例程(tsclib.dll调用) //----  program.cs using System;using System.Collections.Generic;using Sy ...

  8. android中反射机制

    本文介绍Android反射机制实现与原理,在介绍之前,要和Java进行比较,所以先看下Java中的反射相关知识: 一.反射的概念及在Java中的类反射 反射主要是指程序可以访问.检测和修改它本身状态或 ...

  9. quartz 中JobExecutionContext的使用

    假如execute方法中需要一些额外的数据怎么办?比如说execute 中希望发送一封邮件,但是我需要知道邮件的发送者.接收者等信息? 存在两种解决方案: 1.JobDataMap类:    每个Jo ...

  10. Matlab注释多行和取消多行注释的快捷键

    matlab里注释符号是%,只是单行注释,可是没有多行注释符号,就像C/C++/Java中都有多行注释符号/*  */. 如果利用单行注释的方式手工注释一段程序会很麻烦,matlab软件自带快捷键支持 ...