服务中心的最佳位置

Thearticlepresentsanefficientalgorithmforacouriercompanytochoosethebestservicecenterlocationonastreetbyminimizingthetotaldistancetoallregionsusingmedian-basedoptimization.
摘要由CSDN通过智能技术生成
题目描述
一家快递公司希望在一条街道建立新的服务中心。公司统计了该街道中所有区域在地图上
的位置,并希望能够以此为依据为新的服务中心选址:使服务中心
到所有区域的距离的总和最小。
给你一个数组 positions ,其中 positions[i] = [left, right] 表示第 i
个区域在街道上的位置,其中 left
代表区域的左侧的起点, right 表示区域的右侧终点,设择服务中心的位置为 location
* 如果第 i 个区域的右侧起点 right 满足 right < location ,则第 i
个区域到服务中心的距离为 location - right
* 如果第 i 个区域的左侧起点 left 满足 left > location ,则第 i
个区域到服务中心的距离为 left - location
* 如果第 i 个区域的两侧 left, right 满足 left <= location <= right ,则第 i
个区域到服务中心的距离为 0
选择最佳的服务中心的位置为 location
,请返回最佳的服务中心位置到所有区域的距离总和的最小值。
输入描述
先输入区域数组 positions 的长度 n 1 <= n <= 10^5
接下来 n 行每行输入成对的 left right 值,以空格隔开
-10^9 <left <= 10^9
-10^9 <right<= 10^9
输出描述
输出为 location
用例
10
1 2
3 4
3 3
1 5
5 8
4 8
7 10
12 14
14 17
10 20
输出:29
解题思路
这类题一种简单暴力解法,先找到所有区间最左侧值最小值,最右测最大值,然后遍历所有位置到服务列表的距离和,取最小和即可。但这个题还有更优化的地方,就是中位数原理,既在一条直线上的所有点,中点到其他点的距离和一定最小。 那就可以先找到所有区间起点的中位数,再找所有终点的中位数,会得到left->right一个更小的区间,然后遍历这个区间即可,大大缩小范围
public static void main(String[] args) {
		Scanner sc = new Scanner(System.in);
		while(sc.hasNext()) {
			int num = sc.nextInt();
			int nums[][] = new int[num][2];
			for(int i=0; i< num;i++) {
				nums[i][0] = sc.nextInt();
				nums[i][1] = sc.nextInt();
			}
			System.out.println(solution(nums));
		}
	}
	public static int solution(int[][] nums){
		Arrays.sort(nums, (a,b)-> a[0]-b[0]); //所有区间起点排序
		int left = nums[(nums.length-1)/2][0];//根据中位数定理,选择最中间那个,如果是偶数则选择更小那个
		
		Arrays.sort(nums, (a,b)-> a[1]-b[1]); //所有区间中点排序
		int right = nums[(nums.length-1)/2][1];//根据中位数定理,选择最中间那个,如果是偶数则选择更小那个
		int real =Integer.MAX_VALUE;
		// 说明最合理的区间会出现在left ->right之间,遍历即可
		for(int j=left; j<= right;j++) {
			int tmp =0;
			for(int i=0; i< nums.length; i++) {
				if(j > nums[i][1]) {
					tmp = tmp + j -nums[i][1];
				}else if(j < nums[i][0]) {
					tmp = tmp + nums[i][0] -j;
				}
			}
			if(tmp < real) {
				real = tmp;
			}
		}
		return real;
	}

  • 9
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值