5407.管道 (二分+区间合并)

本题链接:5407. 管道 - AcWing题库

题目:

样例:

|-------------------|
| 3 10 1 1 6 5 10 2 |
[输入]

|---|
| 5 |
[输出]

思路:

根据题目意思,给出 n 个阀门,其中 管道有 len 段,随后 n 个阀门对应的位置在 L 点,并且当 S 时刻阀门的水会放开, 其中 放开后 水在 )时刻会使得从第 − ( ) 段到第 + ( ) 段的传感器检测到水流。

问输出 全部段点感应到水流的 最早时间,这里 有可能出现同一时刻 水阀放水的过程,以及放水后 感应到的区域 部分同时感应,所以我们应该联想到 区间合并 ,判断区间是否覆盖完我们的全部管道即可,这里我们 枚举 时间,又因为数据范围较大,枚举时间肯定会 TLE,所以我们又应该联想到二分枚举时间,即可得到最优答案。

在这里需要注意的点是,第 − ( ) 段到第 + ( ) 段的传感器检测到水流ta它的区间应该为:

L = − ( ) - 1 R = + ( )

这里 L 应该为 − ( ) - 1 ,是因为我们感应点应该包括 第 − ( ) 的点,所以我们区间的左端点 应该 - 1。

区间和并模板代码:

cpp 复制代码
inline void Merge(vector<PII>&p)
{
	int sz = p.size();
	if(sz <= 1) return ; // 如果拥有的区间 <= 1 说明没有合并需要
	
	sort(All(p));	// 先根据端点排序好需要合并的区间
	
	vector<PII>tem;	// 临时存储合并后的区间
	
	PII now = *p.begin();	// 先拿出第一个区间
	
	for(int i = 1;i < sz;++i)
	{
		PII next = p[i];	// 拿出下一个区间,进行比较
		
		if(now.y >= next.y) continue;	// 如果下一个区间被 目前的区间包含,那么合并起来
		
		if(now.y >= next.x) now.y = next.y;	// 如果出现 目前的右端点大于下一个的左端点,合并更新现在的右端点
		else
		{
			tem.emplace_back(now);	// 存储好目前的区间
			now = next;	// 更新目前的区间
		}
	}
	tem.emplace_back(now);	// 存储好最后的区间
	
	p = tem;	// 更新合并好的区间
}

代码详解如下:

cpp 复制代码
#include <bits/stdc++.h>
#define int long long
#define endl '\n'
#define x first
#define y second
#define All(x) x.begin(),x.end()
#define IOS ios::sync_with_stdio(false),cin.tie(0),cout.tie(0)
using namespace std;
using PII = pair<int,int>;

int n,len;

vector<PII>v;// 存储阀门所在点以及开阀时刻


inline void Merge(vector<PII>&p)
{
	int sz = p.size();
	if(sz <= 1) return ; // 如果拥有的区间 <= 1 说明没有合并需要
	
	sort(All(p));	// 先根据端点排序好需要合并的区间
	
	vector<PII>tem;	// 临时存储合并后的区间
	
	PII now = *p.begin();	// 先拿出第一个区间
	
	for(int i = 1;i < sz;++i)
	{
		PII next = p[i];	// 拿出下一个区间,进行比较
		
		if(now.y >= next.y) continue;	// 如果下一个区间被 目前的区间包含,那么合并起来
		
		if(now.y >= next.x) now.y = next.y;	// 如果出现 目前的右端点大于下一个的左端点,合并更新现在的右端点
		else
		{
			tem.emplace_back(now);	// 存储好目前的区间
			now = next;	// 更新目前的区间
		}
	}
	tem.emplace_back(now);	// 存储好最后的区间
	
	p = tem;	// 更新合并好的区间
}

// 二分性质
inline bool cheak(int t)
{
	// 开始获取 t 时刻以内的水流感应到的区间
	vector<PII>edge;
	for(int i = 0;i < n;++i)
	{
		PII now = v[i];
		if(now.y > t)break;
		
		int time = t - now.y;
		
		int l = now.x - time - 1;	// 注意这里的左端点需要 -1
		int r = now.x + time;
		
		edge.emplace_back(PII(l,r));
	}	
	
	// 开始合并区间
	Merge(edge);
	
	if(edge.size() != 1) return false;	// 如果区间不唯一,说明出现未覆盖感应到的管道点。
	else
	{
		PII have = *edge.begin();	// 拿出覆盖的区间,比较是否覆盖感应完了所有管道
		
		if(have.x <= 0 && have.y >= len) return true;
		else return false;
	}
}

inline void solve()
{
	int l = 0,r = -1;	// 二分需要的左右端点,小优化
	v.clear();
	cin >> n >> len;
	for(int i = 0;i < n;++i)
	{
		int L,S;
		cin >> L >> S;
		r = max(S,r);	// 获取最大的开阀时刻
		v.emplace_back(PII(L,S));
	}
	
	// 排序开阀时刻为从小到大
	sort(All(v),[](const PII&a,const PII&b)
	{
		return a.y < b.y;
	});	
	
	r <<= 1;	// 这里 最大时刻开两倍
	
	// 开始 二分 枚举 时间点
	while(l < r)
	{
		int mid = l + r >> 1;
		if(cheak(mid)) r = mid;
		else l = mid + 1;
	}
	cout << l << endl;
}

signed main()
{
//	freopen("myout.txt","w+",stdout);
//	freopen("in.txt","r",stdin);
	int _t = 1;
	IOS;
//	cin >> _t;
	while(_t--)
	{
		solve();
	}
	return 0;
}

最后提交:

相关推荐
SteveSenna11 分钟前
Trossen Arm MuJoCo自定义1:改变目标物体
人工智能·学习·算法·机器人
yong999033 分钟前
IHAOAVOA:天鹰优化算法与非洲秃鹫优化算法的混合算法(Matlab实现)
开发语言·算法·matlab
米粒12 小时前
力扣算法刷题 Day 42(股票问题总结)
算法·leetcode·职场和发展
浅念-4 小时前
从LeetCode入门位运算:常见技巧与实战题目全解析
数据结构·数据库·c++·笔记·算法·leetcode·牛客
CoovallyAIHub4 小时前
无人机拍叶片→AI找缺陷:CEA-DETR改进RT-DETR做风电叶片表面缺陷检测,mAP50达89.4%
算法·架构·github
CoovallyAIHub4 小时前
混合训练反而更差?VLM Agent在训练前协调跨数据集标注,文档布局检测F-score从0.860提升至0.883
算法·架构·github
鸿途优学-UU教育4 小时前
教材质量——法考培训的根基与底气
算法
_深海凉_4 小时前
LeetCode热题100-最大数(179)
算法·leetcode·职场和发展
剑挑星河月4 小时前
763.划分字母区间
数据结构·算法·leetcode
programhelp_5 小时前
Snowflake OA 2026 面经|3道高频真题拆解 + 速通攻略
经验分享·算法·面试·职场和发展