【题目来源】
https://www.acwing.com/problem/content/682/
【题目描述】
有 N 根绳子,第 i 根绳子长度为 Li,现在需要 M 根等长的绳子,你可以对 N 根绳子进行任意裁剪(不能拼接),请你帮忙计算出这 M 根绳子最长的长度是多少。
【输入格式】
第一行包含 2 个正整数 N、M,表示原始绳子的数量和需求绳子的数量。
第二行包含 N 个整数,其中第 i 个整数 Li 表示第 i 根绳子的长度。
【输出格式】
输出一个数字,表示裁剪后最长的长度,保留两位小数。
【数据范围】
1≤N,M≤100000,
0<Li<10^9
【输入样例】
3 4
3 5 4
【输出样例】
2.50
【样例解释】
第一根和第三根分别裁剪出一根 2.50 长度的绳子,第二根剪成 2 根 2.50 长度的绳子,刚好 4 根。
【算法分析】
● 因为浮点数的精度很高,只需要逐渐逼近题目要求的精度就可以了。这里需要注意的是,需要预先设定一个阈值 eps,一般是比题目的精度还要高 2 位,比如题目要求的精度是1e-2,那么就可以设eps=1e-4。如本题中的 while(ri-le>1e-4){……}。
● 当数据比较大时,可能会产生溢出。所以,本题中使用 double mid=le+(ri-le)/2; 而不是 double mid=(le+ri)/2;
【算法代码】
#include <bits/stdc++.h>
using namespace std;
const int N=1e5+5;
int a[N];
int n,m;
bool check(double len){
int ans=0;
for(int i=0;i<n;i++){
ans+=a[i]/len;
if(ans>=m) return true;
}
return false;
}
int main() {
cin>>n>>m;
for(int i=0;i<n;i++) cin>>a[i];
double le=0;
double ri=1e9;
while(ri-le>1e-4) {
double mid=le+(ri-le)/2; //Prevent overflow, no using double mid=(le+ri)/2;
if(check(mid)) le=mid;
else ri=mid;
}
printf("%.2f",le);
}
/*
in:
3 4
3 5 4
out:
2.50
*/