区间合并——模板题

题目描述

给定 n 个区间 [li, ri],要求合并所有有交集的区间。注意如果在端点处相交,也算有交集。
输出合并完成后的区间个数。
例如:[1, 3] 和 [2, 6] 可以合并为一个区间 [1, 6]。

输入格式

第一行包含整数 n 。
接下来 n 行,每行包含两个整数 l 和 r。第 i 行的两个数据表示 li, ri。

输出格式

共一行,包含一个整数,表示合并区间完成后的区间个数。

数据范围

1≤n≤100,000

−10^9≤li≤ri≤10^9

输入样例

5
1 2
2 4
5 6
7 8
7 9

输出样例

3

注释版代码

//http://47.110.135.197/problem.php?id=5240
#include<iostream>
#include<algorithm>
#include<vector>
using namespace std;
typedef pair<int,int> PII;
vector<PII> segs;
vector<PII> res;//res用于存放合并完的区间
void merge(vector<PII> &segs)
{
	sort(segs.begin(),segs.end());//先对区间进行排序,pair排序是按照左断点先排序,再按照右端点排序
	int st=-2e9,ed=-2e9;//将st和ed定义为极限小,因为题目的数据范围是10^9,所以定义极限小可以定义2e9
	for(auto seg:segs)
	{
		//对于两区间之间的关系有两种情况
		//①前面区间与后面区间没有交集:那么没有交集就说明前面区间已经不能与后面区间合并
		//那么前面的区间就已经不能再合并了,可以放入结果集了
		if(ed<seg.first)//这样定义ed=-2e9就可以保证第一个有效区间能进行操作
		{
			if(st!=-2e9)//只要他不是我们取的无限小,就可以放入结果集了
			{
				res.push_back({st,ed});
			}
			st=seg.first,ed=seg.second;//然后更新st为后面区间的l和r
		}
		//②前面区间与后面区间有交集:那么我们只需要把ed更新为前面区间和后面区间相比较大的右端点就可以了
		else ed=max(ed,seg.second);
	}
	if(st!=-2e9) res.push_back({st,ed});//如果只有一个区间,我们就需要用到这个步骤
}
int main()
{
	int n,l,r;
	scanf("%d",&n);
	for(int i=0;i<n;i++)
	{
		scanf("%d %d",&l,&r);
		segs.push_back({l,r});//将每一个lr代表的区间存入segs里面
	}
	merge(segs);//对segs区间进行合并操作
	printf("%d",res.size());//输出合并完的区间个数
	return 0;
}
好的,以下是一个简单的用C语言实现区间合并的代码模板: ```c #include <stdio.h> // 定义区间结构体 typedef struct interval { int start; int end; } Interval; // 区间合并函数 Interval* merge_intervals(Interval* intervals, int size, int* returnSize) { if (size <= 1) { *returnSize = size; return intervals; } // 将区间按照起点从小到大排序 for (int i = 0; i < size; i++) { for (int j = i + 1; j < size; j++) { if (intervals[i].start > intervals[j].start) { Interval temp = intervals[i]; intervals[i] = intervals[j]; intervals[j] = temp; } } } // 合并区间 Interval* result = (Interval*)malloc(sizeof(Interval) * size); int index = 0; result[0] = intervals[0]; for (int i = 1; i < size; i++) { if (result[index].end >= intervals[i].start) { result[index].end = fmax(result[index].end, intervals[i].end); } else { index++; result[index] = intervals[i]; } } *returnSize = index + 1; return result; } // 测试 int main() { Interval intervals[] = {{1, 3}, {2, 6}, {8, 10}, {15, 18}}; int size = sizeof(intervals) / sizeof(intervals[0]); int returnSize; Interval* result = merge_intervals(intervals, size, &returnSize); printf("["); for (int i = 0; i < returnSize; i++) { printf("[%d, %d]", result[i].start, result[i].end); if (i != returnSize - 1) { printf(", "); } } printf("]\n"); free(result); return 0; } ``` 这个代码模板的实现思路是:先按照区间起点从小到大排序,然后依次合并相邻的区间,如果相邻两个区间不能合并,则将当前区间加入结果中。最后返回合并后的区间数组。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值