首页 > 其他分享 >LeetCode 7023操作使得分最大

LeetCode 7023操作使得分最大

时间:2023-08-13 22:44:54浏览次数:45  
标签:idx nums 使得 质数 7023 int vector scores LeetCode

7023. 操作使得分最大

题目描述:一个数字的质数分数为其质因数个数;给定一个长度为\(n\)的正整数数组nums和正整数k,可以进行k次如下操作:

  • 选择一个之前没有选择过的非空子数组subnums
  • subnums中选择一个质数分数最大的元素x,如果多个元素质数分数相同且最大,选择下标最小的一个
  • 将得分乘以x

初始时得分为1;求经过k次操作后的最大得分,对\(10^9 + 7\)取模

思路:比较明显,每次选的x越大最终得分越高,所以我们需要每次选择最大的x,然后枚举所有包含x的子数组中,x的质数分数最大且下标最小的数量,假设为m,则对答案的贡献就为\(x^{\min(m, k)}\)。假设scoresnums所对应的质数分数数组,那么对于一个下标\(i\),要满足上述条件,则需要求最小的\(j\)满足对于任意的\(j \leq r < i\)有\(scores_i > scores_r\);求最大的\(k\),满足对于任意的\(i \leq r \leq k\)有\(scores_i \geq scores_r\);则此时\(m = (i - j + 1) \times (k - i + 1)\);根据上述描述,显然可以使用单调栈求解scores数组。

时间复杂度:\(O(nlogn)\)

空间复杂度:\(O(n)\)

参考代码:

class Solution {
public:
    int maximumScore(vector<int>& nums, int k) {
        const int MX = 100005;
        const int mod = 1000'000'007;
        vector<int> primeCount(MX, 0);
        for (int i = 2; i < MX; ++i) {
            if (primeCount[i] != 0) {
                continue;
            }
            for (int j = i; j < MX; j += i) {
                primeCount[j]++;
            }
        }
        int n = nums.size();
        vector<int> scores(n, 0);
        vector<int> index(n);
        for (int i = 0; i < n; ++i) {
            scores[i] = primeCount[nums[i]];
            index[i] = i;
        }
        sort(index.begin(), index.end(), [&](const int a, const int b) {
            return nums[a] > nums[b];
        });
        vector<int> left(n, -1), right(n, n);
        stack<int> s;
        for (int i = 0; i < n; ++i) {
            while (!s.empty() && scores[s.top()] < scores[i]) {
                int pop = s.top();
                s.pop();
                right[pop] = i;
            }
            if (!s.empty()) {
                left[i] = s.top();
            }
            s.push(i);
        }
        function<int(int, int)> quickPow = [](int a, int p) -> int {
            int ret = 1;
            while (p > 0) {
                if ((p & 1) == 1) {
                    ret = 1ll * ret * a % mod;
                }
                p >>= 1;
                a = 1ll * a * a % mod;
            }
            return ret;
        };
        int res = 1;
        for (auto idx : index) {
            long long cur = 1ll * (idx - left[idx]) * (right[idx] - idx);
            if (cur > k) {
                cur = k;
            }
            res = 1ll * res * quickPow(nums[idx], cur) % mod;
            k -= cur;
            if (k == 0) {
                break;
            }
        }
        return res;
    }
};

标签:idx,nums,使得,质数,7023,int,vector,scores,LeetCode
From: https://www.cnblogs.com/cherish-/p/17627434.html

相关文章

  • LeetCode 279.完全平方数
    1.题目:给你一个整数 n ,返回 和为 n 的完全平方数的最少数量 。完全平方数 是一个整数,其值等于另一个整数的平方;换句话说,其值等于一个整数自乘的积。例如,1、4、9 和 16 都是完全平方数,而 3 和 11 不是。 https://leetcode.cn/problems/perfect-squares/description/......
  • Leetcode No.53 Maximum Subarray
    参考资料:考点:子串&动态规划&[题干]Input:nums=[-2,1,-3,4,-1,2,1,-5,4]Output:6Explanation:Thesubarray[4,-1,2,1]hasthelargestsum6.1.心路历程这道题非常经典,蕴含的思想也是精巧无比。2.正解简单来说官解就是找到了题目中......
  • LeetCode 周赛上分之旅 #39 结合中心扩展的单调栈贪心问题
    ⭐️本文已收录到AndroidFamily,技术和职场问题,请关注公众号[彭旭锐]和BaguTreePro知识星球提问。学习数据结构与算法的关键在于掌握问题背后的算法思维框架,你的思考越抽象,它能覆盖的问题域就越广,理解难度也更复杂。在这个专栏里,小彭与你分享每场LeetCode周赛的解题报告,一......
  • LeetCode 322.零钱兑换
    1.题目:给你一个整数数组 coins ,表示不同面额的硬币;以及一个整数 amount ,表示总金额。计算并返回可以凑成总金额所需的 最少的硬币个数 。如果没有任何一种硬币组合能组成总金额,返回 -1 。你可以认为每种硬币的数量是无限的。 https://leetcode.cn/problems/coin-change/示......
  • LeetCode 7022——熟悉TreeSet数据结构及常用方法的使用
    LeetCode7022.限制条件下元素之间的最小绝对差题目描述:给你一个下标从 0 开始的整数数组 nums 和一个整数 x 。请你找到数组中下标距离至少为 x 的两个元素的 差值绝对值 的 最小值 。换言之,请你找到两个下标 i 和 j ,满足 abs(i-j)>=x 且 abs(nums[i......
  • C++STL库 二分查找,以及对set集合进行二分查找,来源于”leetcode7022. 限制条件下元素之
    C++的头文件<algorithm>中有用于二分查找的函数,lower_bound()、upper_bound()以及binary_search():lower_bound():返回大于等于目标值的第一个位置upper_bound():返回大于目标值的第一个位置,binary_search():若目标值存在则返回true,否则返回false参数列表:(起始位置,结束位置,目标值) ......
  • #yyds干货盘点# LeetCode程序员面试金典:数组中的第K个最大元素
    题目:给定整数数组 nums 和整数 k,请返回数组中第 k 个最大的元素。请注意,你需要找的是数组排序后的第 k 个最大的元素,而不是第 k 个不同的元素。你必须设计并实现时间复杂度为 O(n) 的算法解决此问题。 示例1:[3,2,1,5,6,4],示例 2:[3,2,3,1,2,4,5,5,6],代码实现:class......
  • #yyds干货盘点# LeetCode程序员面试金典:查找和最小的 K 对数字
    1.简述:给定两个以 非递减顺序排列 的整数数组 和  , 以及一个整数  。nums1nums2k定义一对值 ,其中第一个元素来自 ,第二个元素来自  。(u,v)nums1nums2请找到和最小的  个数对 , k(u1,v1) (u2,v2)(uk,vk) 示例1:输入:nums1=[1,7,11],nums2=[2,4,6],k=3......
  • LeetCode 377.组和总和IV
    1.题目:给你一个由 不同 整数组成的数组 nums ,和一个目标整数 target 。请你从 nums 中找出并返回总和为 target 的元素组合的个数。题目数据保证答案符合32位整数范围。 https://leetcode.cn/problems/combination-sum-iv/description/示例1:输入:nums=[1,2,3],targ......
  • LeetCode 518.零钱兑换II
    1.题目:给你一个整数数组 coins 表示不同面额的硬币,另给一个整数 amount 表示总金额。请你计算并返回可以凑成总金额的硬币组合数。如果任何硬币组合都无法凑出总金额,返回 0 。假设每一种面额的硬币有无限个。 题目数据保证结果符合32位带符号整数。 https://leetcode.cn/......