二分查找(九)2300. 咒语和药水的成功对数

2300. 咒语和药水的成功对数

给你两个正整数数组 spellspotions ,长度分别为 nm ,其中 spells[i] 表示第 i 个咒语的能量强度,potions[j] 表示第 j 瓶药水的能量强度。

同时给你一个整数 success 。一个咒语和药水的能量强度 相乘 如果 大于等于 success ,那么它们视为一对 成功 的组合。

请你返回一个长度为 n 的整数数组pairs,其中pairs[i] 是能跟第 i 个咒语成功组合的 药水 数目。

示例 1:

复制代码
输入:spells = [5,1,3], potions = [1,2,3,4,5], success = 7
输出:[4,0,3]
解释:
- 第 0 个咒语:5 * [1,2,3,4,5] = [5,10,15,20,25] 。总共 4 个成功组合。
- 第 1 个咒语:1 * [1,2,3,4,5] = [1,2,3,4,5] 。总共 0 个成功组合。
- 第 2 个咒语:3 * [1,2,3,4,5] = [3,6,9,12,15] 。总共 3 个成功组合。
所以返回 [4,0,3] 。

整体思路较为清楚,遍历每一份spells,利用这个spell来进行与potions每个元素乘积结果的判断,使用二分搜索优化,找到第一个大于等于target的位置,后续直接用个数-位置即可

cpp 复制代码
class Solution {
public:
    int lower_bound(int spell, vector<int>& potions, long long target)
    {
        int left = 0, right = potions.size()-1;
        while(left <= right)
        {
            int mid = left + (right-left)/2;
            // long long temp = potions[mid] * spell;
            // if(potions[mid] < target/spell)
            if (1LL * potions[mid] * spell < target)
                left = mid + 1;
            else
                right = mid - 1;
        }
        return left;
    }
    vector<int> successfulPairs(vector<int>& spells, vector<int>& potions, long long success) {
        int n = spells.size(), m = potions.size();
        vector<int> res(n);
        sort(potions.begin(), potions.end());
        for(int i = 0; i < n; i++)
        {
            int index = lower_bound(spells[i], potions, success);
            res[i] = m - index;
        }
        return res;
    }
};

主要问题是记录一下long long型元素的结果溢出

以下错误写法:由于potions和spell元素都是int类型,所以他们会先进行相乘,但结果已经超过他们的存储范围了,这时候再用longlong来接收就已经晚了

方案A:使用1LL

long long temp = 1LL * potions[mid] * spell;

方案B:使用显示类型转换

long long temp = static_cast<long long>(potions[mid]) * spell;

cpp 复制代码
    int lower_bound(int spell, vector<int>& potions, long long target)
    {
        int left = 0, right = potions.size()-1;
        while(left <= right)
        {
            int mid = left + (right-left)/2;
            long long temp = potions[mid] * spell;
            if (potions[mid] * spell < target)
                left = mid + 1;
            else
                right = mid - 1;
        }
        return left;
    }

还有就是把乘法转化为除法的形式:

使用lower_bound库函数

cpp 复制代码
class Solution {
public:
    vector<int> successfulPairs(vector<int>& spells, vector<int>& potions, long long success) {
        vector<int> res;
        res.reserve(spells.size());
        sort(potions.begin(), potions.end());
        for(int spell : spells)
        {
            // long long min_success = 1LL * e * success;
            long long min_p = (success + spell - 1) / spell;
            auto it = lower_bound(potions.begin(), potions.end(), min_p);
            res.push_back(potions.end() - it);
        }
        return res;
    }
};
相关推荐
Pluchon1 分钟前
硅基计划4.0 简单模拟实现AVL树&红黑树
java·数据结构·算法
生锈的键盘10 分钟前
推荐算法实践:交叉特征的理解
算法
小龙报16 分钟前
【51单片机】从 0 到 1 玩转 51 蜂鸣器:分清有源无源,轻松驱动它奏响新年旋律
c语言·数据结构·c++·stm32·单片机·嵌入式硬件·51单片机
dllxhcjla20 分钟前
数据结构和算法
数据结构
乌萨奇也要立志学C++23 分钟前
【洛谷】BFS 求解最短路:从马的遍历到迷宫问题的实战解析
算法·宽度优先
老鼠只爱大米32 分钟前
LeetCode经典算法面试题 #46:全排列(回溯、交换、剪枝等五种实现方案详细解析)
算法·leetcode·剪枝·回溯·全排列·stj算法
Dovis(誓平步青云)1 小时前
《滑动窗口算法:从 “暴力遍历” 到 “线性高效” 的思维跃迁》
运维·服务器·数据库·算法
_OP_CHEN1 小时前
【算法基础篇】(五十七)线性代数之矩阵乘法从入门到实战:手撕模板 + 真题详解
线性代数·算法·矩阵·蓝桥杯·c/c++·矩阵乘法·acm/icpc
天天爱吃肉82181 小时前
【跨界封神|周杰伦×王传福(陶晶莹主持):音乐创作与新能源NVH测试,底层逻辑竟完全同源!(新人必看入行指南)】
python·嵌入式硬件·算法·汽车
im_AMBER1 小时前
Leetcode 114 链表中的下一个更大节点 | 删除排序链表中的重复元素 II
算法·leetcode