zoukankan      html  css  js  c++  java
  • Java实现 蓝桥杯 历届试题 邮局

    问题描述
      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]+" ");
        }
    }
    
  • 相关阅读:
    @字节跳动8年老Android面试官谈;Context都没弄明白凭什么拿高薪?
    @阿里面试官:Android面试这些原理都给我讲明白了,最低都是20k起步!
    @以后面试官再问你三次握手和四次挥手,直接把这一篇文章丢给他
    @备战2020年金三银四,看这一篇面试文章就够了(合适各级Java人员)
    字节跳动面试,第三面挂了,这原因我服了!
    太可惜了,四面字节跳动,我的offer竟被一道“算法题”给拦截了
    @java2019面试题北京
    @2019.07 Android 面试真题集锦
    2018 Java线程热门面试题,你知道多少?
    阿里大厂的148道核心面试题,(程序员必备学习方向)offer收割机 全会月薪50k不难
  • 原文地址:https://www.cnblogs.com/a1439775520/p/13077709.html
Copyright © 2011-2022 走看看