【经典算法】LeetCode 136:只出现一次的数字(Java/C/Python3实现含注释说明,Easy)

个人主页: 进朱者赤

阿里非典型程序员一枚 ,记录平平无奇程序员在大厂的打怪升级之路。 一起学习Java、大数据、数据结构算法(公众号同名

目录

  • 题目描述
  • 思路及实现
    • 方式一:使用异或运算(推荐)
      • 思路
      • 代码实现
        • Java版本
        • C语言版本
        • Python3版本
      • 复杂度分析
    • 方式二:哈希表
      • 思路
      • 代码实现
        • Java版本
        • C语言版本
        • Python3版本
      • 复杂度分析
  • 总结
  • 相似题目
  • 其他小知识
    • 几个有趣的位操作
      • 1. 利用或操作 | 和空格将英文字符转换为小写
      • 2. 利用与操作 & 和下划线将英文字符转换为大写
      • 3. 利用异或操作 ^ 和空格进行英文字符大小写互换
      • 4. 加一
      • 5. 减一
      • 其他

  • 标签(题目类型):位运算

题目描述

136. 只出现一次的数字 
给你一个 非空 整数数组 nums ,除了某个元素只出现一次以外,其余每个元素均出现两次。找出那个只出现了一次的元素。你必须设计并实现线性时间复杂度的算法来解决此问题,且该算法只使用常量额外空间。示例 1 :输入:nums = [2,2,1]
输出:1
示例 2 :输入:nums = [4,1,2,1,2]
输出:4
示例 3 :输入:nums = [1]
输出:1
提示:1 <= nums.length <= 3 * 104
-3 * 104 <= nums[i] <= 3 * 104
除了某个元素只出现一次以外,其余每个元素均出现两次。

原题:
LeetCode: LeetCode 136
力扣 : 力扣 136

思路及实现

方式一:使用异或运算(推荐)

思路

利用异或运算的性质:任何数和0异或等于它本身,任何数和其自身异或等于0,异或运算满足交换律和结合律。因此,我们可以将数组中的所有数字进行异或运算,出现两次的数字会相互抵消,最终剩下的就是只出现一次的数字。

  • 以[4,1,3,4,3]以例

面试官最期待的解法思路,考察计算机基础知识

代码实现

Java版本
class Solution {public int singleNumber(int[] nums) {int x = 0;for (int num : nums)  // 1. 遍历 nums 执行异或运算x ^= num;return x;            // 2. 返回出现一次的数字 x}
}

说明: 通过遍历数组中的每个数字,并使用异或运算将结果保存在result变量中,最终返回result即可。

C语言版本
#include <stdio.h>int singleNumber(int* nums, int numsSize) {int x = 0;for (int i = 0; i < numsSize; i++) {  // 1. 遍历 nums 执行异或运算x ^= nums[i];}return x;                            // 2. 返回出现一次的数字 x
}

说明: 使用for循环遍历数组,将每个元素与result进行异或运算,并更新result的值。

Python3版本
class Solution:def singleNumber(self, nums: List[int]) -> List[int]:x = 0for num in nums:  # 1. 遍历 nums 执行异或运算x ^= num      return x;         # 2. 返回出现一次的数字 x

说明: Python中的实现与Java和C类似,使用for循环遍历数组,并通过异或运算找出只出现一次的数字。

复杂度分析

  • 时间复杂度:O(n),其中 n 是数组 nums 的长度。这是因为我们只需要遍历一次数组,对每个元素进行一次异或操作。
  • 空间复杂度:O(1),因为我们只使用了常数个额外的变量来存储中间结果,与输入数组的大小无关。

方式二:哈希表

思路

使用哈希表记录每个数字出现的次数,最后找到出现次数为 1 的数字即为答案。

下面以HashMap为例,HashSet也是可以的

代码实现

Java版本
class Solution {public int singleNumber(int[] nums) {Map<Integer, Integer> counts = new HashMap<>();for (int num : nums) {counts.put(num, counts.getOrDefault(num, 0) + 1); // 统计每个数字出现的次数}for (int num : counts.keySet()) {if (counts.get(num) == 1) {return num;  // 返回出现次数为 1 的数字}}return -1; //理论上不会到达这里}
}

说明:
创建一个哈希表 counts,用于存储每个数字出现的次数。
遍历数组 nums,统计每个数字出现的次数并更新到 counts 中。
遍历 counts,找到出现次数为 1 的数字并返回。

C语言版本
#include <stdio.h>
#include <stdlib.h>// 假设数字范围在 0 到 10000 之间,可以使用数组模拟哈希表
int singleNumber(int* nums, int numsSize) {int counts[10001] = {0};for (int i = 0; i < numsSize; i++) {counts[nums[i]]++;  // 统计每个数字出现的次数}for (int i = 0; i < 10001; i++) {if (counts[i] == 1) {return i;  // 返回出现次数为 1 的数字}}return -1; //理论上不会到达这里
}

说明
使用数组模拟哈希表,假设数字范围在 0 到 10000 之间。逻辑与 Java 版本类似。

Python3版本

class Solution:def singleNumber(self, nums: List[int]) -> int:counts = {}for num in nums:counts[num] = counts.get(num, 0) + 1  # 统计每个数字出现的次数for num, count in counts.items():if count == 1:return num  # 返回出现次数为 1 的数字return -1 #理论上不会到达这里

说明:

复杂度分析

  • 时间复杂度:O(n),其中 n 是数组 nums 的长度。需要遍历一次数组进行计数,以及遍历哈希表查找出现次数为 1 的数字。
  • 空间复杂度:O(n),最坏情况下哈希表需要存储所有不同的数字,空间复杂度为 O(n)。

总结

方式优点缺点时间复杂度空间复杂度其他
异或运算代码简洁,效率高不直观,需要理解异或运算的特性O(n)O(1)适用于数字类型的题目
哈希表思路直观,易于理解空间复杂度较高,需要额外的存储空间O(n)O(n)适用于各种数据类型的题目

相似题目

相似题目难度链接
两个数组的交集简单leetcode-349
数组中数字出现的次数简单leetcode-136
只出现一次的数字 II中等leetcode-137
找出数组中消失的数字简单leetcode-448
数组中重复的数据简单leetcode-287

其他小知识

几个有趣的位操作

1. 利用或操作 | 和空格将英文字符转换为小写

('a' | ' ') = 'a'
('A' | ' ') = 'a'

2. 利用与操作 & 和下划线将英文字符转换为大写

('b' & '_') = 'B'
('B' & '_') = 'B'

3. 利用异或操作 ^ 和空格进行英文字符大小写互换

('d' ^ ' ') = 'D'
('D' ^ ' ') = 'd'

说明:
以上操作能够产生奇特效果的原因在于 ASCII 编码。ASCII 字符其实就是数字,恰巧空格和下划线对应的数字通过位运算就能改变大小写。

eg: 以1. 利用或操作 | 和空格将英文字符转换为小写为例

字符 ‘A’ 的 ASCII 值是 65。
字符 ’ '(空格)的 ASCII 值是 32。

按位或操作是这样的:

65 (二进制: 01000001)
|32 (二进制: 00100000)
97 (二进制: 01100001)
得到的结果 97 是字符 ‘a’ 的 ASCII 值。

因此,‘A’ | ’ ’ 的结果是字符 ‘a’。

注意:
这种操作通常不用于处理文本或字符串,因为它依赖于字符的特定 ASCII 编码,并且可能会导致非预期的结果或难以理解的代码。在大多数情况下,处理字符和字符串时,应使用语言提供的字符串处理函数和操作符,而不是按位操作符。

4. 加一

int n = 1;
n = -~n;
// 现在 n = 2

5. 减一

int n = 2;
n = ~-n;
// 现在 n = 1

其他

技巧技巧描述示例
技巧1判断奇偶性
使用与运算符(&)和1进行位与运算,结果为0则为偶数,结果为1则为奇数
int num = 5;
boolean isEven = (num & 1) == 0;
技巧2交换两个数
使用异或运算符(^)进行交换两个数,不需要额外的临时变量
int a = 3, b = 5;
a = a ^ b;
b = a ^ b;
a = a ^ b;
技巧3取反操作
使用取反运算符(~)进行取反操作
int num = 7;
int result = ~num;
技巧4清除最低位的1
使用减一后进行与运算操作
int num = 12;
int result = num & (num - 1);
技巧5判断是否为2的幂
通过n与n-1进行与运算,结果为0则是2的幂
int num = 16;
boolean isPowerOfTwo = (num & (num - 1)) == 0;
技巧6位移操作
左移(<<)和右移(>>)操作进行位移运算
int num = 8;
int leftShifted = num << 1;
int rightShifted = num >> 1;
技巧7判断两个数是否异号
通过异或运算符(^)进行判断两数的符号位是否相同
int x = -1, y = 2;
boolean isOpposite = ((x ^ y) < 0);

说明:技巧7 判断两个数是否异号
利用的是补码编码的符号位。整数编码最高位是符号位,负数的符号位是 1,非负数的符号位是 0,再借助异或的特性,可以判断出两个数字是否异号。

当然,如果不用位运算来判断是否异号,需要使用 if else 分支,还挺麻烦的。你可能想利用乘积来判断两个数是否异号,但是这种处理方式容易造成整型溢出,从而出现错误。

欢迎一键三连(关注+点赞+收藏),技术的路上一起加油!!!代码改变世界

  • 关于我:阿里非典型程序员一枚 ,记录平平无奇程序员在大厂的打怪升级之路。 一起学习Java、大数据、数据结构算法(公众号同名

⬇️⬇️欢迎关注下面的公众号:进朱者赤,认识不一样的技术人。⬇️

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/623325.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Day01-环境准备与镜像案例

Day01-环境准备与镜像案例 1. 容器架构1.1 Iaas Paas Saas (了解)1.2 什么是容器1.3 容器vs虚拟机1.4 Docker极速上手指南1&#xff09;配置docker源(用于安装docker)2&#xff09;docker下载镜像加速的配置3&#xff09;自动补全 1.5 Docker C/S架构1.6 Docker的镜像管理1&…

MS1002激光测距用高精度时间测量(TDC)电路

产品简述 MS1002 是一款高精度时间测量 (TDC) 电路&#xff0c;具有更高的精度 和更小的封装&#xff0c;适合于低成本时间测量的工业应用领域。 MS1002 具有双通道、多脉冲的采样能力、高速 SPI 通讯、 多种测量模式&#xff0c;适合于激光雷达和激光测距。 主要特点…

百度沈抖:传统云计算不再是主角,智能计算呼唤新一代“操作系统”

Create 2024 百度AI开发者大会 4月16日&#xff0c;Create 2024 百度AI开发者大会在深圳召开。期间&#xff0c;百度集团执行副总裁、百度智能云事业群总裁沈抖正式发布新一代智能计算操作系统——万源&#xff0c;通过对AI原生时代的智能计算平台进行抽象与封装设计&#xff…

Python学习笔记(37)——用xlwings库生成excel

老规矩先pip入xlwings库 STEP1:下载xlwings库 windowsr>>cmd>>pip install xlwings (如果需要不同版本可以到pypi上搜&#xff09; STEP2:完成EXCEL初级创建 请打开您的编写软件~~~~~&#xff08;小编的显示结果为PYCHARM编写的&#xff0c;因为颜色标注好看(…

普通人做抖音小店真的能赚钱吗?可以,但更取决于个人

大家好&#xff0c;我是电商花花。 现在做抖音小店的基本上都是一些新商家&#xff0c;对于我们众多零基础的朋友来说&#xff0c;是期待也是一份挑战。 抖音小店作为一个充满机会的新兴平台&#xff0c;许多人都欣喜的投入其中&#xff0c;期望能够借此来改变自己的命运&…

Python 物联网入门指南(四)

原文&#xff1a;zh.annas-archive.org/md5/4fe4273add75ed738e70f3d05e428b06 译者&#xff1a;飞龙 协议&#xff1a;CC BY-NC-SA 4.0 第九章&#xff1a;构建光学字符识别的神经网络模块 本章介绍以下主题&#xff1a; 使用光学字符识别&#xff08;OCR&#xff09;系统 使…

springboot+vue全栈开发【3.前端篇之Vue基础语法1】

目录 前言Vue基础语法1.内容渲染指令2.属性绑定指令3.使用JavaScript表达式 前言 hi&#xff0c;这个系列是我自学开发的笔记&#xff0c;适合具有一定编程基础&#xff08;html、css那些基础知识要会&#xff01;&#xff09;的同学&#xff0c;有问题及时指正&#xff01; …

揭示边缘计算网关的市场价格趋势(购买指南)-天拓四方

在数字化转型的大潮中&#xff0c;边缘计算网关作为连接云端与终端设备的核心节点&#xff0c;其重要性日益凸显。然而&#xff0c;面对市场上琳琅满目的边缘计算网关产品&#xff0c;对于许多企业和个人用户来说&#xff0c;边缘计算网关的价格成为选择产品时的重要考量因素。…

最新AI创作系统ChatGPT网站源码AI绘画,GPTs,AI换脸支持,GPT联网提问、DALL-E3文生图

一、前言 SparkAi创作系统是基于ChatGPT进行开发的Ai智能问答系统和Midjourney绘画系统&#xff0c;支持OpenAI-GPT全模型国内AI全模型。本期针对源码系统整体测试下来非常完美&#xff0c;那么如何搭建部署AI创作ChatGPT&#xff1f;小编这里写一个详细图文教程吧。已支持GPT…

JS-35-jQuery04-操作DOM

jQuery的选择器很强大&#xff0c;用起来又简单又灵活&#xff0c;但是搞了这么久&#xff0c;我拿到了jQuery对象&#xff0c;到底要干什么&#xff1f; 答案当然是操作对应的DOM节点啦&#xff01; 回顾一下修改DOM的CSS、文本、设置HTML有多么麻烦&#xff0c;而且有的浏览…

SpringBoot学习(三)数据访问、基础特性、核心原理

文章目录 数据访问示例自动配置原理jdbc场景自动配置数据源等基本信息MyBatisAutoConfiguration配置MyBatis整合流程 基础特性SpringApplication自定义banner自定义SpringApplicationFluentBuilder API Profiles使用指定环境环境激活环境包含 Profiles配置文件 外部化配置配置优…

选择最佳 PoE 布线系统的 3 个步骤

选择合适的 POE 布线系统的重要性 在不断发展的信息和通信技术 &#xff08;ICT&#xff09; 领域&#xff0c;以太网供电 &#xff08;PoE&#xff09; 布线系统已成为一种革命性的解决方案&#xff0c;它简化了网络设备的部署和管理&#xff0c;同时优化了电力传输。从智能建…