leetcode 3453(二分法)

3453: 分割正方形Ⅰ

*思路:浮点二分

squares[i] = [xi, yi, li] 表示一个与 x 轴平行的正方形的左下角坐标和正方形的边长。

所有正方形的面积之和为

枚举正方形 (xi​,++yi​++ ,li​),如果水平线在正方形底边上方,即 yi​<y,那么这个正方形在水平线下方的面积为

否则在水平线下方的面积为 0。

细节: 二分的左边界为 0,右边界为 max(yi​+li​)。++这里无需讨论开闭区间,因为我们算的是小数。++ 推荐的写法是固定一个循环次数,因为浮点数有舍入误差,可能算出的 mid 和 left 相等,此时 left=mid 不会更新 left,导致死循环。

循环次数:

复制代码
for(int i=0;i<47;i++){
    double mid=(left+right)/2;
    (check(mid)? right:left)=mid;
}
return (left+right)/2; //取中点误差极小

固定做 47 次二分(计算过程如上)

  • 47 次可以把区间长度缩小到 (max_y-0)/2^47,对于 double 的 53 位有效精度来说已经足够
复制代码
class Solution {
public:
    double separateSquares(vector<vector<int>>& squares) {
        long long tot_area=0;
        int max_y=0;
        for(auto& sq:squares){
            int l=sq[2]; //正方形边长
            tot_area+=(long long)l*l;
            max_y=max(max_y,sq[1]+l);
        }
        auto check=[&](double y)->bool{
            double area=0;
            for(auto& sq:squares){
                double yi=sq[1];
                if(yi<y){
                    double l=sq[2];
                    area+=l*min(y-yi,l);
                }
            }
            return area>=tot_area/2.0; //返回true,说明可行y可以更小
        };
        double left=0,right=max_y;
        for(int i=0;i<47;i++){
            double mid=(left+right)/2;
            (check(mid)? right:left)=mid;
        }
        return (left+right)/2; //取中点误差极小
    }
};
相关推荐
少许极端12 小时前
算法奇妙屋(四十六)-二分答案
算法·二分答案
漂流瓶jz13 小时前
UVA-120 煎饼 题解答案代码 算法竞赛入门经典第二版
数据结构·c++·算法·排序·aoapc·算法竞赛入门经典·uva
paeamecium13 小时前
【PAT甲级真题】- Stack (30)
数据结构·算法·pat考试·pat
黎阳之光13 小时前
黎阳之光核工厂202应急管控平台|全域实景孪生,筑牢核安全最后一道防线
大数据·人工智能·算法·安全·数字孪生
莫等闲-13 小时前
代码随想录一刷记录Day31——leetcode56. 合并区间 738.单调递增的数字
数据结构·c++·算法·leetcode
克里普crirp13 小时前
短波通信的可用频率计算方法
人工智能·算法·机器学习
剑挑星河月13 小时前
45.跳跃游戏Ⅱ
数据结构·算法·leetcode
MegaDataFlowers14 小时前
1.两数之和
算法
AGV算法笔记14 小时前
二维码检测又卷出新方向:如何在一张图里稳定读取几十甚至上百个二维码?
算法·目标检测·二维码·视觉算法
sparEE14 小时前
进阶排序算法:归并排序
数据结构·算法·排序算法