NC19916 [CQOI2010]撲克牌(二分)

題目鏈接

題意:
nici有n種牌,第i種有c_i張
mjoker有m張joker牌
可以用每種牌各一張來組成一套牌
joker1也可以用一張joker和除了某一種牌以外的其他牌各一張組成1套牌
最多能組成多少套牌
題解:
直接二分答案
xmidxmid假設答案是x,如果二分值mid小於x,那麼mid套牌一定可以組成
x如果大於x,那麼一定組成不了
所以證實了二分的想法

x然後其實很容易,就是看組成x套牌差多少張組不成一套的
joker但是題目中還有一個限制,每一套牌只能用一張joker牌
所以如果差的張數比當前組成的套數還要多
joker那說明肯定有其中的某一套用了多張joker牌
mx所以保證差的數量小於m的同時還要小於x

AC代碼

/*
    Author:zzugzx
    Lang:C++
    Blog:blog.csdn.net/qq_43756519
*/
#include<bits/stdc++.h>
using namespace std;
#define fi first
#define se second
#define pb push_back
#define mp make_pair
#define all(x) (x).begin(),(x).end()
#define endl '\n'
typedef long long ll;
typedef pair<int, int> pii;
typedef pair<ll, ll> pll;
const int mod=1e9+7;
//const int mod=998244353;
const double eps = 1e-10;
const double pi=acos(-1.0);
const int maxn=1e6+10;
const ll inf=0x3f3f3f3f;
const int dir[4][2]={{0,1},{1,0},{0,-1},{-1,0}};

ll n,m;
ll a[maxn];
bool ok(ll x){
    ll res=0;
    for(int i=1;i<=n;i++)
        if(a[i]<x)res+=x-a[i];
    return res<=min(m,x);
}

int main()
{
    ios::sync_with_stdio(false);
    cin.tie(0);cout.tie(0);
    //freopen("in.txt","r",stdin);
    //freopen("out.txt","w",stdout);
    cin>>n>>m;
    for(int i=1;i<=n;i++)cin>>a[i];
    ll l=0,r=1e9,ans;
    while(l<=r){
        ll mid=l+r>>1;
        if(ok(mid))ans=mid,l=mid+1;
        else r=mid-1;
    }
    cout<<ans;
    return 0;
}

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章