区间合并【模板】

文章讲述了如何使用C++编程解决洛谷NOIP2005普及组问题P1047,通过区间合并处理重叠部分,计算在移除m个区间后数轴上包含的正整数点总数。
摘要由CSDN通过智能技术生成

  洛谷 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;
}

 

  • 12
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
好的,以下是一个简单的用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; } ``` 这个代码模板的实现思路是:先按照区间起点从小到大排序,然后依次合并相邻的区间,如果相邻两个区间不能合并,则将当前区间加入结果中。最后返回合并后的区间数组。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值