区间合并
题目
给定 n 个区间 [li,ri],要求合并所有有交集的区间。
注意如果在端点处相交,也算有交集。
输出合并完成后的区间个数。
例如:[1,3]和 [2,6]可以合并为一个区间 [1,6]。
输入格式
第一行包含整数 n。
接下来 n 行,每行包含两个整数 l 和 r。
输出格式
共一行,包含一个整数,表示合并区间完成后的区间个数。
数据范围
1≤n≤100000,
−109≤li≤ri≤109
输入样例:
5
1 2
2 4
5 6
7 8
7 9
输出样例:
3
思路
AC代码
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<vector>
using namespace std;
typedef pair<int,int> PII;
const int N=100010;
int n;
vector<PII> segs;
void merge(vector<PII> &segs)
{
vector<PII> res;
//pair排序在C++里会优先以左端点排序,然后再以右端点排序
sort(segs.begin(),segs.end());
//从前往后扫描,扫描的过程中维护当前区间
int st=-2e9,ed=-2e9;//设边界值
for(auto seg:segs)//从前往后扫描所有线段
{//每次分两种情况看
if(ed<seg.first)//维护的区间严格在枚举的区间的左边,没有任何交集
{
if(st!=-2e9)//不能是初始区间
{
res.push_back({st,ed});
}
st=seg.first,ed=seg.second;
}
else//当前区间和维护的区间存在交集
{
ed=max(ed,seg.second);//右端点更新成维护的区间的右端点以及当前区间的右端点的最大值
}
}
if(st!=-2e9)//防止空区间
{
res.push_back({st,ed});//最后的区间加到答案里
}
segs=res;//把区间更新成res
}
int main()
{
cin>>n;
for(int i=0;i<n;i++)
{
int l,r;
cin>>l>>r;//读入每个区间的左右端点
segs.push_back({l,r});
}
merge(segs);//区间合并
cout<<segs.size()<<endl;
return 0;
}