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

剑指offer-50、数组中重复的数字

题目描述

在⼀个⻓度为 n 的数组⾥的所有数字都在 0 到n-1 的范围内。 数组中某些数字是重复的,但不知
道有⼏个数字是重复的。也不知道每个数字重复⼏次。请找出数组中第⼀个重复的数字。 例如,如果输⼊⻓度为 7 的数组 [2,3,1,0,2,5,3] ,那么对应的输出是第⼀个重复的数字 2 。没有重复的数字
返回 -1 。

示例1

输⼊
[ 2, 3, 1, 0, 2, 5, 3 ]

返回值
2

思路及解答

借用Set

⾸先可能想到的做法,就是借助 set ,如果元素不存在 set 中,就将元素添加进去,如果 set
中包含该元素,就返回该元素即可。如果⼀直都没有重复的,那么最后返回 -1 。

public class Solution {public int duplicate(int[] numbers) {if (numbers != null) {Set<Integer> set = new HashSet<>();for (int i = 0; i < numbers.length; i++) {if (set.contains(numbers[i])) {return numbers[i];} else {set.add(numbers[i]);}}}return -1;}
}
  • 时间复杂度:O(n) ,最差的情况可能遍历完所有的元素
  • 空间复杂度: O(n) ,最⼤需 要set ⼤⼩为 n

借助数组

可以直接借助数组,因为所有数字都在 0 到 n-1 的范围内,⽤⼀个⼤⼩为 n 的数组,就可以对所有的数字进⾏统计个数,如果个数超过 1 ,那么肯定是重复的数字,如果没有重复的数字,则返回 -1 ;

public class Solution {public int duplicate(int[] numbers) {if (numbers != null) {int[] nums = new int[numbers.length];for (int i = 0; i < numbers.length; i++) {if (nums[numbers[i]] == 1) {return numbers[i];} else {nums[numbers[i]] = 1;}}}return -1;}
}

同样这种做法的时间复杂度和空间复杂度都是 O(n) ,并没有优化太多。

那么有没有空间复杂度为O(1) 的做法呢?

操作原数组(最优)

不借助额外的空间,那么就只能操作原数组了。如果没有重复的情况,那么这些数字排序后,数字i 和数组下标i 应该是⼀⼀对应的。不会出现多个数字i 的情况。

基于这个原则,在遍历数组的时候,将元素 i 调整到下标 i 的位置,如果下标i的位置已经有元
素,那么说明冲突了,这个元素肯定是重复的,否则继续调整后⾯的。如果没有发现重复的数字,就返回 -1 。

public class Solution {public int duplicate(int[] numbers) {int i = 0;while(i < numbers.length) {if(numbers[i] == i) {i++;continue;}if(numbers[numbers[i]] == numbers[i]) return numbers[i];int tmp = numbers[i];numbers[i] = numbers[tmp];numbers[tmp] = tmp;}return -1;}
}

但是上⾯的做法,不适合求解多个重复数字的例⼦,因为调换的时候,很容易将后⾯的数字换到前⾯去,就会导致求解出来不是第⼀个重复的数字(可以⽤来求解任意的重复数字),可能是第2,3... 或者其他的重复数字。譬如: [6,3,2,0,2,5,0] 正确的解应该是 2 ,但是由于第⼀次把 6 和最后
的0 调换了位置,就会导致求解出来的值为 0 。

http://www.jsqmd.com/news/97696/

相关文章:

  • CosyVoice语音模型微调实战:从零到一打造专属语音助手
  • 2025年年终智能学习机品牌推荐:基于千名用户真实反馈与多维度评测的10款高口碑型号深度解析 - 十大品牌推荐
  • DeepSeek-V3模型转换终极指南:从避坑到性能飞跃的完整实战手册
  • 【赵渝强老师】Kafka消息的消费模式
  • 云端AI集成革命:MemGPT企业级长上下文记忆管理技术深度解析
  • LangFlow与CI/CD流水线集成实现AI自动化测试
  • 电流探头能否测量交流冲击电流及相关测试要点
  • Shell脚本安全终极指南:5步构建坚不可摧的防护体系
  • 在 SAP 里,“平行分类账(Parallel Ledger)” 并不是让同一笔业务在 BKPF 里生成多套凭证号,而是“一行 BKPF 记录 + 多行 ACDOCA/FAGLFLEXA 记录” 的模
  • IsaacLab终极版本兼容性指南:快速解决Isaac Sim升级难题
  • 在 SAP 里,想让“同一笔业务”在多个账套(平行分类账)中生成不同编号的会计凭证,标准做法就是
  • 终极指南:3种强制开启USB调试模式的实用方案
  • 如何快速掌握OpenCLIP:多模态AI的完整实践指南
  • FileBrowser API扩展功能:一键配置效率提升的完整指南
  • 终极窗口切换神器:AltTab让你的macOS效率翻倍
  • 5分钟学会Pts物理引擎:从零构建粒子碰撞系统
  • gumbo-parser完整教程:C语言HTML5解析终极指南
  • manga-image-translator终极交互设计:如何用智能界面简化复杂翻译流程
  • 11、Unix 实用工具创建与系统调整
  • 第七十五篇:Kubernetes入门:Pod, Deployment, Service核心概念深度解析
  • 多智能体协同决策:应对复杂业务场景的技术突围之路
  • 12、Unix系统优化与管理脚本实用指南
  • AI绘图革命:用自然语言创建专业图表的新时代
  • 精通FreeRTOS与WolfSSL v5.6.4集成:嵌入式安全通信深度实战
  • Qwen-Image-Lightning:8步极速文生图技术重塑AI创作效率边界
  • Keyboard-Layout-Editor:重新定义键盘设计的在线创作平台
  • 13、Unix 系统管理脚本实用指南
  • LSUnusedResources:让你的iOS项目轻装上阵的专业清理工具
  • 14、系统管理:用户管理脚本详解
  • 突破性能瓶颈:CanvasKit渲染引擎的5大核心技术揭秘