当前位置: 首页 > news >正文

每日算法刷题Day38 6.25:leetcode前缀和3道题,用时1h40min

5. 1749.任意子数组和的绝对值的最大值(中等,学习)

1749. 任意子数组和的绝对值的最大值 - 力扣(LeetCode)

思想

1.给你一个整数数组 nums 。一个子数组 [numsl, numsl+1, ..., numsr-1, numsr]和的绝对值abs(numsl + numsl+1 + ... + numsr-1 + numsr)
请你找出 nums和的绝对值 最大的任意子数组(可能为空),并返回该 最大值
abs(x) 定义如下:

  • 如果 x 是负整数,那么 abs(x) = -x
  • 如果 x 是非负整数,那么 abs(x) = x
    2.此题**abs(numsl + numsl+1 + ... + numsr-1 + numsr)转化为abs(s[r+1]-s[l])的最大值,即一个s数组,选取两个元素,让他们两个差的绝对值最大,所以为s数组的最大值mx减去s数组的最小值mn**,因为子数组可能为空(即s[0]=0),所以mx,mn初始化为0
    3.此题转化的含义为:
  • 变成一条曲线,表示和的累计值,然后mx,mn就是最大值点和最小值点,他们直接的区间就是这个子数组和的累计值,表示他们的变化量最大(绝对值)
  • mx的坐标>mn的坐标,表示正的和的最大值
  • mx的坐标<mn的坐标,表示负的和的最小值,绝对值变成最大值
代码

c++:

class Solution {
public:int maxAbsoluteSum(vector<int>& nums) {int n = nums.size();int s = 0, mx = 0,mn = 0; // s[0]=0,所以mx,mn初始值为0,因为子数组可能为空for (int i = 0; i < n; ++i) {s += nums[i];mx = max(mx, s);mn = min(mn, s);}return mx - mn;}
};
6. 2389.和有限的最长子序列(简单,想到二分查找优化)

2389. 和有限的最长子序列 - 力扣(LeetCode)

思想

1.给你一个长度为 n 的整数数组 nums ,和一个长度为 m 的整数数组 queries
返回一个长度为 m 的数组 answer ,其中 answer[i]nums 中 元素之和小于等于 queries[i]子序列最大 长度 。
子序列 是由一个数组删除某些元素(也可以不删除)但不改变剩余元素顺序得到的一个数组。
2.因为是子序列,所以是任意选取的,不要求连续,只要找到一个子序列能够满足条件就可以更新长度,所以贪心思想元素越小越好,所以先对nums数组排序,然后求得前缀和s数组,接下来就是寻找最后一个满足s[tmp]<=queries[i]的下标tmp,因为s数组是有序的,所以可以将顺序查找优化为二分查找,因为s[tmp]对应nums[0...tmp-1]数组和,长度就为tmp

代码

c++:

class Solution {
public:vector<int> answerQueries(vector<int>& nums, vector<int>& queries) {int n = nums.size(), m = queries.size();sort(nums.begin(), nums.end());vector<int> s(n + 1);s[0] = 0;for (int i = 0; i < n; ++i) {s[i + 1] = s[i] + nums[i];}vector<int> res;for (int i = 0; i < m; ++i) {int j = 0;while (j + 1 < n + 1 && s[j + 1] <= queries[i])++j;res.emplace_back(j);}return res;}
};

二分优化:

class Solution {
public:vector<int> answerQueries(vector<int>& nums, vector<int>& queries) {int n = nums.size(), m = queries.size();sort(nums.begin(), nums.end());vector<int> s(n + 1);s[0] = 0;for (int i = 0; i < n; ++i) {s[i + 1] = s[i] + nums[i];}vector<int> res;for (int i = 0; i < m; ++i) {int left = 0, right = n, tmp = 0;while (left <= right) {int mid = left + ((right - left) >> 1);// 找到满足条件的,更新答案,找更大的if (s[mid] <= queries[i]) {tmp = mid;left = mid + 1;} elseright = mid - 1;}// s[tmp]:nums[0...tmp-1]长度为tmpres.emplace_back(tmp);}return res;}
};
7. 3361.两个字符串的切换距离(中等,学习环形延长一倍变成非环形)

3361. 两个字符串的切换距离 - 力扣(LeetCode)

思想

1.给你两个长度相同的字符串 st ,以及两个整数数组 nextCostpreviousCost
一次操作中,你可以选择 s 中的一个下标 i ,执行以下操作 之一

  • s[i] 切换为字母表中的下一个字母,如果 s[i] == 'z' ,切换后得到 'a' 。操作的代价为 nextCost[j] ,其中 j 表示 s[i] 在字母表中的下标。
  • s[i] 切换为字母表中的上一个字母,如果 s[i] == 'a' ,切换后得到 'z' 。操作的代价为 previousCost[j] ,其中 js[i] 在字母表中的下标。
    切换距离 指的是将字符串 s 变为字符串 t最少 操作代价总和。
    请你返回从 st切换距离
    2.就是提前计算出nextCost和previousCost的前缀和数组,然后分类讨论考虑时要考虑清楚区间范围是什么,然后决定前缀和数组的下标
    3.字母表是环形的,可以把前缀和数组延长一倍,可以大大简化讨论
代码

c++:

class Solution {
public:long long shiftDistance(string s, string t, vector<int>& nextCost,vector<int>& previousCost) {long long res = 0;int n = s.size();vector<long long> sNext(27);vector<long long> sPre(27);sNext[0] = 0;sPre[0] = 0;for (int i = 0; i < 26; ++i) {sNext[i + 1] = sNext[i] + nextCost[i];sPre[i + 1] = sPre[i] + previousCost[i];}for (int i = 0; i < n; ++i) {int ids = s[i] - 'a', idt = t[i] - 'a';long long mn = 0;if (ids < idt) {// 向后:nextCost[ids...idt-1],向前排除:preCost[ids+1...idt]mn = min(sNext[idt] - sNext[ids],sPre[26] - (sPre[idt + 1] - sPre[ids + 1]));}// 向前:PreCost[idt+1..ids],向后排除:nextCost[idt...ids-1]elsemn = min(sPre[ids + 1] - sPre[idt + 1],sNext[26] - (sNext[ids] - sNext[idt]));res += mn;}return res;}
};

延长一倍优化循环数组:

class Solution {
public:long long shiftDistance(string s, string t, vector<int>& nextCost,vector<int>& previousCost) {long long res = 0;int n = s.size();const int SIGMA = 26;vector<long long> sNext(2 * SIGMA + 1, 0);vector<long long> sPre(2 * SIGMA + 1, 0);sNext[0] = 0;sPre[0] = 0;for (int i = 0; i < 2 * SIGMA; ++i) {sNext[i + 1] = sNext[i] + nextCost[i % SIGMA];sPre[i + 1] = sPre[i] + previousCost[i % SIGMA];}for (int i = 0; i < n; ++i) {int x = s[i] - 'a', y = t[i] - 'a';// 向后是ids->idt-1,所以s数组是ids->idt,向前是idt+1->ids,所以s数组是idt+1->ids+1// sNext只有y<x时,y才加;sPre只有x<y时,x才加res += min(sNext[y < x ? y + SIGMA : y] - sNext[x],sPre[(x < y ? x + SIGMA : x) + 1] - sPre[y + 1]);}return res;}
};
http://www.lqws.cn/news/538453.html

相关文章:

  • 第七章:总结
  • 【RabbitMQ】多系统下的安装配置与编码使用(python)
  • Spring Task定时任务详解与实战应用
  • java中的anyMatch和allMatch方法
  • OSEK/VDX OS ISO17356-3,【1】规范概述
  • SpringBoot项目快速开发框架JeecgBoot——Web处理!
  • linux cp与mv那个更可靠
  • MySQL5.7和8.0 破解root密码
  • 快速傅里叶变换(FFT)是什么?
  • python中学物理实验模拟:斜面受力分析
  • 圆周期性显示和消失——瞬态实现(CAD c#二次开发、插件定制)
  • Nordic nRF54L15 SoC对包含电池监测、中断处理和电源轨控制的定制 nPM1300 示例
  • springcloud 尚硅谷 看到9开头
  • 华为云鸿蒙应用入门级开发者认证 实验(HCCDA-HarmonyOS Cloud Apps)
  • 玄机抽奖Spring Web项目
  • Maven Javadoc 插件使用详解
  • [论文阅读]RaFe: Ranking Feedback Improves Query Rewriting for RAG
  • 解决uniapp vue3版本封装组件后:deep()样式穿透不生效的问题
  • react-嵌套路由 二级路由
  • 事件循环(Event Loop)机制对比:Node.js vs 浏览器​
  • python+requests接口自动化测试
  • 大脑感官:视觉系统中将感观信息转换为神经信号
  • @Autowired 和 @Resource 有什么区别?
  • Java常用设计模式详解
  • linux网络编程socket套接字
  • 【论文阅读】--Instruction Backdoor Attacks Against Customized LLMs
  • Neo4j2.0.1桌面端使用教程(简化版)
  • MySQL 锁的分类
  • WinAppDriver 自动化测试:C#篇
  • EMQ X Broker 配置HTTP 的外部鉴权接口