题目描述
一家快递公司希望在一条街道建立新的服务中心。公司统计了该街道中所有区域在地图上
的位置,并希望能够以此为依据为新的服务中心选址:使服务中心
到所有区域的距离的总和最小。
给你一个数组
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
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;
}