AcWing 680:剪绳子 ← 浮点数二分

该篇文章详细展示了如何在C++中使用二分查找方法,处理浮点数精度,以计算从给定绳子中裁剪出M根等长绳子的最大可能长度,避免溢出问题。
摘要由CSDN通过智能技术生成

【题目来源】
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
*/





【参考文献】
https://www.acwing.com/solution/content/29443/









 

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值