JOIOI王国 - 二分+贪心
题面

题解
通过一句经典的话“最大值的最小值” 我判断它是二分题,
不难发现,整个图形中两个省的分界线是一条单调不递减或单调不递增的折线。
而且,越到后来它的最大值只会越来越大,最小值只会越来越小,极差只会越来越大。
所以如果我们把ans上界定下来了,我们就可以贪心的让它其中一个区域的值在 [maxa - ans , maxa] 的灰色地带发展,然后判断另一个区域是否合法。由于另一个区域的点数越少越好,所以在灰色地带要尽量发展得广。
我们只能先定义一个 p 表示当前行两省的分界, 然后在第一行尽量扩展得远,下一行在 [0 , p]之间扩展得远,这样不仅可以让另一个区域的点更少,而且可以让下一行有更大的发展空间。
其它的贪心方法就不行了,所以分别行数从小到大和从大到小各枚举一次,然后做个左右对称再来。
CODE
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
#include<queue>
#include<stack>
#include<vector>
#include<map>
#include<set>
#define LL long long
using namespace std;
inline LL read() {
LL f = 1,x = 0;char s = getchar();
while(s < '0' || s > '9') {if(s == '-')f = -1;s = getchar();}
while(s >= '0' && s <= '9'){x = x * 10 + s - '0';s = getchar();}
return x * f;
}
LL n,m,i,j,s,o,k,flag = 0;
LL a[2005][2005];
int b[2005],c[2005];
int x1 = 1,x2 = 1,y1 = 1,y2 = 1;
LL maxl[2005][2005];
LL maxr[2005][2005];
LL minl[2005][2005];
LL minr[2005][2005];
LL ans = 1e18;
bool check(LL mid) {
LL minn = a[x1][y1] - mid;
// cout<<minn<<endl;
LL p = m,mi1 = 0x7f7f7f7f,mi2 = 0x7f7f7f7f,ma1=0,ma2=0;
for(int i = 1;i <= n;i ++) {
int pp = p;
for(int j = 0;j <= pp;j ++) {
if(minl[i][j] >= minn) {
p = j;
}
}
mi1 = min(mi1,minl[i][p]);
ma1 = max(ma1,maxl[i][p]);
mi2 = min(mi2,minr[i][p + 1]);
ma2 = max(ma2,maxr[i][p + 1]);
}
// cout<<"ok1"<<endl;
if(max(ma1 - mi1,ma2 - mi2) <= mid) return 1;
p = m,mi1 = 0x7f7f7f7f,mi2 = 0x7f7f7f7f,ma1=0,ma2=0;
for(int i = n;i > 0;i --) {
int pp = p;
for(int j = 0;j <= pp;j ++) {
if(minl[i][j] >= minn) {
p = j;
}
}
mi1 = min(mi1,minl[i][p]);
ma1 = max(ma1,maxl[i][p]);
mi2 = min(mi2,minr[i][p + 1]);
ma2 = max(ma2,maxr[i][p + 1]);
}
// cout<<"ok2"<<endl;
if(max(ma1 - mi1,ma2 - mi2) <= mid) return 1;
p = 0,mi1 = 0x7f7f7f7f,mi2 = 0x7f7f7f7f,ma1=0,ma2=0;
for(int i = 1;i <= n;i ++) {
int pp = p;
for(int j = m;j >= pp;j --) {
if(minr[i][j + 1] >= minn) {
p = j;
}
}
mi1 = min(mi1,minl[i][p]);
ma1 = max(ma1,maxl[i][p]);
mi2 = min(mi2,minr[i][p + 1]);
ma2 = max(ma2,maxr[i][p + 1]);
}
// cout<<"ok3"<<endl;
if(max(ma1 - mi1,ma2 - mi2) <= mid) return 1;
p = 0,mi1 = 0x7f7f7f7f,mi2 = 0x7f7f7f7f,ma1=0,ma2=0;
for(int i = n;i > 0;i --) {
int pp = p;
for(int j = m;j >= pp;j --) {
if(minr[i][j + 1] >= minn) {
p = j;
}
}
// printf("p---%d\n",p);
mi1 = min(mi1,minl[i][p]);
ma1 = max(ma1,maxl[i][p]);
mi2 = min(mi2,minr[i][p + 1]);
ma2 = max(ma2,maxr[i][p + 1]);
}
// cout<<"ok4"<<endl;
if(max(ma1 - mi1,ma2 - mi2) <= mid) return 1;
return 0;
}
LL solve(LL l,LL r) {
// printf("%lld %lld\n",l,r);
if(l >= r - 1) {
if(check(l)) return l;
return r;
}
LL mid = (l + r) / 2;
if(check(mid)) return solve(l,mid);
return solve(mid,r);
}
int main() {
n = read();m = read();
for(int i = 1;i <= n;i ++) {
minl[i][0] = 1e18;
for(int j = 1;j <= m;j ++) {
a[i][j] = read();
if(a[i][j] > a[x1][y1]) x1 = i,y1 = j;
if(a[i][j] < a[x2][y2]) x2 = i,y2 = j;
minl[i][j] = min(minl[i][j - 1],a[i][j]);
maxl[i][j] = max(maxl[i][j - 1],a[i][j]);
}
minr[i][m + 1] = 1e18;
for(int j = m;j > 0;j --) {
minr[i][j] = min(minr[i][j + 1],a[i][j]);
maxr[i][j] = max(maxr[i][j + 1],a[i][j]);
}
// for(int j = 1;j <= m;j ++) {
// printf("%lld ",minr[i][j]);
// }putchar('\n');
}
printf("%lld\n",solve(0,a[x1][y1] - a[x2][y2]));
return 0;
}
JOIOI王国 - 二分+贪心的更多相关文章
- 「JOI 2017 Final」JOIOI 王国
「JOI 2017 Final」JOIOI 王国 题目描述 题目译自 JOI 2017 Final T3「 JOIOI 王国 / The Kingdom of JOIOI」 JOIOI 王国是一个 H ...
- 「题解」JOIOI 王国
「题解」JOIOI 王国 题目描述 考场思考 正解 题目描述 点这里 考场思考 因为时间不太够了,直接一上来就着手暴力.但是本人太菜,居然暴力爆 000 ,然后当场自闭- 一气之下,发现对 60pts ...
- Codeforces Gym 100231B Intervals 线段树+二分+贪心
Intervals 题目连接: http://codeforces.com/gym/100231/attachments Description 给你n个区间,告诉你每个区间内都有ci个数 然后你需要 ...
- 2016-2017 ACM-ICPC CHINA-Final Ice Cream Tower 二分+贪心
/** 题目:2016-2017 ACM-ICPC CHINA-Final Ice Cream Tower 链接:http://codeforces.com/gym/101194 题意:给n个木块,堆 ...
- 【bzoj2097】[Usaco2010 Dec]Exercise 奶牛健美操 二分+贪心
题目描述 Farmer John为了保持奶牛们的健康,让可怜的奶牛们不停在牧场之间 的小路上奔跑.这些奶牛的路径集合可以被表示成一个点集和一些连接 两个顶点的双向路,使得每对点之间恰好有一条简单路径. ...
- Codeforces_732D_(二分贪心)
D. Exams time limit per test 1 second memory limit per test 256 megabytes input standard input outpu ...
- CF732D Exams 二分 贪心
思路:二分+贪心 提交次数:10次以上 错因:刚开始以为二分(边界,$+1or-1$)写错了,调了半天,后来才发现是$ck()$写错了.开始只判了最后是否小于零,而应该中间一旦小于零就$return\ ...
- $CF949D\ Curfew$ 二分/贪心
正解:二分/贪心 解题报告: 传送门$QwQ$ 首先这里是二分还是蛮显然的?考虑二分那个最大值,然后先保证一个老师是合法的再看另一个老师那里是否合法就成$QwQ$. 发现不太会搞这个合不合法的所以咕了 ...
- $bzoj2067\ szn$ 二分+贪心
正解:二分+贪心 解题报告: 传送门$QwQ$ 题目大意就说有一棵树,然后要用若干条线覆盖所有边且不能重叠.问最少要用几条线,在用线最少的前提下最长的线最短是多长. 昂首先最少用多少条线这个还是蛮$e ...
随机推荐
- ShardingSphere-proxy-5.0.0部署之分表实现(一)
一.说明 环境准备:JDK8+ mysql 5.x 官网:https://shardingsphere.apache.org/ 下载地址:https://archive.apache.org/ ...
- 开源流程引擎osworkflow、jbpm、activiti、flowable、camunda哪个好?
市场上比较有名的开源流程引擎有osworkflow.jbpm.activiti.flowable.camunda.其中:Jbpm4.Activiti.Flowable.camunda四个框架同宗同源, ...
- spring boot用ide新建项目遇到的restcontroller不能导入的问题
才开始学习spring boot,第一个程序helloworld就碰到@RestController和@RequestMapping(/hello)的注解都会报错的问题. 我个人的解决方法: 1.sp ...
- SAP -SE30 程序运行时间分析
运行SE30 选中Program,点击Excute 点击运行 分析结果
- js 表面学习 - 认识结构2
单行注释以 // 开头. 多行注释以 /* 开头,以 */ 结尾. 任何位于 /* 和 */ 之间的文本都会被 JavaScript 忽略. JavaScript 数据类型 JavaScript 变量 ...
- Vue动态组件的实践与原理探究
我司有一个工作台搭建产品,允许通过拖拽小部件的方式来搭建一个工作台页面,平台内置了一些常用小部件,另外也允许自行开发小部件上传使用,本文会从实践的角度来介绍其实现原理. ps.本文项目使用Vue CL ...
- TypeScript let与var的区别
1.作用域不同 用var声明的变量,只有函数作用域和全局作用域,没有块级作用域.而let可以实现块级作用域,只能在代码块{}内有效,在{}之外不能访问,如下代码所示: { let a = 0; var ...
- 面试官:Redis 过期删除策略和内存淘汰策略有什么区别?
作者:小林coding 计算机八股文网站:https://xiaolincoding.com 大家好,我是小林. Redis 的「内存淘汰策略」和「过期删除策略」,很多小伙伴容易混淆,这两个机制虽然都 ...
- CSS 盒子模型(一)
CSS 盒子模型(一) 本人在校学生,主学后端,后来发现前端的基础都忘得差不多了才想着写文章回来复习!欢迎留言交流. 什么是盒子呢? 拿下举例,我们可以把每个红框都比作一个盒子,他们可以是任意的 HT ...
- 如何优化API?8个实用技巧!【eolink翻译】
使用 API 可以让公司利用现代连接的力量来帮助他们扩大全球影响力.传输数据和改进集成.由于 API 使企业能够简化流程并增强可用性,所以企业会使用一些优化策略,不断优化流程,比如接下来要说到的8个技 ...