蓝桥杯算法题:区间移位

题目描述

数轴上有n个闭区间:D1,...,Dn。
其中区间Di用一对整数[ai, bi]来描述,满足ai < bi。
已知这些区间的长度之和至少有10000。
所以,通过适当的移动这些区间,你总可以使得他们的“并”覆盖[0, 10000]——也就是说[0, 10000]这个区间内的每一个点都落于至少一个区间内。
你希望找一个移动方法,使得位移差最大的那个区间的位移量最小。


具体来说,假设你将Di移动到[ai+ci, bi+ci]这个位置。你希望使得maxi{|ci|} 最小。

输入

输入的第一行包含一个整数n,表示区间的数量。
接下来有n行,每行2个整数ai, bi,以一个空格分开,表示区间[ai, bi]。
保证区间的长度之和至少是10000。

输出

输出一个数字,表示答案。如果答案是整数,只输出整数部分。如果答案不是整数,输出时四舍五入保留一位小数。

样例输入

2
10 5010
4980 9980
样例输出 

20

首先说思路,在满足条件的所有值中取最大值,常见的二分问题,而又涉及到区间遍历,可能会有考贪心。这道题,大致看起来,可以按一般贪心的思路,将全部区间按右端点值由小到大排序,然后采用二分,判断mid值是否可以让所有区间通过移位达到连续不断,并且长度大于10000。

①二分:这道题看数据会出现小数,其实最小也就0.5,我看别人的做法是把有关区间以及距离什么的都乘2,这样算出来的mid最小值也就为1,可以用整数二分。但我直接用的实数二分,也是差不多的,可能算的次数会多一点。

②check函数:每次二分,都要判断mid是否满足条件,就要用到check函数。其实我的想法是遍历区间,拿遍历到的某个区间[l,r]跟之前遍历完所有区间形成的右端最大值mr比较,如果l-mid>mr,那说明我往左移也跟前面所有区间形成不了交集,那当然后边的区间也一样,所以直接判断mid不符合要求(当时是这样想的,不过细想其实有错),然后就这样遍历完,如果mr≥10000,不就说明mid可以实现区间移位连续嘛,那就继续二分,直到二分的区间缩成一个值。

但是,最后程序无论怎么改,都只过了九个测试点,挠头

(过九个测试点): 

#include<bits/stdc++.h>
using namespace std;
struct Point{
	int l,r;
	Point(int a,int b){
		l=a,r=b;
	}
	friend bool operator<(const Point& p1,const Point& p2){
		if(p1.r!=p2.r)return p1.r<p2.r;//为什么要优先用r排序?贪心思想, 
		else return p1.l<p2.l;
	}
};
vector<Point>pv;
bool check(double mid){
	cout<<mid<<endl;
	vector<Point>temp(pv);//复制vector来进行模拟
	double mr=0;//表示所有区间通过不同移动能将右端点扩大到的范围 
	for(int i=0;i<temp.size();i++){
		int l=temp[i].l,r=temp[i].r;
		if(mr+1e-6>l-mid&&mr<r+mid+1e-6){
		//mr>=l-mid说明可以通过对区间[l,r]进行左移操作实现与原来的区间有交,也就是使整个区间连续,mr<=r+mid是指通过这个区间[l,r]右移,可以实现mr变大,不然处理[l,r]区间又有什么意义呢 
			if(mr<l){
				mr=r-l+mr;
			}else{
				if(l+mid>mr)mr+=(r-l);
			else mr+=(r-l-(mr-l-mid));
			}
		}
		if(mr<l-mid)return false;
	} 
	return mr+1e-6>10000;//循环正常结束也不一定能行,要区间右端最大值大于等于10^4 
}
int main(){//这道题,典型的大于中找最小,大于是指mid超过某一个值,就可以实现所有区间移动覆盖整个范围,最小是指要找的mid要满足条件且最小,直接套模板 
	int n;
	cin>>n;
	int l,r;
	for(int i=0;i<n;i++){
		cin>>l>>r;
		pv.push_back(Point(l,r));
	}
	sort(pv.begin(),pv.end()); 
double L=0,R=10000,mid;
	while(R-L>1e-6){
		cout<<mid<<" "<<L<<" "<<R<<endl;
		mid=(L+R)/2;//维护左边
		if(check(mid)){
			R=mid;
		}else{
			L=mid;
		}
	}
	if(mid>1e-6)cout<<mid<<endl;
	else cout<<0<<endl;
}

好吧,现在讲讲AC的代码,我当时看别人的题解也好奇,为什么一次遍历的事,他还要加个while循环,这不增加工作量嘛,我绞尽脑汁也想不明白while有什么用处,后边才发现,原来是弥补贪心策略的不足!!!我先举个例子:

对于数据 :

3

1000 2000

800 3000

2200 10000

考虑贪心,区间按右端点值从小到大排吗?这样输出的答案可不对哦,输出的答案会是1000,正确的答案可是800!!!

那要不考虑区间按左端点从大到小排?

对于数据:

3

2 1000

3 8

1002 1000 

按左排输出的答案可就是994了喔!!!所以,我自己也不知道到底要怎样贪心了

可能ac的人也不知道吧,所以他们添了一层循环,为的是保证一个之前没用到的区间,以后还有机会用,这样就加大了正确的可能,毕竟,谁也不知道哪个区间先用上嘛,不同区间用的先后次序不同,可能产生不同结果。

自我安慰:这道题主要考二分,其他不懂的是次要的(好吧,还是自己贪心不行)

所以,要是有更好的策略,可以跟我分享,救救菜狗

(叠甲,这篇博客是在精神不正常状态下写的,语无伦次,看题解可以去那些大佬博客)

 (AC):

#include<bits/stdc++.h>
using namespace std;
struct Point{
	int l,r;
	Point(int a,int b){
		l=a,r=b;
	}
	friend bool operator<(const Point& p1,const Point& p2){
		if(p1.r!=p2.r)return p1.r<p2.r;//为什么要优先用r排序?贪心思想, 
		else return p1.l<p2.l;
	}
};
vector<Point>pv;
bool check(double mid){
	vector<Point>temp(pv);//复制vector来进行模拟
	double mr=0;//表示所有区间通过不同移动能将右端点扩大到的范围 
	while(true){//避免一些特殊区间影响,弥补贪心的不足
		bool flag=false;
		for(int i=0;i<temp.size();i++){
		int l=temp[i].l,r=temp[i].r;
		if(mr+1e-6>l-mid&&mr<r+mid+1e-6){
		//mr>=l-mid说明可以通过对区间[l,r]进行左移操作实现与原来的区间有交,也就是使整个区间连续,mr<=r+mid是指通过这个区间[l,r]右移,可以实现mr变大,不然处理[l,r]区间又有什么意义呢 
			flag=true;
			if(mr<l){
				mr=r-l+mr;
			}else{
				if(l+mid>mr)mr+=(r-l);
			else mr+=(r-l-(mr-l-mid));
			}
      temp.erase(temp.begin()+i);
			break;
		}
		
	} 
  if(!flag||mr+1e-6>10000)break;
	} 
	
	return mr+1e-6>10000;//循环正常结束也不一定能行,要区间右端最大值大于等于10^4 
}
int main(){//这道题,典型的大于中找最小,大于是指mid超过某一个值,就可以实现所有区间移动覆盖整个范围,最小是指要找的mid要满足条件且最小,直接套模板 
	int n;
	cin>>n;
	int l,r;
	for(int i=0;i<n;i++){
		cin>>l>>r;
		pv.push_back(Point(l,r));
	}
	sort(pv.begin(),pv.end()); 
double L=0,R=10000,mid;
	while(R-L>1e-6){
		mid=(L+R)/2;//维护左边
		if(check(mid)){
			R=mid;
		}else{
			L=mid;
		}
	}
	if(mid>1e-6)cout<<mid<<endl;
	else cout<<0<<endl;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/517479.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

zdpcss_transparent_animation_login:使用纯HTML+CSS+JS开发支持设置主题和带动画的科技风登录界面

废话不多说&#xff0c;先上图&#xff0c;有图有真相&#xff1a; 在左下角有一排颜色&#xff0c;点击可以设置主题色&#xff1a; 比如&#xff0c;我这里点击了橙色&#xff0c;登录界面就变成了如下样子&#xff1a; 另外&#xff0c;在输入账号和密码的时候&#x…

集合系列(十七) -List集合移除元素相关的操作介绍

一、问题由来 在实际开发的时候&#xff0c;我们经常会碰到这么一个困难&#xff1a;一个集合容器里面有很多重复的对象&#xff0c;里面的对象没有主键&#xff0c;但是根据业务的需求&#xff0c;实际上我们需要根据条件筛选出没有重复的对象。 比较暴力的方法&#xff0c;…

【Visual Studio】将项目下的文件夹所有文件随编译自动复制输出到运行目录

要将项目根目录下的文件夹内容输出到运行目录&#xff0c;去处理其中的子文件夹和文件&#xff0c;逐个手动设置文件属性或进行复制显然不是一个可行的方法&#xff0c;因为这既繁琐又低效&#xff0c;那有没有更加高效的方式呢 文章目录 选择文件夹修改配置文件输出文件夹 这里…

亚远景科技-ASPICE评估输入

评估输入应在评估的数据收集阶段之前确定&#xff0c;并得到评估发起人的批准。 评估输入的任何更改都应征得发起人或发起人授权人的同意&#xff0c;并记录在评估记录中。 评估输入至少应明确以下内容&#xff1a; 原文链接&#xff1a;ASPICE评估-ASPICE评估输入-亚远景

Linux集群(一)Nginx搭建

目录 一、Nginx介绍 1.什么是Nginx 2.Nginx的特点 二、Nginx配置 1.jdk的安装 1.1检查jdk版本 1.2上传并安装jdk 2.安装Tomcat 3.下载Nginx 3.1安装依赖包 ​编辑 3.2安装Nginx 3.3运行 三、Nginx中的常用命令​编辑 一、Nginx介绍 1.什么是Nginx Nginx&#xff08;…

网易RAG问答知识库开源了,Star 6K!!

网易RAG问答知识库开源了&#xff0c;Star 6K&#xff01;&#xff01; RAG 问答知识库 QAnything 开源了QAnything 架构设计剖析整个架构的工作流程主要包含三个环节为什么需要两阶段检索&#xff1f;使用的基座大模型相关技术组件 QAnything 本地部署一键部署安装&#xff0c…

实操:flatpicker-时间选择工具

官网 flatpicker是一个轻量级且功能强大的日期时间选择器。精益、用户体验驱动和可扩展&#xff0c;但它不依赖于任何库。用户界面很少&#xff0c;但主题很多。丰富、公开的API和事件系统使其适用于任何环境。 https://flatpickr.js.org/ 依赖 <link rel"stylesheet…

Coursera上Learning Linux for LFCA Certification专项课程01:Linux Fundamentals 学习笔记

Linux Fundamentals Course Certificate 本文是 Linux Fundamentals 这门课的学习笔记&#xff0c;如有侵权&#xff0c;请联系删除。 文章目录 Linux FundamentalsWeek 01: Linux Operating SystemLearning Objectives Specialization OverviewHistory of LinuxQuiz: Hist…

51单片机学习笔记12 SPI接口 使用1302时钟

51单片机学习笔记12 SPI接口 使用1302时钟 一、DS1302简介1. 功能特性2. 涓流充电3. 接口介绍时钟数据和控制线&#xff1a;电源线&#xff1a;备用电池连接&#xff1a; 二、寄存器介绍1. 控制寄存器2. 时间寄存器3. 日历/时钟寄存器 三、BCD码介绍四、DS1302时序1. 读时序2. …

基于SpringBoot的“数码论坛系统设计与实现”的设计与实现(源码+数据库+文档+PPT)

基于SpringBoot的“数码论坛系统设计与实现”的设计与实现&#xff08;源码数据库文档PPT) 开发语言&#xff1a;Java 数据库&#xff1a;MySQL 技术&#xff1a;SpringBoot 工具&#xff1a;IDEA/Ecilpse、Navicat、Maven 系统展示 系统总体结构图 系统首页界面图 数码板…

线段树练习

1.单点修改区间查询 P3374 【模板】树状数组 1 题目描述 如题&#xff0c;已知一个数列&#xff0c;你需要进行下面两种操作&#xff1a; 将某一个数加上 x 求出某区间每一个数的和 输入格式 第一行包含两个正整数 n,m&#xff0c;分别表示该数列数字的个数和操作的总个…

x86汇编写矩阵乘法问题(实现一个3×3矩阵乘法的汇编代码)

&#x1f3c6;本文收录于「Bug调优」专栏&#xff0c;主要记录项目实战过程中的Bug之前因后果及提供真实有效的解决方案&#xff0c;希望能够助你一臂之力&#xff0c;帮你早日登顶实现财富自由&#x1f680;&#xff1b;同时&#xff0c;欢迎大家关注&&收藏&&…

c++的学习之路:10、string(2)

本章主要说一下模拟实现string类的部分功能&#xff0c;文章末附上所有代码。 目录 一、构造函数与析构函数 二、拷贝构造 三、c_str 四、【】和迭代器的遍历与访问 五、size 六、判断 七、reserve 八、push_back 九、resize 十、append 十一、 十二、insert 十…

单例(Singleton)设计模式

2.1 设计模式概述 设计模式是在大量的实践中总结和理论化之后优选的代码结构、编程风格、以及解决问题的思考方式。设计模式免去我们自己再思考和摸索。就像是经典的棋谱&#xff0c;不同的棋局&#xff0c;我们用不同的棋谱。"套路" 经典的设计模式共有23种。每个…

linux基础篇:linux挂载本地yum源——centos7.9为例

linux挂载本地yum源——centos7.9为例 一、Linux本地yum源介绍 Linux本地YUM源是一个本地存储的软件包仓库&#xff0c;它允许用户在不连接互联网的情况下安装、更新和管理软件包。本地YUM源可以提高软件包安装速度&#xff0c;降低网络带宽消耗&#xff0c;并提高软件包管理…

隐私计算实训营学习七:隐语SCQL的架构详细拆解

文章目录 一、SCQL Overview1.1 SCQL背景1.2 SCQL Overview 二、SCQL CCL三、SCQL架构 一、SCQL Overview 1.1 SCQL背景 SCQL&#xff1a;属于隐私计算BI范畴&#xff0c;允许多个互不信任参与方在不泄露各自隐私数据的条件下进行联合数据分析。 如下数据在不同机构&#xf…

【机器学习】如何通过群体智慧解决机器学习的挑战“

机器学习的发展日新月异&#xff0c;但其成功实施的关键之一仍然是获取高质量的、标注良好的数据集。在这篇文章中&#xff0c;我们将探讨如何通过群体智慧来构建和改善机器学习的数据集&#xff0c;尤其是通过reCAPTCHA和带有目的的游戏&#xff08;Games with a Purpose, GWA…

深度学习-计算机视觉入门-part1

计算机视觉入门 文章目录 计算机视觉入门一、 从卷积到卷积神经网络1.图像的基本表示2. 卷积操作3.卷积遇见深度学习3.1 通过学习寻找卷积核3.2 参数共享&#xff1a;卷积带来参数量骤减3.3 稀疏交互&#xff1a;获取更深入的特征 二、手撕卷积代码三、经典CNN模型介绍四、CNN模…

【一】web框架的本质

模板文件和静态文件 ​ 模板文件和静态文件是在 web 开发中常用的两种资源文件&#xff0c;用于构建网站和 web 应用。 模板文件&#xff1a;模板文件是用于定义网页结构和内容的文件。 它们通常使用模板引擎&#xff08;例如Jinja2或Django模板&#xff09;来插入动态数据&a…

【算法练习】28:选择排序学习笔记

一、选择排序的算法思想 弄懂选择排序算法&#xff0c;先得知道两个概念&#xff1a;未排序序列&#xff0c;已排序序列。 原理&#xff1a;以升序为例&#xff0c;选择排序算法的思想是&#xff0c;先将整个序列当做未排序的序列&#xff0c;以序列的第一个元素开始。然后从左…