题解—P2218 [HAOI2007]覆盖问题
一道不错的题,主要就是一个思路点,想到就行了,想不到就一直卡着。
看完题解之后发现挺简单,实际上自己挣扎半天也咩有想到。
一开始想类比成一维之后贪心,后来被同机房大佬 \(hack\) 掉了。
solution
首先二分答案,单调行很好证明,因为大矩形包含小矩形,现在的问题就是 \(check\) 函数
其实一个突破口就是只用三个矩形覆盖。
一维里面用几个矩形理论上都是通过输入给出的,所以一般的题的变量他通过常量给出来,可能就是突破口。
对于题目中所有的点,肯定可以用一个大矩形套住,我们称之为“外接矩形”
思考一下可以发现,外接矩形的边上肯定有点,那么也一定有三个矩形的边(放边的原因是多放到外面肯定不如放到里面优)
一个外接矩形有四条边,这就意味着,对于我们的三个矩形,肯定有一个要和两条边重合(也就是和一个角重合)
我们枚举是哪个角,之后把它范围内的点覆盖上,然后问题就变成了一个子问题(因为小于等于三个矩形都必须覆盖角,所以枚举角就可以了)。
然后经典的搜索和回溯就可以写完 \(check\) 函数。
code
(放着么美的代码有没有亮瞎您的眼?)
#include <cstring>
#include <algorithm>
#include <cstdio>
#define mp make_pair
#define R register int
#define int long
#define printf Ruusupuu = printf
int Ruusupuu ;
using namespace std ;
typedef long long L ;
typedef long double D ;
typedef unsigned long long G ;
typedef pair< int , int > PI ;
const int N = 2e4 + 10 ;
const int Inf = 1e9 + 10 ;
inline int read(){
int w = 0 ; bool fg = 0 ; char ch = getchar() ;
while( ch < '0' || ch > '9' ) fg |= ( ch == '-' ) , ch = getchar() ;
while( ch >= '0' && ch <= '9' ) w = ( w << 1 ) + ( w << 3 ) + ( ch ^ '0' ) , ch = getchar() ;
return fg ? -w : w ;
}
int n , mid , x [N] , y [N] ;
bool cvr [N] ;
int st [N] , top ;
void sc(){
n = read() ;
for( R i = 1 ; i <= n ; i ++ ) x [i] = read() , y [i] = read() ;
}
inline void rmcvr( int x ){ while( top > x ) cvr [st [top --]] = 0 ; }
inline bool check( int now ){
if( top == n ) return 1 ;
if( now == 4 ) return 0 ;
int l = Inf , r = -Inf , u = -Inf , d = Inf , bg = top ;
bool fg = 0 ;
for( R i = 1 ; i <= n ; i ++ ) if( !cvr [i] ){
l = min( l , x [i] ) , r = max( r , x [i] ) ;
d = min( d , y [i] ) , u = max( u , y [i] ) ;
}
// printf( "%ld %ld %ld %ld %ld\n" , now , l , r , u , d ) ;
int ix , iy ;
// left up
ix = l , iy = u ;
for( R i = 1 ; i <= n ; i ++ ) if( !cvr [i] ){
if( x [i] >= ix && x [i] <= ix + mid && y [i] >= iy - mid && y [i] <= iy ) cvr [i] = 1 , st [++ top] = i ;
} fg |= check( now + 1 ) ; rmcvr( bg ) ; if( fg ) return 1 ;
// left down
ix = l , iy = d ;
for( R i = 1 ; i <= n ; i ++ ) if( !cvr [i] ){
if( x [i] >= ix && x [i] <= ix + mid && y [i] >= iy && y [i] <= iy + mid ) cvr [i] = 1 , st [++ top] = i ;
} fg |= check( now + 1 ) ; rmcvr( bg ) ; if( fg ) return 1 ;
// right up
ix = r , iy = u ;
for( R i = 1 ; i <= n ; i ++ ) if( !cvr [i] ){
if( x [i] >= ix - mid && x [i] <= ix && y [i] >= iy - mid && y [i] <= iy ) cvr [i] = 1 , st [++ top] = i ;
} fg |= check( now + 1 ) ; rmcvr( bg ) ; if( fg ) return 1 ;
// right down
ix = r , iy = d ;
for( R i = 1 ; i <= n ; i ++ ) if( !cvr [i] ){
if( x [i] >= ix - mid && x [i] <= ix && y [i] >= iy && y [i] <= iy + mid ) cvr [i] = 1 , st [++ top] = i ;
} fg |= check( now + 1 ) ; rmcvr( bg ) ; if( fg ) return 1 ;
return 0 ;
}
void work(){
int lside = 1 , rside = (int) 2e9 + 10 , ans ;
while( lside <= rside ){
mid = ( 0ll + lside + rside ) >> 1 ;
if( check( 1 ) ) ans = mid , rside = mid - 1 ;
else lside = mid + 1 ;
} printf( "%ld\n" , ans ) ;
}
signed main(){
sc() ;
work() ;
return 0 ;
}
题解—P2218 [HAOI2007]覆盖问题的更多相关文章
- 洛谷 P2218 [HAOI2007]覆盖问题 解题报告
P2218 [HAOI2007]覆盖问题 题目描述 某人在山上种了\(N\)棵小树苗.冬天来了,温度急速下降,小树苗脆弱得不堪一击,于是树主人想用一些塑料薄膜把这些小树遮盖起来,经过一番长久的思考,他 ...
- P2218 [HAOI2007]覆盖问题
传送门 首先可以想到二分答案,然后考虑判断 注意到所有点的外包矩形的四条边一定要被覆盖到,而正方形只有 $3$ 个,所以一定有一个正方形在角落 考虑爆搜,枚举正方形在当前外包矩形的那个角,然后对剩下的 ...
- BZOJ 1052: [HAOI2007]覆盖问题
BZOJ 1052: [HAOI2007]覆盖问题 题意:给定平面上横纵坐标在-1e9~1e9内的20000个整数点的坐标,用三个大小相同边平行于坐标轴的正方形覆盖(在边界上的也算),问正方形的边长最 ...
- 【BZOJ1052】 [HAOI2007]覆盖问题
BZOJ1052 [HAOI2007]覆盖问题 前言 小清新思维题. 最近肯定需要一些思维题挽救我这种碰到题目只会模板的菜鸡. 这题腾空出世? Solution 考虑一下我们二分答案怎么做? 首先转换 ...
- 【BZOJ 1052】 1052: [HAOI2007]覆盖问题 (乱搞)
1052: [HAOI2007]覆盖问题 Description 某人在山上种了N棵小树苗.冬天来了,温度急速下降,小树苗脆弱得不堪一击,于是树主人想用一些塑料薄 膜把这些小树遮盖起来,经过一番长久的 ...
- [BZOJ1052][HAOI2007]覆盖问题 二分+贪心
1052: [HAOI2007]覆盖问题 Time Limit: 10 Sec Memory Limit: 162 MB Submit: 2053 Solved: 959 [Submit][Sta ...
- 【BZOJ】1052: [HAOI2007]覆盖问题(贪心)
http://www.lydsy.com/JudgeOnline/problem.php?id=1052 首先膜拜题解orz,表示只能想到二分... 贪心就是每一次找到一个最小的能包围所有点的矩阵,然 ...
- bzoj1052 [HAOI2007]覆盖问题 - 贪心
Description 某人在山上种了N棵小树苗.冬天来了,温度急速下降,小树苗脆弱得不堪一击,于是树主人想用一些塑料薄膜把这些小树遮盖起来,经过一番长久的思考,他决定用3个L*L的正方形塑料薄膜将小 ...
- 1052: [HAOI2007]覆盖问题 - BZOJ
Description 某人在山上种了N棵小树苗.冬天来了,温度急速下降,小树苗脆弱得不堪一击,于是树主人想用一些塑料薄膜把这些小树遮盖起来,经过一番长久的思考,他决定用3个L*L的正方形塑料薄膜将小 ...
随机推荐
- buu firmware
一.路由器固件,给的是bin文件,要用binwalk将固件文件系统提取出来,同时binwalk的版本要完整不然解压不了文件,下面说的很清楚了. https://blog.csdn.net/QQ1084 ...
- 使用Hugo框架搭建博客的过程 - 功能拓展
前言 本文介绍一些拓展功能,如文章页面功能增加二级菜单,相关文章推荐和赞赏.另外,使用脚本会大大简化写作后的上传流程. 文章页面功能 这部分功能的拓展主要是用前端的JS和CSS,如果对前端不了解,可以 ...
- 「CF516D」 Drazil and Morning Exercise
「CF516D」 Drazil and Morning Exercise 传送门 这个 \(f_i\) 显然可以通过树形 \(\texttt{DP}\) 直接求. 然后看到这种差值问题感觉就可以二分转 ...
- ERROR 1045 (28000): Access denied for user ‘root‘@‘localhost‘ (using password: NO)解决办法
问题:ERROR 1045 (28000): Access denied for user 'root'@'localhost' (using password: NO) 很久没用这台电脑的mysql ...
- [刘阳Java]_MySQL数据优化总结_查询备忘录
数据库优化是在后端开发中必备技能,今天写一篇MySQL数据优化的总结,供大家看看 一.MySQL数据库优化分类 我们通过一个图片形式来看看数据优化一些策略问题 不难看出,优化有两条路可以选择:硬件与技 ...
- 在 .NET 中创建对象的几种方式的对比
在 .net 中,创建一个对象最简单的方法是直接使用 new (), 在实际的项目中,我们可能还会用到反射的方法来创建对象,如果你看过 Microsoft.Extensions.DependencyI ...
- pagehelper插件使用时查询不到数据
刚用mybatis 的分页插件时,老项目中分页封装的分页类起始为( pageno-1)* pagesize 于是直直接在pagehelper.start(start,pagesize)来进行分页.结 ...
- Linux文件浏览(less、more、head、tail)以及vim编辑器命令总结
结合小编学习实践,整理了Linux下查看内容(less.more.head.tail)和编辑内容(vim)的部分基础命令解析: 1.less命令 分页浏览内容,如果在shell命令行下按回车键,则一点 ...
- linux 之awk 次数统计
sort +awk+uniq 统计文件中出现次数 jps -v |grep jar|grep -v Jps|awk 'BEGIN{FS=".jar "} {print $1}' ...
- JavaScript之DOM、DOM树
一 DOM JavaScript操作网页的接口,全称为"文档对象模型"(Document Object Model). 有这几个概念:文档.元素.节点 整个文档是一个文档节点 每个 ...