问题描述
某校大门外长度为L的马路上有一排树,每两棵相邻的树之间的间隔都是1米。我们可以把马路看成一个数轴,马路的一端在数轴0的位置,另一端在L的位置;数 轴上的每个整数点,即0,1,2,……,L,都种有一棵树。
由于马路上有一些区域要用来建地铁。这些区域用它们在数轴上的起始点和终止点表示。已 知任一区域的起始点和终止点的坐标都是整数,区域之间可能有重合的部分。现在要把这些区域中的树(包括区域端点处的两棵树)移走。你的任务是计算将这些树 都移走后,马路上还有多少棵树。
输入格式
输入文件的第一行有两个整数L(1 <= L <= 10000)和 M(1 <= M <= 100),L代表马路的长度,M代表区域的数目,L和M之间用一个空格隔开。接下来的M行每行包含两个不同的整数,用一个空格隔开,表示一个区域的起始点 和终止点的坐标。
输出格式
输出文件包括一行,这一行只包含一个整数,表示马路上剩余的树的数目。
样例输入
500 3
150 300
100 200
470 471
样例输出
298
校门外的树(暴力)https://blog.csdn.net/weixin_48898946/article/details/123538568
区间合并算法:
1.将给出的区间按左端点从小到大进行排序。
2.从左向右遍历区间,开始假设第一个区间[L,R]为基准
(1)当 Li < R 时,取 R = max(R , Ri)
(2)当 Li > R 时,此时一个新区间更新完成并保存,再继续更新 [L,R] 为 [ Li , Ri ],直到将区间遍历完毕。
Java 代码:
import java.io.*;
import java.util.*;
public class Main {//区间合并算法
public static void main(String[] args) throws IOException {
BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
String[] split = br.readLine().split(" ");
int sum = Integer.parseInt(split[0]);
int n = Integer.parseInt(split[1]);
List<Segment> list = new ArrayList<>();
for(int i = 0; i < n; i++) {
split = br.readLine().split(" ");
int left = Integer.parseInt(split[0]);
int right = Integer.parseInt(split[1]);
list.add(new Segment(left, right));
}
Collections.sort(list);
int ans = 0;
int l = list.get(0).left;//l,r为基准区间
int r = list.get(0).right;
for(int i = 1; i < n; i++) {
if(list.get(i).left < r) {//有交集,可以合并
r = Math.max(r, list.get(i).right);
}else {//无交集,需要更新基准
ans += r - l + 1;
l = list.get(i).left;
r = list.get(i).right;
}
}
ans += r - l + 1;
System.out.println(sum + 1 - ans);
}
}
class Segment implements Comparable<Segment>{//用于将区间排序
int left;
int right;
public Segment(int left, int right) {
super();
this.left = left;
this.right = right;
}
@Override
public int compareTo(Segment o) {
return this.left - o.left;
}
}