【Codeforces Round #786 (Div. 3)】 E Breaking the Wall

题面意思

给你一个数组,一种操作,使得这个数组至少2个数小于等于0。
操作:选定一个位置i,i位置的数-2,i-1和i+1位置的数-1(若存在的话)。

思路

首先分类讨论可以得到三种情况:

  1. 若最小值的两个数距离大于1,那么只需要计算让这两个数一直-2直到小于等于0就好了。
    样例: 4 9 9 9 3
  2. 若最小值的两个数距离等于1,那么只需要先打中间的数使得两边的数各-1直到较小的数为0,然后再让大的数-2直到小于等于0。
    样例 10 30 10
  3. 最小值的两个数相邻,这个情况比较复杂。
    样例: 10 10 40

最后将三种情况的最小值比较一下就能得到答案。

可以发现第一种情况和第二种情况都可以直接计算得到,但是第三种情况比较复杂,再次分类讨论又比较麻烦。

所以可以通过二分搜索来找这个解。

首先先二分搜索需要多少次才能使得这两个数都小于等于0,我们假设为cnt次。但是这样还不够,因为cnt次还不知道要怎么分开。
因此可以再套一个二分搜索,此时mid为较小的数-2的次数,cnt-mid为较大的数-2的次数,如果找到能使得两者都变为0的情况,说明cnt是成立的,这样就可以找到最少的次数了。

#include<bits/stdc++.h> 
#define fs first
#define se second
#define pb push_back
#define cppio ios::sync_with_stdio(false);cin.tie(0)
using namespace std;

typedef long long ll;
typedef pair<int,int> pii;
typedef vector<int> VI;

const int maxn=2e5+5;
const ll inf=0x3f3f3f3f;
const ll mod=32768;

int a[maxn];

int check(int num,int x,int y){
	int l=0,r=num;
	while(l<=r){
		int mid=(l+r)>>1;
		int tx=x,ty=y;
		tx-=mid*2;ty-=(num-mid)*2;
		tx-=(num-mid);ty-=mid;
		if(tx<=0&&ty<=0){
			return 1;
		}
		if(tx>0){
			l=mid+1;
		}
		else {
			r=mid-1;
		}
	}
	return 0;
}

int cal(int x,int y){
	if(x>y) swap(x,y);
	int l=(x+y+2)/3,r=1e6,ans=inf;
	while(l<=r){
		int mid=(l+r)>>1;
		if(check(mid,x,y)){
			ans=mid;
			r=mid-1;
		}
		else l=mid+1;
	}
	return ans;
}

int main(){
	int n;
	scanf("%d",&n);
	int mx1=inf,mx2=inf;
	for(int i=1;i<=n;i++){
		scanf("%d",a+i);
		if(a[i]<mx1){
			mx2=mx1;
			mx1=a[i];
		}
		else if(a[i]<mx2){
			mx2=a[i];
		}
	}
	int ans=(mx1+1)/2+(mx2+1)/2;
	for(int i=2;i<=n-1;i++){
		int t1=min(a[i-1],a[i+1]);
		int t2=max(a[i-1],a[i+1]);
		ans=min(ans,t1+(t2-t1+1)/2);
	}
	for(int i=1;i<=n-1;i++){
		ans=min(ans,cal(a[i],a[i+1]));
	}
	printf("%d\n",ans);
	return 0;
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值