分巧克力 刷题笔记

/*

分巧克力 解题思路

二分

直接检查看答案是否符合题目条件

对于一块边长分别为x 和y的巧克力\\

假设我们输入检查的数为k

其能分割成的 k*k 的巧克力的块数为

(x/k)*(y/k)

因为c++里面的除法是下取整的所以我们不用考虑奇偶数 是否能整除

将每一块巧克力能分成的k*k的巧克力块数加上计数器

一旦计数器超过了孩子数 我们就返回true;

如果check 不通过的话 可能是分的太大了

所以答案小于mid

于是我们让r=mid-1

如果check通过

则答案>=mid 所以我们让l=mid

重点 讨论边界情况

例如案例中

2 10

6 5

5 6

输出2

当 l指向2 r指向3

mid=(l+r)>>1;的话 mid 是2

此时check可以通过

但是l=2,r=3;

如果还是l=mid=2则陷入死循环

于是 我们让mid=(l+r+1)>>1

让其进行上取整

则 mid=3;

check不通过

此时 r=mid-1=l;

退出循环

输出l或者r即可

*/

代码

#include<iostream>

#include<algorithm>

#include<cstdio>

#include<cstring>

using namespace std;

const int N=1e5+10;

struct node{

int x;

int y;

}a[N];

int n,k;

bool check(int p){

int cnt=0;

bool flag=false;

// cout<<"p is "<<p<<endl;

for(int i=0;i<n;i++){

cnt=cnt+(a[i].x /p)*(a[i].y /p);

//cout <<cnt<<endl;

if(cnt>=k){

flag= true;

break;

}

}

return flag;

}

int main(){

cin>>n>>k;

int r=0;

for(int i=0;i<n;i++){

cin>>a[i].x >>a[i].y;

if(a[i].x >a[i].y ){

if(a[i].x >r){

r=a[i].x ;

}

}else{

if(a[i].y >r){

r=a[i].y ;

}

}

}

// cout<<r<<endl;

int l=0;

while(l<r){

int mid=(l+r+1)>>1;

//cout<<mid<<endl;

if(check(mid)){

l=mid;

}else{

r=mid-1;

}

//cout<<"l is"<<l<<endl<<"r is "<<r<<endl;

}

cout <<l;

return 0;

}

相关推荐
会编程的土豆32 分钟前
01背包与完全背包详解
开发语言·数据结构·c++·算法
汀、人工智能1 小时前
[特殊字符] 第86课:最大正方形
数据结构·算法·数据库架构·图论·bfs·最大正方形
hetao17338371 小时前
2026-04-12~14 hetao1733837 的刷题记录
c++·算法
Elaine3361 小时前
【软件测试系统学习笔记:从理论基础到接口实战】
软件测试·笔记·学习·接口测试
xuhaoyu_cpp_java1 小时前
Maven学习(二)
java·经验分享·笔记·学习·maven
lxh01131 小时前
正则表达式匹配
算法
SuperChe2 小时前
用AI Native的方式优化前端性能
前端·算法
jinyishu_2 小时前
几道链表经典算法题
c语言·数据结构·算法·链表
sparEE2 小时前
进阶排序算法:快速排序
数据结构·算法·排序算法
智者知已应修善业2 小时前
【51单片机4位数循环小数位移数值位移】2023-6-9
c++·经验分享·笔记·算法·51单片机