Java详解LeetCode 热题 100(17):LeetCode 41. 缺失的第一个正数(First Missing Positive)详解
文章目录
- 1. 题目描述
- 2. 理解题目
- 3. 解法一:排序法(不满足题目要求)
- 3.1 思路
- 3.2 Java代码实现
- 3.3 代码详解
- 3.4 复杂度分析
- 3.5 不足之处
- 4. 解法二:哈希表法
- 4.1 思路
- 4.2 Java代码实现
- 4.3 代码详解
- 4.4 复杂度分析
- 4.5 不足之处
- 5. 解法三:原地哈希法(最优解)
- 5.1 思路
- 5.2 Java代码实现
- 5.3 代码详解
- 5.4 复杂度分析
- 5.5 适用场景
- 6. 详细步骤分析与示例跟踪
- 6.1 示例跟踪:原地哈希法
- 6.2 复杂示例跟踪
- 7. 常见错误与优化
- 7.1 常见错误
- 7.2 优化技巧
- 8. 解法对比
- 9. 扩展题目与应用
- 9.1 相关题目
- 9.2 实际应用
- 10. 实际应用场景
- 10.1 数据库ID管理
- 10.2 网络协议中的丢包检测
- 10.3 内存分配器
- 11. 完整的 Java 解决方案
- 12. 总结与技巧
- 12.1 解题要点
- 12.2 学习收获
- 12.3 面试技巧
- 13. 参考资料
1. 题目描述
给你一个未排序的整数数组 nums
,请你找出其中没有出现的最小的正整数。
请你实现时间复杂度为 O(n)
并且只使用常数级别额外空间的解决方案。
示例 1:
输入:nums = [1,2,0]
输出:3
解释:数组中包含了 1 和 2,所以缺失的最小正整数是 3
示例 2:
输入:nums = [3,4,-1,1]
输出:2
解释:数组中包含了 3、4 和 1,所以缺失的最小正整数是 2
示例 3:
输入:nums = [7,8,9,11,12]
输出:1
解释:数组中不包含任何正数 1~5,所以缺失的最小正整数是 1
提示:
- 1 <= nums.length <= 5 * 10^5
- -2^31 <= nums[i] <= 2^31 - 1
2. 理解题目
这道题要求我们找出数组中缺失的第一个正整数。具体来说:
- 我们要找的是最小的未出现在数组中的正整数
- 正整数意味着我们只考虑大于0的整数(即1, 2, 3, …)
- 如果数组中包含了1到n的所有正整数,那么答案就是n+1
关键点:
- 数组可能包含负数、0或重复元素,这些都不是我们要找的答案
- 数组可能不包含任何正整数,此时答案是1
- 时间复杂度要求O(n),空间复杂度要求O(1),这意味着我们不能使用额外的数据结构如哈希表(除非能够证明它只使用常数空间)
这个问题的难点在于时空复杂度的限制,让我们不能使用排序(O(nlogn))或额外的数组/哈希表来跟踪数字的出现情况。
3. 解法一:排序法(不满足题目要求)
3.1 思路
虽然这种解法不满足题目的时间复杂度要求,但它是最直观的方法,适合初学者理解问题:
- 对数组进行排序
- 遍历排序后的数组,找到第一个不连续的正整数
3.2 Java代码实现
class Solution {public int firstMissingPositive(int[] nums) {// 对数组进行排序Arrays.sort(nums);// 记录期望的下一个正整数,初始为1int expectedNum = 1;// 遍历排序后的数组for (int num : nums) {// 如果当前数字等于期望的数字,更新期望值if (num == expectedNum) {expectedNum++;} // 如果当前数字大于期望的数字,说明找到了缺失的正整数else if (num > expectedNum) {return expectedNum;}// 忽略小于等于0或重复的数字}// 如果数组中的所有正整数都是连续的,返回最后一个正整数+1return expectedNum;}
}
3.3 代码详解
// 对数组进行排序
Arrays.sort(nums);
- 使用Java内置的排序函数对数组进行排序,这样相同的数字会挨在一起,正数会按从小到大排列
// 记录期望的下一个正整数,初始为1
int expectedNum = 1;
- 我们从1开始检查,因为1是最小的正整数
// 遍历排序后的数组
for (int num : nums) {// 如果当前数字等于期望的数字,更新期望值if (num == expectedNum) {expectedNum++;} // 如果当前数字大于期望的数字,说明找到了缺失的正整数else if (num > expectedNum) {return expectedNum;}// 忽略小于等于0或重复的数字
}
- 遍历排序后的数组,跟踪下一个预期出现的正整数
- 如果当前数字等于预期数字,说明这个数字存在,我们增加预期值
- 如果当前数字大于预期数字,说明预期数字不在数组中,我们找到了答案
- 如果当前数字小于预期数字(包括负数、0或重复值),我们直接忽略它
// 如果数组中的所有正整数都是连续的,返回最后一个正整数+1
return expectedNum;
- 如果我们遍历完整个数组都没有返回,说明数组中包含了从1到某个数的所有正整数,缺失的第一个正整数就是这个数加1
3.4 复杂度分析
- 时间复杂度:O(n log n),主要由排序操作决定。
- 空间复杂度:O(1) 或 O(log n),取决于排序算法的实现(一些排序算法可能需要O(log n)的栈空间)。
3.5 不足之处
虽然这种方法很直观,但它的时间复杂度是O(n log n),不满足题目要求的O(n)。此外,某些排序算法可能需要额外的空间。因此,我们需要探索更高效的解法。
4. 解法二:哈希表法
4.1 思路
利用哈希表记录数组中存在的正整数,然后从1开始检查,找到第一个不在哈希表中的正整数:
- 创建一个哈希表(HashSet)
- 将数组中所有大于0的整数加入哈希表
- 从1开始遍历正整数,找到第一个不在哈希表中的数
4.2 Java代码实现
class Solution {public int firstMissingPositive(int[] nums) {// 创建哈希表存储数组中的正整数Set<Integer> positiveNums = new HashSet<>();// 将所有正整数加入哈希表for (int num : nums) {if (num > 0) {positiveNums.add(num);}}// 从1开始检查每个正整数是否存在int missingPositive = 1;while (positiveNums.contains(missingPositive)) {missingPositive++;}return missingPositive;}
}
4.3 代码详解
// 创建哈希表存储数组中的正整数
Set<Integer> positiveNums = new HashSet<>();
- 使用HashSet作为哈希表,它能够以O(1)的时间复杂度检查一个数字是否存在
// 将所有正整数加入哈希表
for (int num : nums) {if (num > 0) {positiveNums.add(num);}
}
- 遍历数组,只将正整数(大于0的数)添加到哈希表中
- 负数和0不是我们要找的答案,可以直接忽略
// 从1开始检查每个正整数是否存在
int missingPositive = 1;
while (positiveNums.contains(missingPositive)) {missingPositive++;
}
- 从最小的正整数1开始,检查哈希表中是否包含该数字
- 如果包含,则增加计数器,继续检查下一个正整数
- 一旦找到一个不在哈希表中的正整数,就找到了答案
4.4 复杂度分析
- 时间复杂度:O(n),我们需要遍历数组一次将元素加入哈希表,然后在最坏情况下检查从1到n的每个正整数。
- 空间复杂度:O(n),我们需要一个哈希表来存储数组中的正整数,最坏情况下存储n个不同的正整数。
4.5 不足之处
这种方法的时间复杂度满足题目要求,但空间复杂度是O(n),不满足O(1)的要求。因此,我们需要继续寻找更优的解法。
5. 解法三:原地哈希法(最优解)
5.1 思路
这种方法的核心思想是利用数组本身作为哈希表,通过将每个数值放置到对应的位置上,然后再遍历一次数组找出第一个不符合要求的位置。具体步骤如下:
- 遍历数组,将每个在范围[1, n]内的正整数放到对应的位置上(nums[i-1] = i)
- 再次遍历数组,找到第一个不满足nums[i-1] = i的位置,该位置对应的正整数i就是缺失的第一个正整数
关键思想:对于长度为n的数组,如果数组包含了1到n的所有正整数,那么缺失的第一个正整数就是n+1。因此,我们只需要关心范围在[1, n]内的正整数。
5.2 Java代码实现
class Solution {public int firstMissingPositive(int[] nums) {int n = nums.length;// 第一次遍历,将每个数放到对应的位置上for (int i = 0; i < n; i++) {// 当前值在[1,n]范围内,且还未放到正确位置上while (nums[i] > 0 && nums[i] <= n && nums[nums[i] - 1] != nums[i]) {// 交换nums[i]和nums[nums[i]-1]int temp = nums[nums[i] - 1];nums[nums[i] - 1] = nums[i];nums[i] = temp;}}// 第二次遍历,找到第一个不符合要求的位置for (int i = 0; i < n; i++) {if (nums[i] != i + 1) {return i + 1;}}// 如果数组中包含了1到n的所有正整数,则返回n+1return n + 1;}
}
5.3 代码详解
int n = nums.length;
- 获取数组长度,这是我们寻找缺失正整数的范围上限
// 第一次遍历,将每个数放到对应的位置上
for (int i = 0; i < n; i++) {// 当前值在[1,n]范围内,且还未放到正确位置上while (nums[i] > 0 && nums[i] <= n && nums[nums[i] - 1] != nums[i]) {// 交换nums[i]和nums[nums[i]-1]int temp = nums[nums[i] - 1];nums[nums[i] - 1] = nums[i];nums[i] = temp;}
}
- 遍历数组,尝试将每个数放到其对应的位置上:
- 对于值为k的元素,其正确位置应该是索引k-1处(因为数组索引从0开始)
- 条件
nums[i] > 0 && nums[i] <= n
确保我们只处理范围在[1,n]内的正整数 - 条件
nums[nums[i] - 1] != nums[i]
确保当前数字还未放置到正确位置,避免无限循环 - 使用
while
循环而不是if
是因为交换后的新值也可能需要放置到正确位置
// 第二次遍历,找到第一个不符合要求的位置
for (int i = 0; i < n; i++) {if (nums[i] != i + 1) {return i + 1;}
}
- 再次遍历数组,检查每个位置上的数字是否正确:
- 索引i处应该是数字i+1
- 如果发现某个位置上的数字不是预期值,说明该预期值在数组中缺失,返回这个值
// 如果数组中包含了1到n的所有正整数,则返回n+1
return n + 1;
- 如果遍历完整个数组都没有返回,说明数组中包含了1到n的所有正整数,缺失的第一个正整数就是n+1
5.4 复杂度分析
-
时间复杂度:O(n)
- 第一次遍历中,虽然有一个嵌套的while循环,但每个元素最多只会被交换一次到它的正确位置,所以总的交换操作不会超过n次
- 第二次遍历是一个简单的O(n)操作
- 因此,总的时间复杂度是O(n)
-
空间复杂度:O(1)
- 我们只使用了有限的变量,没有额外的数据结构,满足题目的空间要求
5.5 适用场景
这种解法是该问题在O(n)时间和O(1)空间约束下的最优解,适用于所有包含此类约束的缺失正整数问题。
6. 详细步骤分析与示例跟踪
让我们通过一个具体例子来跟踪原地哈希算法的执行过程,以加深理解。
6.1 示例跟踪:原地哈希法
输入:nums = [3,4,-1,1]
初始状态:
- 数组:
[3,4,-1,1]
- 长度 n = 4
第一次遍历(原地哈希过程):
-
i = 0, nums[0] = 3
- 3应该在索引2的位置(3-1=2)
- 交换nums[0]和nums[2]: [3,4,-1,1] → [-1,4,3,1]
- 现在nums[0] = -1,不在[1,n]范围内,不需要再交换
-
i = 1, nums[1] = 4
- 4应该在索引3的位置(4-1=3)
- 交换nums[1]和nums[3]: [-1,4,3,1] → [-1,1,3,4]
- 现在nums[1] = 1,它应该在索引0的位置
- 交换nums[1]和nums[0]: [-1,1,3,4] → [1,-1,3,4]
- 现在nums[1] = -1,不在[1,n]范围内,不需要再交换
-
i = 2, nums[2] = 3
- 3已经在正确的位置上(索引2),不需要交换
-
i = 3, nums[3] = 4
- 4已经在正确的位置上(索引3),不需要交换
经过第一次遍历后,数组变为:[1,-1,3,4]
第二次遍历(查找缺失的正整数):
-
i = 0, nums[0] = 1
- 1应该在索引0的位置(1-1=0)✓
- 符合预期,继续检查下一个位置
-
i = 1, nums[1] = -1
- 索引1处应该是数字2(1+1=2)✗
- nums[1] ≠ 2,找到了缺失的第一个正整数:2
因此,返回结果:2
6.2 复杂示例跟踪
输入:nums = [7,8,9,11,12]
初始状态:
- 数组:
[7,8,9,11,12]
- 长度 n = 5
第一次遍历(原地哈希过程):
-
i = 0, nums[0] = 7
- 7超出了数组长度5,不在[1,5]范围内,不需要交换
-
i = 1, nums[1] = 8
- 8超出了数组长度5,不在[1,5]范围内,不需要交换
-
i = 2, nums[2] = 9
- 9超出了数组长度5,不在[1,5]范围内,不需要交换
-
i = 3, nums[3] = 11
- 11超出了数组长度5,不在[1,5]范围内,不需要交换
-
i = 4, nums[4] = 12
- 12超出了数组长度5,不在[1,5]范围内,不需要交换
经过第一次遍历后,数组保持不变:[7,8,9,11,12]
第二次遍历(查找缺失的正整数):
- i = 0, nums[0] = 7
- 索引0处应该是数字1(0+1=1)✗
- nums[0] ≠ 1,找到了缺失的第一个正整数:1
因此,返回结果:1
7. 常见错误与优化
7.1 常见错误
-
未处理负数和零:
// 错误:忽略了负数和零的处理 while (nums[i] != i + 1) {// 交换逻辑 }// 正确:只处理范围在[1,n]内的正整数 while (nums[i] > 0 && nums[i] <= n && nums[nums[i] - 1] != nums[i]) {// 交换逻辑 }
-
未考虑重复元素:
// 错误:没有检查目标位置是否已经有正确的值 while (nums[i] > 0 && nums[i] <= n) {// 交换逻辑,可能导致无限循环 }// 正确:避免无限循环 while (nums[i] > 0 && nums[i] <= n && nums[nums[i] - 1] != nums[i]) {// 交换逻辑 }
-
交换逻辑错误:
// 错误:交换顺序不正确 nums[i] = nums[nums[i] - 1]; nums[nums[i] - 1] = nums[i]; // 这里的nums[i]已经被更改// 正确:使用临时变量进行交换 int temp = nums[nums[i] - 1]; nums[nums[i] - 1] = nums[i]; nums[i] = temp;
-
循环条件错误:
// 错误:使用if而不是while for (int i = 0; i < n; i++) {if (nums[i] > 0 && nums[i] <= n && nums[nums[i] - 1] != nums[i]) {// 交换逻辑,但只交换一次} }// 正确:使用while循环确保所有元素都被放到正确位置 for (int i = 0; i < n; i++) {while (nums[i] > 0 && nums[i] <= n && nums[nums[i] - 1] != nums[i]) {// 交换逻辑} }
7.2 优化技巧
-
提前检查特殊情况:
// 优化:快速检查1是否存在 boolean containsOne = false; for (int num : nums) {if (num == 1) {containsOne = true;break;} }if (!containsOne) {return 1; // 如果不包含1,直接返回 }
-
预处理数组:
// 优化:将所有无效值(负数、零和大于n的数)替换为统一的值 for (int i = 0; i < n; i++) {if (nums[i] <= 0 || nums[i] > n) {nums[i] = n + 1; // 使用n+1作为标记值} }
-
使用标记法代替交换:
另一种原地哈希的变种,通过修改数组元素的符号来标记存在的数字,避免了频繁的交换操作。class Solution {public int firstMissingPositive(int[] nums) {int n = nums.length;// 将所有非正数替换为n+1for (int i = 0; i < n; i++) {if (nums[i] <= 0) {nums[i] = n + 1;}}// 使用负号标记存在的数字for (int i = 0; i < n; i++) {int num = Math.abs(nums[i]);if (num <= n) {nums[num - 1] = -Math.abs(nums[num - 1]);}}// 找到第一个正数的位置for (int i = 0; i < n; i++) {if (nums[i] > 0) {return i + 1;}}return n + 1;} }
8. 解法对比
解法 | 时间复杂度 | 空间复杂度 | 优点 | 缺点 |
---|---|---|---|---|
排序法 | O(n log n) | O(1) 或 O(log n) | 简单易懂,容易实现 | 不满足O(n)时间复杂度要求 |
哈希表法 | O(n) | O(n) | 思路清晰,容易理解 | 不满足O(1)空间复杂度要求 |
原地哈希法(交换) | O(n) | O(1) | 满足所有复杂度要求 | 实现较复杂,边界条件多 |
原地哈希法(标记) | O(n) | O(1) | 满足所有复杂度要求,减少交换操作 | 修改了原始数据,可能不适合某些场景 |
最优解:原地哈希法,无论是交换版本还是标记版本,都满足题目的所有时空复杂度要求。选择哪种具体实现取决于对原始数据修改的限制和个人偏好。
9. 扩展题目与应用
9.1 相关题目
-
LeetCode 268. 丢失的数字(Missing Number):
- 给定一个包含 [0, n] 范围内不重复整数的数组 nums,找出丢失的那个数字
- 与本题类似,但范围包括0,且保证数组中不重复
-
LeetCode 448. 找到所有数组中消失的数字(Find All Numbers Disappeared in an Array):
- 给定一个范围在 [1, n] 内的整数数组,找出数组中所有没有出现的数字
- 与本题类似,但需要找出所有缺失的数字而不仅仅是第一个
-
LeetCode 287. 寻找重复数(Find the Duplicate Number):
- 给定一个包含 n + 1 个整数的数组 nums,所有数字都在 [1, n] 范围内,找出唯一重复的数字
- 与本题操作数组的方式类似,但目标不同
9.2 实际应用
-
数据完整性检查:
- 验证一组连续ID是否有缺失
- 检查系统中是否有未分配的资源ID
-
网络包序列号检测:
- 检测网络传输中是否有丢失的数据包
- 实现可靠的数据传输协议
-
分布式系统中的ID分配:
- 在分布式环境中高效地分配唯一ID
- 回收未使用的ID以避免ID耗尽
-
内存管理:
- 在内存分配器中找出第一个可用的内存块
- 实现高效的内存回收和重用机制
10. 实际应用场景
10.1 数据库ID管理
在数据库系统中,通常需要为记录分配唯一的ID。当某些记录被删除后,可能需要重用这些ID以避免ID过大导致的性能问题。这时可以使用缺失的第一个正整数算法快速找出可重用的最小ID:
public class DatabaseIdManager {private Set<Integer> usedIds = new HashSet<>();private int maxId = 0;public int allocateId() {// 找出当前使用的ID中缺失的第一个正整数int[] ids = usedIds.stream().mapToInt(Integer::intValue).toArray();int missingId = findFirstMissingPositive(ids);if (missingId <= maxId) {// 重用之前的IDusedIds.add(missingId);return missingId;} else {// 分配新IDmaxId++;usedIds.add(maxId);return maxId;}}public void releaseId(int id) {usedIds.remove(id);}private int findFirstMissingPositive(int[] nums) {// 实现原地哈希算法// ...}
}
10.2 网络协议中的丢包检测
在网络传输中,数据包通常带有序列号。接收方可以使用缺失的第一个正整数算法来检测是否有丢失的数据包,以便请求重传:
public class PacketLossDetector {public int detectFirstLostPacket(int[] receivedPacketSequences) {// 对接收到的序列号排序(如果未排序)Arrays.sort(receivedPacketSequences);// 找出缺失的第一个序列号int expectedSeq = 1;for (int seq : receivedPacketSequences) {if (seq == expectedSeq) {expectedSeq++;} else if (seq > expectedSeq) {return expectedSeq; // 找到缺失的第一个序列号}}return expectedSeq; // 下一个期望的序列号}
}
10.3 内存分配器
在内存管理中,可以使用缺失的第一个正整数算法快速找出第一个可用的内存块索引:
public class MemoryAllocator {private boolean[] usedBlocks; // 标记内存块是否被使用public MemoryAllocator(int totalBlocks) {usedBlocks = new boolean[totalBlocks];}public int allocate() {// 找到第一个未使用的内存块for (int i = 0; i < usedBlocks.length; i++) {if (!usedBlocks[i]) {usedBlocks[i] = true;return i + 1; // 返回块ID(从1开始)}}throw new OutOfMemoryError("No available memory blocks");}public void free(int blockId) {if (blockId > 0 && blockId <= usedBlocks.length) {usedBlocks[blockId - 1] = false;}}
}
11. 完整的 Java 解决方案
以下是结合了最佳实践和优化技巧的完整解决方案:
class Solution {public int firstMissingPositive(int[] nums) {if (nums == null || nums.length == 0) {return 1;}int n = nums.length;// 优化1: 快速检查1是否存在boolean contains1 = false;for (int num : nums) {if (num == 1) {contains1 = true;break;}}if (!contains1) {return 1;}// 优化2: 将所有无效值替换为1// 1已经存在,且我们只关心[1,n]范围内的数,// 所以用1替换所有负数、0和大于n的数不会影响结果for (int i = 0; i < n; i++) {if (nums[i] <= 0 || nums[i] > n) {nums[i] = 1;}}// 使用原地哈希法(标记版本)for (int i = 0; i < n; i++) {int val = Math.abs(nums[i]);if (val == n) {// 特殊处理n,使用索引0来标记nums[0] = -Math.abs(nums[0]);} else {// 使用负号标记值为val的数字存在nums[val] = -Math.abs(nums[val]);}}// 找到第一个正数的位置for (int i = 1; i < n; i++) {if (nums[i] > 0) {return i;}}// 检查n是否存在if (nums[0] > 0) {return n;}// 如果1到n都存在,返回n+1return n + 1;}
}
这个解决方案结合了多种优化技巧:
- 提前检查1是否存在,如果不存在直接返回1
- 将所有无效值(负数、0和大于n的数)统一替换为1
- 使用标记法而不是交换法来减少操作
- 巧妙利用数组索引来标记存在的值
12. 总结与技巧
12.1 解题要点
-
理解问题约束:这类问题的关键是理解时间和空间复杂度的严格限制。
-
利用数值范围:对于长度为n的数组,我们只需要关心[1,n+1]范围内的正整数。
-
原地哈希思想:将数组自身作为哈希表,通过交换或标记来记录数字的存在情况。
-
处理边界情况:注意处理负数、零、重复值和超出范围的数字。
12.2 学习收获
通过学习本题,你可以掌握:
- 原地哈希的思想和实现
- 如何在严格的空间复杂度约束下解决问题
- 处理数组元素交换和标记的技巧
- 分析问题并识别关键约束的能力
12.3 面试技巧
如果在面试中遇到此类问题:
- 先提出简单直观的解法(如排序法或哈希表法)
- 分析其复杂度,指出不满足题目要求的地方
- 引入原地哈希的思想,解释如何利用数组本身作为哈希表
- 讨论实现细节和边界条件
- 提供不同版本的原地哈希实现(交换法vs标记法)及其优缺点
记住,面试官通常更关注你的思考过程和问题分析能力,而不仅仅是最终的解法。
13. 参考资料
- LeetCode 官方题解:缺失的第一个正数
- LeetCode 题目链接:缺失的第一个正数
相关文章:
Java详解LeetCode 热题 100(17):LeetCode 41. 缺失的第一个正数(First Missing Positive)详解
文章目录 1. 题目描述2. 理解题目3. 解法一:排序法(不满足题目要求)3.1 思路3.2 Java代码实现3.3 代码详解3.4 复杂度分析3.5 不足之处 4. 解法二:哈希表法4.1 思路4.2 Java代码实现4.3 代码详解4.4 复杂度分析4.5 不足之处 5. 解…...
Kafka消息路由分区机制深度解析:架构设计与实现原理
一、消息路由系统的核心架构哲学 1.1 分布式系统的三元悖论 在分布式消息系统的设计过程中,架构师需要平衡三个核心诉求:数据一致性、系统可用性和分区容忍性。Kafka的分区路由机制本质上是对CAP定理的实践解: 一致性维度:通过…...
用C语言实现了——一个基于顺序表的插入排序演示系统
一、知识要点、 插入排序是一种简单直观的排序算法,它的工作方式类似于我们整理扑克牌。 基本原理: 插入排序通过构建有序序列来工作。它每次从无序序列中取出一个元素,然后将其插入到已排序序列的适当位置。这个过程重复进行,…...
linux libdbus使用案例
以下是一个基于 Linux libdbus 的详细指南,包含服务端和客户端的完整代码示例,涵盖 方法调用、信号发送 和 异步消息处理。libdbus 是 D-Bus 的底层 C 库,直接操作 D-Bus 协议,适合需要精细控制的场景。 1. libdbus 的核心机制 连接管理:通过 dbus_bus_get 连接系统总线或…...
Apple Vision Pro空间视频创作革命:从180度叙事到沉浸式语法的重构——《Adventure》系列幕后技术深度解析
🌌 引言:沉浸式媒体的“语法实验室” Apple Vision Pro的推出标志着空间计算时代的到来,而《Adventure》系列作为其原生内容标杆,正在成为沉浸式叙事的“语法实验室”。导演Charlotte Mikkelborg与播客主持人Kent Bye的对话揭示了这一领域的技术突破、创作挑战与行业生态…...
[特殊字符] 苍穹外卖项目中的 WebSocket 实战:实现来单与催单提醒功能
🚀 苍穹外卖项目中的 WebSocket 实战:实现来单与催单提醒功能 在现代 Web 应用中,实时通信成为提升用户体验的关键技术之一。WebSocket 作为一种在单个 TCP 连接上进行全双工通信的协议,被广泛应用于需要实时数据交换的场景&#…...
【C/C++】深度解析C++ Allocator:优化内存管理的关键
文章目录 深度解析C Allocator:优化内存管理的关键1 默认 std::allocator2 自定义 Allocator3 自定义 Allocator 的实现3.1 基本结构3.2 使用自定义 Allocator 4 关键特性详解4.1 rebind 机制4.2 状态化 Allocator 5 应用示例:内存池 Allocator5.1 简单内…...
gitlab+portainer 实现Ruoyi Vue前端CI/CD
1. 场景 最近整了一个Ruoyi Vue 项目,需要实现CICD,经过一番坎坷,最终达成,现将技术要点和踩坑呈现。 具体操作流程和后端大同小异,后端操作参考连接如下: https://blog.csdn.net/leinminna/article/detai…...
CAPL编程系列_04
1_ 测试模块TestModule:基本使用 1)在Simulation Setup 中创建并配置 Test Module节点 2)编写测试脚本 【1】测试用例函数(testcase):实现具体测试逻辑 【2】主测试函数(Main Test)&…...
Weblogic SSRF漏洞复现(CVE-2014-4210)【vulhub靶场】
漏洞概述: Weblogic中存在一个SSRF漏洞,利用该漏洞可以发送任意HTTP请求,进而攻击内网中redis、fastcgi等脆弱组件。 漏洞形成原因: WebLogic Server 的 UDDI 组件(uddiexplorer.war)中的 SearchPublicR…...
科技的成就(六十八)
623、杰文斯悖论 杰文斯悖论是1865年经济学家威廉斯坦利杰文斯提出的一悖论:当技术进步提高了效率,资源消耗不仅没有减少,反而激增。例如,瓦特改良的蒸汽机让煤炭燃烧更加高效,但结果却是煤炭需求飙升。 624、代码混…...
知从科技闪耀2025上海车展:以创新驱动未来出行新篇章
上海,2025年4月23日——全球汽车科技领域的年度盛会——2025上海国际汽车工业展览会(简称“上海车展”)于5月2日圆满落幕。作为智能汽车软件与系统解决方案的领军企业,知从科技受邀参展,并在活动期间全方位展示了其在智…...
【iOS安全】Dopamine越狱 iPhone X iOS 16.6 (20G75) | 解决Jailbreak failed with error
Dopamine越狱 iPhone X iOS 16.6 (20G75) Dopamine兼容设备 参考:https://www.bilibili.com/opus/977469285985157129 A9 - A11(iPhone6s-X):iOS15.0-16.6.1 A12-A14(iPhoneXR-12PM…...
医疗数据迁移质量与效率的深度研究:三维六阶框架与实践创新
引言 随着医疗信息化建设的深入推进,医疗数据作为医疗机构的核心资产,其价值与日俱增。在医院信息系统升级、迁移或整合过程中,数据迁移的质量与效率直接关系到医疗服务的连续性、患者信息的安全性以及医院运营的稳定性。传统数据迁移方法往往面临时间长、风险高、成本大等…...
[6-8] 编码器接口测速 江协科技学习笔记(7个知识点)
1 2 在STM32微控制器的定时器模块中,CNT通常指的是定时器的计数器值。以下是CNT是什么以及它的用途: 是什么: • CNT:代表定时器的当前计数值。在STM32中,定时器从0开始计数,直到达到预设的自动重装载值&am…...
java类加载阶段与双亲委派机制
java执行过程:.java->.class->然后被jvm加载解释执行。 一、类加载机制的三个阶段 加载(Loading) 任务:通过类的全限定名获取二进制字节流(如从文件系统、网络等),将字节流转换为方…...
医院网络安全托管服务(MSS)深度解读与实践路径
医疗行业网络安全挑战与MSS的应运而生 医疗行业在数智化转型的过程中面临着前所未有的网络安全挑战。根据2025年的最新数据,医疗行业将面临大量网络攻击,其中高达91%与勒索软件有关,且45%的数据泄露事件源于第三方供应商。医疗机构的平均数据…...
计算图存储采用矩阵吗,和张量关系
计算图存储采用矩阵吗,和张量关系 计算图的存储方式与张量的关系 一、计算图的存储方式 计算图(Computational Graph)是一种用于描述数学运算的有向无环图(DAG),其节点代表运算(如加减乘除、矩阵乘法、激活函数等),边代表运算的输入和输出(通常是张量)。计算图的…...
RPA 自动化实现自动发布
📕我是廖志伟,一名Java开发工程师、《Java项目实战——深入理解大型互联网企业通用技术》(基础篇)、(进阶篇)、(架构篇)清华大学出版社签约作家、Java领域优质创作者、CSDN博客专家、…...
博途软件直接寻址AMS348i读取位置值详解
一、AMS348i简介 AMS348i是一种高性能绝对值编码器,常用于工业自动化领域的位置检测。它具有以下特点: 高精度位置测量 多种通信接口(如SSI、PROFIBUS、PROFINET等) 坚固的工业设计 支持多种安装方式 二、元器件及配件 设备…...
MySQL 学习(十)执行一条查询语句的内部执行过程、MySQL分层
目录 一、MySQL 执行流程图二、MySQL的分层2.1 连接阶段2.2 查询缓存阶段(Query Cache,MySQL 8.0已移除)2.3 解析与预处理阶段(词法分析、语法分析、预处理器)2.4 查询优化阶段2.5 执行引擎阶段 三、常见面试题3.1 MyS…...
C语言中的指定初始化器
什么是指定初始化器? C99标准引入了一种更灵活、直观的初始化语法——指定初始化器(designated initializer), 可以在初始化列表中直接引用结构体或联合体成员名称的语法。通过这种方式,我们可以跳过某些不需要初始化的成员,并且可以以任意顺序对特定成员进行初始化。这…...
什么是 NB-IoT ?窄带IoT 应用
物联网使各种应用能够与大量无线通信设备进行连接和通信。它有望为智能城市、公用事业、制造设施、农业应用、远程工业机械等提供动力。这些应用均可使用窄带物联网(NB-IoT )网络协议。 例如,智能城市可使用 NB-IoT 监控整个城市的街道照明、…...
CSRF 和 XSS 攻击分析与防范
CSRF 和 XSS 攻击分析与防范 CSRF (跨站请求伪造) 什么是 CSRF? CSRF (Cross-Site Request Forgery) 是一种攻击方式,攻击者诱使用户在已登录目标网站的情况下,执行非预期的操作。 攻击流程: 用户登录可信网站 A在不登出 A 的…...
Window下Jmeter多机压测方法
1.概述 Jmeter多机压测的原理,是通过单个jmeter客户端,控制多个远程的jmeter服务器,使他们同步的对服务器进行压力测试。 以此方式收集测试数据的好处在于: 保存测试采样数据到本地机器通过单台机器管理多个jmeter执行引擎测试…...
Apache RocketMQ ACL 2.0 全新升级
📖知识延伸:本文相关知识库已收录至「RocketMQ 中文社区」,同步更新更多进阶内容 引言 RocketMQ 作为一款流行的分布式消息中间件,被广泛应用于各种大型分布式系统和微服务中,承担着异步通信、系统解耦、削峰填谷和消…...
第九讲 | 模板进阶
模板进阶 一、非类型模板参数1、模板参数的分类2、应用场景3、array4、注意 二、模板的特化1、概念2、函数模板特化3、类模板特化(1)、全特化:全部模板参数都特化成具体的类型(2)、偏/半特化:部分模板参数特…...
联合建模组织学和分子标记用于癌症分类|文献速递-深度学习医疗AI最新文献
Title 题目 Joint modeling histology and molecular markers for cancer classification 联合建模组织学和分子标记用于癌症分类 01 文献速递介绍 癌症是对人类致命的恶性肿瘤,早期准确诊断对癌症治疗至关重要。目前,病理诊断仍是癌症诊断的金标准…...
会计要素+借贷分录+会计科目+账户,几个银行会计的重要概念
1.借贷分录还是借贷分路 正确表述是“借贷分录”。 “分录”即会计分录,它是指预先确定每笔经济业务所涉及的账户名称,以及计入账户的方向和金额的一种记录,简称分录。 在借贷记账法下,会计分录通过“借”和“贷”来表示记账方向…...
【C++】set和multiset的常用接口详解
前⾯我们已经接触过STL中的部分容器如:string、vector、list、deque、array、forward_list等,本篇文章将介绍一下map和multiset的使用。 1. 序列式容器和关联式容器 在介绍set之前我们先简单介绍一下什么是序列式容器和关联式容器。 前⾯我们已经接触过S…...
PostgreSQL 联合索引生效条件
最近面试的时候,总会遇到一个问题 在 PostgreSQL 中,联合索引在什么条件下会生效? 特此记录~ 前置信息 数据库版本 PostgreSQL 14.13, compiled by Visual C build 1941, 64-bit 建表语句 CREATE TABLE people (id SERIAL PRIMARY KEY,c…...
聊聊redisson的lockWatchdogTimeout
序 本文主要研究一下redisson的lockWatchdogTimeout lockWatchdogTimeout redisson/src/main/java/org/redisson/config/Config.java private long lockWatchdogTimeout 30 * 1000;/*** This parameter is only used if lock has been acquired without leaseTimeout param…...
数据结构第七章(三)-树形查找:红黑树
树形查找(二) 红黑树一、红黑树1.定义2.黑高3.性质 二、插入1.插入步骤2.举例 总结 红黑树 红黑树来喽~ 我们在上一篇说了二叉排序树(BST)和平衡二叉树(AVL),那么既然都有这两个了,…...
C++篇——多态
目录 引言 1,什么是多态 2. 多态的定义及实现 2_1,多态的构成条件 2_2,虚函数 2_3,虚函数的重写 2_4,虚函数重写的两个例外 2_4_1,协变(基类与派生类虚函数返回值类型不同) 2_4_2. 析构函数的重写(基类…...
AI实时对话的通信基础,WebRTC技术综合指南
在通过您的网络浏览器进行音频和视频通话、屏幕共享或实时数据传输时,您可能并不常思考其背后的技术。推动这些功能的核心力量之一就是WebRTC。2011年由谷歌发布的这个开源项目,如今已发展成为一个高度全面且不断扩展的生态系统。尤其是在AI技术大幅突破…...
【寻找Linux的奥秘】第五章:认识进程
请君浏览 前言1. 冯诺依曼体系结构数据流动 2. 操作系统(Operating System)2.1 概念2.2 设计OS的目的2.3 如何理解“管理”2.4 系统调用和库函数概念 3. 进程3.1 基本概念3.1.1 查看进程3.1.2 创建进程 3.2 进程状态3.2.1 简单介绍3.2.2 运行&&阻…...
uniapp微信小程序-长按按钮百度语音识别回显文字
流程图: 话不多说,上代码: <template><view class"content"><view class"speech-chat" longpress"startSpeech" touchend"endSpeech"><view class"animate-block" …...
支付宝创建商家订单收款码(统一收单线下交易预创建).net开发的软件附带大型XML文件可以删除吗?AlipaySDKNet.OpenAPI.xml
支付宝创建商家订单收款码(统一收单线下交易预创建)一个程序55MB,XML就带了35MB AlipaySDKNet.OpenAPI.xml,BouncyCastle.Crypto.xml 支付宝店铺收款码创建的程序,这些文件可以不用吗 在支付宝店铺收款码创建的程序中…...
Profinet转Ethernet/IP网关模块通信协议适配配置
案例背景 在某自动化生产车间中,现有控制系统采用了西门子 S7 - 1500 PLC 作为主要控制器,负责生产流程的核心控制。同时,由于部分设备的历史原因,存在使用 AB 的 PLC 进行特定环节控制的情况。为了实现整个生产系统的信息交互与…...
4.6/Q1,GBD数据库最新文章解读
文章题目:Global burden, subtype, risk factors and etiological analysis of enteric infections from 1990-2021: population based study DOI:10.3389/fcimb.2025.1527765 中文标题:1990-2021 年肠道感染的全球负担、亚型、危险因素和病因…...
数字孪生技术:开启未来的“镜像”技术
想象一下,你拥有一个与现实世界一模一样的 “数字分身”,它不仅长得像你,行为举止、思维方式也和你毫无二致,甚至能提前预知你的下一步行动。这听起来像是科幻电影里的情节,但数字孪生技术却让它在现实中成为了可能。数…...
Java 序列化(Serialization)
一、理论说明 1. 序列化的定义 Java 序列化是指将对象转换为字节流的过程,以便将其存储到文件、数据库或通过网络传输。反序列化则是将字节流重新转换为对象的过程。通过实现java.io.Serializable接口,类可以被标记为可序列化的,该接口是一…...
Python解析Excel入库如何做到行的拆分
我们读取解析Excel入库经常会遇到这种场景,那就是行的拆分,如图: 比如我们入库,要以name为主键,可是表格name的值全是以逗号分割的多个,这怎么办呢?这就必须拆成多行了啊。 代码如下ÿ…...
信创国产化监控 | 达梦数据库监控全解析
达梦数据库(DM Database)是国产数据库的代表产品之一,在政府、金融、电信、能源等多个关键行业应用广泛,它具有高兼容性、高安全性、高可用性、高性能、自主可控等特点。随着国产化替代进程加速,达梦数据库在关键信息基…...
Parsec解决PnP连接失败的问题
提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 前言一、准备环境二、DMZ三、端口映射1.Parsec设置固定端口2.路由器设置端口转发3.重启被控端Parsec四、多少一句1.有光猫管理员账号2.没有光猫管理员账号总结 前言…...
LLM笔记(二)LLM数据基础
核心目标: 构建 LLM 的数据基础,将原始文本转化为模型可处理的、包含丰富语义和结构信息的数值形式。 一、 环境与库准备 (Environment & Libraries): 必要库确认: 在开始之前,确保 torch (PyTorch深度学习框架) 和 tiktoken (OpenAI的高效BPE分词…...
让三个线程(t1、t2、t3)按顺序依次打印 A、B、C
public class ThreadWait {private static final Object lock = new Object();private static boolean t1Output=true;private static boolean t2Output=false;private static boolean t3Output=false;public static void main(String[] args) {//线程1new Thread(new Runnable…...
2、ubantu系统配置OpenSSH | 使用vscode或pycharm远程连接
1、OpenSSH介绍 OpenSSH(Open Secure Shell)是一套基于SSH协议的开源工具,用于在计算机网络中提供安全的加密通信。它被广泛用于远程系统管理、文件传输和网络服务的安全隧道搭建,是保护网络通信免受窃听和攻击的重要工具。 1.1…...
idea启动报错:java: 警告: 源发行版 11 需要目标发行版 11(亲测解决)
引起原因 idea的jdk没有替换干净 1.配置project file–Project Structrue–Project 2.配置Modules-Sources file–Project Structrue–Modules-Sources 改为jdk11 3.配置Modules-Dependencies file–Project Structrue–Modules-Dependencies...
Pycharm IDEA加载大文件时报错:The file size exceeds configured limit
解决方案:配置一下idea.properties文件 文件里面写入代码: idea.max.intellisense.filesize50000重启IDEA即可;...