导弹拦截(Java实现)

经过11年的韬光养晦,某国研发出了一种新的导弹拦截系统,凡是与它的距离不超过其工作半径的导弹都能够被它成功拦截。当工作半径为0时,则能够拦截与它位置恰好相同的导弹。但该导弹拦截系统也存在这样的缺陷:每套系统每天只能设定一次工作半径。而当天的使用代价,就是所有系统工作半径的平方和。

某天,雷达捕捉到敌国的导弹来袭。由于该系统尚处于试验阶段,所以只有两套系统投入工作。如果现在的要求是拦截所有的导弹,请计算这一天的最小使用代价。

输入描述:

 

第一行包含4个整数x1、y1、x2、y2,每两个整数之间用一个空格隔开,表示这两套导弹拦截系统的坐标分别为(x1, y1)、(x2, y2)。

第二行包含1个整数N,表示有N颗导弹。接下来N行,每行两个整数x、y,中间用一个空格隔开,表示一颗导弹的坐标(x, y)。不同导弹的坐标可能相同。

输出描述:

输出只有一行,包含一个整数,即当天的最小使用代价。

示例1

输入

复制0 0 10 0 2 -3 3 10 0

0 0 10 0
2
-3 3
10 0

输出

复制18

18

说明

要拦截所有导弹,在满足最小使用代价的前提下,两套系统工作半径的平方分别为18和0。

示例2

输入

复制0 0 6 0 5 -4 -2 -2 3 4 0 6 -2 9 1

0 0 6 0 
5 
-4 -2 
-2 3 
4 0 
6 -2 
9 1

输出

复制30

30

说明

 

样例中的导弹拦截系统和导弹所在的位置如下图所示。要拦截所有导弹,在满足最小使用代价的前提下,两套系统工作半径的平方分别为20和10。

备注:

 

两个点(x1, y1)、(x2, y2)之间距离的平方是(x1−x2)2+(y1−y2)2。

两套系统工作半径r1、r2的平方和,是指r1、r2分别取平方后再求和,即r12+r22。

对于10%的数据,N=1

对于20%的数据,1≤N≤2

对于40%的数据,1≤N≤100

对于70%的数据,1≤N≤1000

对于100%的数据,1≤N≤100000,且所有坐标分量的绝对值都不超过1000。

 对于每个导弹都有两种情况,他们要么在一导弹防御系统内,要么在二导弹防御系统内。

对:

导弹一  0导弹二 0
r1r2
r1r2
r1r2
r1r2

此时已经按r1做降序排序:

错误思路:每次找r1和r2的最小值,然后找挑选过之后整个r1组内和r2组内的最大值,这样并不每次都是全优解,举个例子:

r1 =500r2=300 
r1=700r2=800
r1=100r2=200

按照错误思路,应该为1000,可是选择r1 700 r2 0 也能满足此题解,所以应该按照思路2,将每个导弹第一次归为一系统,后面将其一直归为二系统,如果能使系统半径和减小,说明他归为二系统更好,否则说明他在一系统更好。

思路解析:

导弹一  0导弹二 0

这个为初始状态,没有导弹,然后将r1归为一系统,然后不断加入导弹,当前导弹归为一系统,已经考虑过的系统归给二系统,每次找r2的最大值,然后和当前r1相加,得到此时的局部最优解,然后所有局部最优解求最优得到全局最优解。

代码:

// 不易理解思路,请结合代码和debug和草稿纸推算

import java.util.*;
public class Main{
    public static void main(String[] args){
        Scanner input = new Scanner(System.in);
        int x1 = input.nextInt();
        int y1 = input.nextInt();
        int x2 = input.nextInt();
        int y2 = input.nextInt();
        Bomb[] array = new Bomb[100005];
        int n = input.nextInt();
        for(int i = 1; i <= n; i++){
            int x = input.nextInt();
            int y = input.nextInt();
            array[i] = new Bomb(x,y);
            array[i].r1 = (int) (Math.pow(x - x1, 2) + Math.pow(y - y1, 2));
            array[i].r2 = (int) (Math.pow(x - x2, 2) + Math.pow(y - y2, 2));
        }
        Arrays.sort(array, 1, n+1, new BombComparator()); // 将array以r1降序的方式进行排序
        int ars = 0;
        int ans = Integer.MAX_VALUE;
        array[0] = new Bomb(0,0);
        for (int i = 1; i <= n; i++) {
            ars = Math.max(ars, array[i-1].r2); // 寻找r2的最大值
            ans = Math.min(ans, array[i].r1 + ars); // 寻找r1 + r2 的最小值
            
        }
        System.out.println(ans);
    }

    public static class Bomb{
        int x;
        int y;
        int r1;
        int r2;

        public Bomb(int x, int y) {
            this.x = x;
            this.y = y;
        }

    }
    public static class BombComparator implements Comparator<Bomb>{
        @Override
        public int compare(Bomb o1, Bomb o2) {
            return o2.r1 - o1.r1;
        }
    }
}

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

Studying~

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

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

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

打赏作者

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

抵扣说明:

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

余额充值