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

最后提交:

相关推荐
SoraLuna11 分钟前
「Mac玩转仓颉内测版7」入门篇7 - Cangjie控制结构(下)
算法·macos·动态规划·cangjie
我狠狠地刷刷刷刷刷14 分钟前
中文分词模拟器
开发语言·python·算法
鸽鸽程序猿15 分钟前
【算法】【优选算法】前缀和(上)
java·算法·前缀和
九圣残炎21 分钟前
【从零开始的LeetCode-算法】2559. 统计范围内的元音字符串数
java·算法·leetcode
YSRM33 分钟前
Experimental Analysis of Dedicated GPU in Virtual Framework using vGPU 论文分析
算法·gpu算力·vgpu·pci直通
韭菜盖饭1 小时前
LeetCode每日一题3261---统计满足 K 约束的子字符串数量 II
数据结构·算法·leetcode
geng小球1 小时前
LeetCode 78-子集Ⅱ
java·算法·leetcode
AnFany1 小时前
LeetCode【0028】找出字符串中第一个匹配项的下标
python·算法·leetcode·字符串·kmp·字符串匹配
远望清一色1 小时前
基于MATLAB的图片中文字的提取及识别
算法·matlab
weixin_307779131 小时前
证明存在常数c, C > 0,使得在一系列特定条件下,某个特定投资时刻出现的概率与天数的对数成反比
人工智能·算法·机器学习