【优选算法必刷100题】第031~32题(前缀和算法):连续数组、矩阵区域和

【优选算法必刷100题】第031~32题(前缀和算法):连续数组、矩阵区域和

🔥艾莉丝努力练剑:个人主页

专栏传送门:《C语言》《数据结构与算法》C/C++干货分享&学习过程记录Linux操作系统编程详解笔试/面试常见算法:从基础到进阶

⭐️为天地立心,为生民立命,为往圣继绝学,为万世开太平


🎬艾莉丝的简介:



🎬艾莉丝的算法专栏简介:


目录

031  连续数组

1.1  解法一:暴力解法

1.2  解法二:前缀和在哈希表中

1.3  算法实现

1.3.1  C++实现

1.3.2  Java实现

1.4  博主手记

032  矩阵区域和

2.1  算法思路:二维前缀和

2.2  算法实现

2.2.1  C++实现

2.2.2  Java实现

2.3  博主手记

结尾


031  连续数组

力扣链接:525. 连续数组

力扣题解链接:前缀和 + 哈希表解决【连续数组】问题

题目描述:

1.1  解法一:暴力解法

暴力解法就是枚举所有的子数组,然后判断子数组是否满足要求,这里不再赘述。

1.2  解法二:前缀和在哈希表中

其实这里稍微转化一下题目,就会变成我们熟悉的题——

(1)本题让我们找出一段连续的区间,和出现的次数相同;

(2)如果将0记为-1,1记为1,问题就变成了找出一段区间,这段区间的和等于0;

(3)于是,这道题就和560.和为K的子数组的思路一样了。

i 为数组中的任意位置,用sum[ i ]表示[0 , 1]区间中的所有元素的和。

想知道最大的【以为结尾的和为的子数组】,就要找到从左往右第一个x1使得[x1 , i]区间内的所有元素的和为0。那么[0 , x1 - 1]区间内的和是不是就是sum[i]了。于是这个问题就变成了——

找到在[0 , i - 1]区间内,第一次出现sum[ i ]的位置即可。

我们不用真的初始化一个前缀和数组,因为我们只关心在 i 位置之前,第一个前缀和等于sum[ i ]的位置。因此,我们仅需用一个哈希表,一边求当前位置的前缀和,一边记录第一次出现该前缀和的位置。 

1.3  算法实现

1.3.1  C++实现

class Solution { public: int findMaxLength(vector<int>& nums) { unordered_map <int,int> hash; // 创建哈希,统计前缀和出现的次数 hash[0] = -1; // 本题存的是下标,默认有一个前缀和为0的情况 // 用变量来标记一下,不用真的创建一个前缀和数组 int sum = 0,ret = 0; // ret标记最终长度 // for(auto x : nums) // 不能用这个万能for,这里是访问下标 for(int i = 0;i < nums.size();++i) { // 计算当前位置的前缀和 sum += nums[i] == 0 ? -1 : 1; // 三目表达式判断一下,是0就-1 if(hash.count(sum)) // 存在:如果找到sum,说明此时hash[sum]里面存了前面那个的下标 ret = max(ret,i - hash[sum]); // 更新长度,i - j即可,j就是hash[sum] // 前面其实相当于判断过了,这里只要一个else就行 else hash[sum] = i; } return ret; } };
时间复杂度:O(n),空间复杂度:O(1)。

1.3.2  Java实现

class Solution { public int findMaxLength(int[] nums) { Map<Integer, Integer> hash = new HashMap<Integer, Integer>(); hash.put(0, -1); // 默认存在⼀个前缀和为 0 的情况 int sum = 0, ret = 0; for (int i = 0; i < nums.length; i++) { sum += (nums[i] == 0 ? -1 : 1); // 计算当前位置的前缀和 if (hash.containsKey(sum)) ret = Math.max(ret, i - hash.get(sum)); else hash.put(sum, i); } return ret; } }
时间复杂度:O(n),空间复杂度:O(1)。

1.4  博主手记

本题整个的思路、算法原理、解题过程博主在纸上推导了一遍,大家可以参考一下手记的推导过程!最好做题的过程中自己也推导一遍!!!自己能够推导很重要!


032  矩阵区域和

力扣链接:1314. 矩阵区域和

力扣题解链接:二维前缀和解决【矩阵前缀和】问题

题目描述:

2.1  算法思路:二维前缀和

二维前缀和的简单应用题,关键就是我们在填写结果矩阵的时候,要找到原矩阵对应区域的【左上
角】以及【右下角】的坐标(这里艾莉丝推荐uu们画图,一目了然)——

(1)左上角坐标:x1 = i - k,y1 = j - k,但是由于会「超过矩阵」的范围,因此需要对0取一个max。因此修正后的坐标为:x1 = max(0,i - k),y1 = max(0,j - k);

(2)右下角坐标:x1 = i + k,y1 = j + k,但是由于会【超过矩阵】的范围,因此需要对m - 1,以及n - 1取一个min。因此修正后的坐标为:x2 = min(m - 1 , i + k),y2 = min(n - 1 , j + k)。

然后我们将求出来的坐标代入到【二维前缀和矩阵】的计算公式上即可,但是要注意下标的映射关
系,这是本题的一个细节问题,在【博主手记】里面艾莉丝举了几个例子,大家可以看看。

2.2  算法实现

2.2.1  C++实现

class Solution { public: vector<vector<int>> matrixBlockSum(vector<vector<int>>& mat, int k) { // 二维前缀和 int m = mat.size(),n = mat[0].size(); // 行和列 // 1、预处理一个前缀和矩阵 vector<vector<int>> dp(m + 1,vector<int>(n + 1)); // m + 1行n + 1列,方便处理边界情况 // 填写矩阵 for(int i = 1;i <= m;i++) for(int j = 1;j <= n;j++) dp[i][j] = dp[i - 1][j] + dp[i][j - 1] - dp[i - 1][j - 1] + mat[i - 1][j - 1]; // mat这里统一是修改后的下标,要-1,-1就是为了处理下标的隐射关系 // 2、使用前缀和矩阵 vector<vector<int>> ret(m,vector<int>(n)); // 跟原始矩阵同等规模 for(int i = 0;i < m;i++) for(int j = 0;j < n;j++) { int x1 = max(0,i - k) + 1,y1 = max(0,j - k) + 1; // +1就是为了在dp表里面直接可以找到 int x2 = min(m - 1,i + k) + 1,y2 = min(n - 1,j + k) + 1; // 结果 ret[i][j] = dp[x2][y2] - dp[x1 - 1][y2] - dp[x2][y1 - 1] + dp[x1 - 1][y1 - 1]; } return ret; } };
时间复杂度:O(n),空间复杂度:O(1)。

2.2.2  Java实现

class Solution { public int[][] matrixBlockSum(int[][] mat, int k) { int m = mat.length, n = mat[0].length; // 1、预处理前缀和矩阵 int[][] dp = new int[m + 1][n + 1]; for (int i = 1; i <= m; i++) for (int j = 1; j <= n; j++) dp[i][j] = dp[i - 1][j] + dp[i][j - 1] - dp[i - 1][j - 1] + mat[i - 1][j - 1]; // 2、使⽤ int[][] ret = new int[m][n]; for (int i = 0; i < m; i++) for (int j = 0; j < n; j++) { int x1 = Math.max(0, i - k) + 1, y1 = Math.max(0, j - k) + 1; int x2 = Math.min(m - 1, i + k) + 1, y2 = Math.min(n - 1, j + k) + 1; ret[i][j] = dp[x2][y2] - dp[x1 - 1][y2] - dp[x2][y1 - 1] + dp[x1 - 1][y1 - 1]; } return ret; } }
时间复杂度:O(n),空间复杂度:O(1)。

2.3  博主手记

本题整个的思路、算法原理、解题过程博主在纸上推导了一遍,大家可以参考一下手记的推导过程!最好做题的过程中自己也推导一遍!!!自己能够推导很重要!


结尾

往期回顾:

【优选算法必刷100题】第029~30题(前缀和算法):寻找数组的中心下标、除自身以外数组的乘积

结语:既然都看到这里啦!就请大佬不要忘记给博主来个“一键四连”哦!

🗡博主在这里放了一只小狗,大家看完了摸摸小狗放松一下吧!🗡

૮₍ ˶ ˊ ᴥ ˋ˶₎ა


Read more

❿⁄₁₁ ⟦ OSCP ⬖ 研记 ⟧ 密码攻击实践 ➱ NTLM哈希传递攻击

❿⁄₁₁ ⟦ OSCP ⬖ 研记 ⟧ 密码攻击实践 ➱ NTLM哈希传递攻击

郑重声明:本文所涉安全技术仅限用于合法研究与学习目的,严禁任何形式的非法利用。因不当使用所导致的一切法律与经济责任,本人概不负责。任何形式的转载均须明确标注原文出处,且不得用于商业目的。 🔋 点赞 | 能量注入 ❤️ 关注 | 信号锁定 🔔 收藏 | 数据归档 ⭐️ 评论 | 保持连接💬 🌌 立即前往 👉晖度丨安全视界🚀 ▶ 信息收集  ▶ 漏洞检测 ▶ 初始立足点  ▶ 权限提升 ▶ 横向移动 ➢ 密码攻击 ➢ NTLM哈希传递攻击🔥🔥🔥 ▶ 报告/分析 ▶ 教训/修复 目录 1.密码破解 1.1 破解Windows哈希实践 1.1.1 NTLM哈希传递攻击概述 1.1.1.1 什么是NTLM哈希传递? 1.1.1.2 攻击应用场景 1.1.

By Ne0inhk

【数学建模】(LeetCode 1227)小鸟回笼/飞机座位问题

题目 有 nnn 只小鸟,各有自己的笼子(编号 1,2,⋯ ,n1, 2, \cdots, n1,2,⋯,n)。第一天,第一只小鸟(编号 1)没有回到自己的笼子(笼 1),而是随机进了其它某个笼子。后续的小鸟每天回来时,如果自己的笼子空着就进自己的笼子,否则从剩下的空笼子中随机选一个。 问:最后一只回笼的小鸟回到自己笼子的概率是多少? 这个问题和经典的飞机座位问题等价(见下),但需要注意的时初始条件不同,下面的问题第一个人位置也是随机的(可能回到自己的位置),而上面小鸟回笼问题则是在没有回到自己的笼子情况下。 有nnn位乘客即将登机,飞机正好有nnn个座位。第一位乘客的票丢了,他随便选了一个座位坐下。剩下的乘客将会:如果他们自己的座位还空着,就坐到自己的座位上;当他们自己的座位被占用时,随机选择其他座位,问: 第nnn位乘客坐在自己的座位上的概率是多少? 解答 **解:**设当所有鸟回到笼子后鸟和笼子编号的映射为 f(n)=m

By Ne0inhk
Flutter 组件 vnlunar 适配鸿蒙 HarmonyOS 实战:高精度农历算法,构建民俗文化日期与节气治理架构

Flutter 组件 vnlunar 适配鸿蒙 HarmonyOS 实战:高精度农历算法,构建民俗文化日期与节气治理架构

欢迎加入开源鸿蒙跨平台社区:https://openharmonycrossplatform.ZEEKLOG.net Flutter 组件 vnlunar 适配鸿蒙 HarmonyOS 实战:高精度农历算法,构建民俗文化日期与节气治理架构 前言 在鸿蒙(OpenHarmony)生态迈向全球化部署、涉及多语言本地化(L10n)及深层文化特性适配的背景下,如何实现准确的阴阳历(农历)转换、二十四节气计算及民俗节日提醒,已成为提升应用“人文温度”与本地化竞争力的核心要素。在鸿蒙设备这类强调分布式时间同步与低功耗常驻显示(AOD)的环境下,如果应用依然依赖简单的查表法或通过网络接口获取农历信息,由于由于闰月计算的复杂性或离线环境限制,极易由于由于计算偏移导致传统节日提醒的误报。 我们需要一种能够实现天文级算法推演、支持高精度节气定位且具备纯 Dart 离线运作能力的历法治理方案。 vnlunar 为 Flutter 开发者引入了标准化的阴阳历转换协议。它不仅支持对天干地支、生肖及闰月的精确解构,更针对东南亚等地区的历法细微差异提供了专项适配。在适配到鸿蒙 HarmonyOS 流程

By Ne0inhk
Flutter 三方库 libsignal 的鸿蒙化适配指南 - 实现 Signal 协议加密通信、双大鼠(Double Ratchet)算法与前向安全性保障

Flutter 三方库 libsignal 的鸿蒙化适配指南 - 实现 Signal 协议加密通信、双大鼠(Double Ratchet)算法与前向安全性保障

欢迎加入开源鸿蒙跨平台社区:https://openharmonycrossplatform.ZEEKLOG.net Flutter 三方库 libsignal 的鸿蒙化适配指南 - 实现 Signal 协议加密通信、双大鼠(Double Ratchet)算法与前向安全性保障 前言 在 Flutter for OpenHarmony 的高度安全通信领域,Signal 协议是目前全球公认的即时通讯加密标准。libsignal 是 Signal 协议的核心 Dart 实现。它能够为鸿蒙应用提供从身份认证到会话加密的全套解决方案,确保每一个字节的通信都具备前向安全性(Forward Secrecy)。本文将深入解析如何在鸿蒙端利用该库构建极致安全的加密通信能力。 一、原理解析 / 概念介绍 1.1 基础原理 Signal 协议的核心在于“双大鼠(Double Ratchet)”算法。它结合了 Diffie-Hellman

By Ne0inhk