洛谷 P1047 [NOIP2005 普及组] 校门外的树
题意:给定一个长度为l, 再给m个区间,每个区间可能会重叠;0是起点,l是数轴终点;求将m个区间去掉后(包括两个端点),有多少个正整数点。
思路:将每个重叠的区间合并为一个,最后用每段不重叠的区间的最末端点减去开始端点,再加1是这段区间包括的所有正整数点,结果就是用l+1减去每个区间包括的所有正整数点。
#include<iostream>
#include<vector>
#include<algorithm>
using namespace std;
typedef pair<int,int>pII; //将数对pair<int,int>命名为pII
vector<pII> segs; //创建一个vector数组 segs 其中存放的类型为pII
void merge(vector<pII>&segs){ //区间合并模板
vector<pII> res; //用来存放合并后的区间
sort(segs.begin(), segs.end()); //先将每个区间按从小到大排序
int st = -1e9, ed = -1e9;//边界 负无穷
for(vector<pII>::iterator it = segs.begin(); it != segs.end(); it++){ //遍历整个vector数组
if(ed < it->first){
if(st != -1e9) res.push_back({st,ed}); //情况1,即两个区间没有重叠部分,将前一个区间放入结果数组
st = it->first; //更新每次比较的端点
ed = it->second;
}else{
ed = max(ed,it->second); //情况2,即两个区间有重叠部分,令两个区间中末尾靠后的端点为新区间的末尾端点
}
}
if(st != -1e9){ //将最后一个区间放入结果数组
res.push_back({st,ed});
}
segs = res;
}
int main(){
int l , m;
cin >> l >> m;
for(int i = 0; i < m; i++){
int l,r;
cin >> l >> r;
segs.push_back({l,r});
}
merge(segs);
int num = l +1;
for(vector<pII>::iterator it = segs.begin(); it != segs.end(); ++it){
num -= (it->second - it->first +1);
}
cout << num << endl;
return 0;
}