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

leetcode136,137,260:只出现一次的数字 | || |||

文章目录

      • leetcode 136.只出现一次的数字 |
      • leetcode 137. 只出现一次的数字 ||
      • leetcode 260.只出现一次的数字 |||

leetcode 136.只出现一次的数字 |

在这里插入图片描述
用位运算就可以,相当的简单。
异或:不相同异或为1,相同异或为0。
结论:相同的数字异或为0。任何数和0异或都是自己本身。
验证
在这里插入图片描述
在这里插入图片描述

对于这道题,可以用0和数组里的每个数都异或,最终结果就只出现一次的数字。对吧?因为数组里相同的数字异或结果为0,0和只出现一次的数字异或为其本身。

题解

class Solution {
public:
    int singleNumber(vector<int>& nums) 
    {
        int n=0;
        for(int num:nums)
        {
            n=n^num;
        }
        return n;
    }
};

不会用范围for的,可以走循环,自己实现,都可以。


leetcode 137. 只出现一次的数字 ||

在这里插入图片描述
审题:只用一个数字出现一次,其余的数字都出现三次。
举例:[2,2,2,3]
在这里插入图片描述
如果只有[2,2,2],那么统计结果就是0 3 0,这三个数都是可以整除3的。
因为有了[3]的加入,导致统计结果为0 4 1,可以发现第一位是可以整除3,后两位都不能整除3。那么可以得出结论:
出现一次的数 它的某个 位上是0不会影响统计结果,某个 位上是1,会影响结果(和无法整除3)。

推广:如果是:每个数都出现4次,只有一个数出现1次,和上面一样统计每个位1的总和,如果那个位上总和不能整除4,说明只出现一次的数在那个位上为1;如果依旧可以整除4,那么只出现一次的数在那个位上为0。

题解

class Solution {
public:
    int singleNumber(vector<int>& nums) 
    {
        int n=0;
        for(size_t i=0;i<32;i++)
        {
            int total=0;
            //遍历所有nums的第i位
            for(int num:nums)
            {
                total+=(num>>i&1);
            }
            //无法整除3的位,是1。
            if(total%3)
            {
                n=n|1<<i;
            }
        }
        return n;
    }
};

代码解释

total+=(num>>i&1);

num取得是nums其中一个数,num向右移动i位,如果和1按位与结果为1,那么说明num的第i位为1。
在这里插入图片描述

           if(total%3)
            {
                n=n|1<<i;
            }

如果某个位1的和,无法整除3,那么那个位上是1对于只出现一次的数。
使1向左移动i位,再和n按位或,那么肯定会使n的第i位变成1。
在这里插入图片描述
就是这样,没毛病。


leetcode 260.只出现一次的数字 |||

在这里插入图片描述
审题:只有两个元素只出现一次,其余的都出现两次,要求返回一个数组,数组里有这两个元素,不在意两个元素的次序。
举例:[1,1,2,2,3,5]
用位运算可以解,这道题最简单的是用map来做,统计一下每个元素出现的次数,为1的就是我们要找到。这样做比较简单不讲,我们主要来看看用位运算如何做,也就是进阶版。
在这里插入图片描述
思路
依旧先用0和数组的每个数进行异或,那么得到结果就是只出现一次的两个数的异或,这就需要我们想办法将这两个数分开。
用0去异或[1,1,2,2,3,5],得到[3,5]两数的异或结果n。
在这里插入图片描述
n=110,n这个数,二进制位为1的时候说明一个问题:3和5在这个位上是不同的。也就是说:我们可不可以根据这个位·为0/为1,来将数组分成两派。那么3和5必然会被分成两派。对于上面的数组可以分成[115] [223]。[115]的第一位是0,[223]的第一位是1。分成两派就好搞了。这不就是分开的第一题嘛?直接对每一派都用0去异或就可以分别得到只出现一次的数字。
我们要去找n这个数的二进制位第一次为1的位,依此来分派。
题解

class Solution {
public:
    vector<int> singleNumber(vector<int>& nums) {
        //得到只出现一次得两数的异或结果
        int n = 0;
        for (int num: nums) {
            n ^= num;
        }
        // 防止溢出
        //lsb就是n第一次位为1的位置
        int lsb = (n == INT_MIN ? n : n & (-n));
        int type1 = 0, type2 = 0;
        for (int num: nums) {
            if (num & lsb) {
                type1 ^= num;
            }
            else {
                type2 ^= num;
            }
        }
        vector<int> arr;
        arr.push_back(type1);
        arr.push_back(type2);
        return arr;
    }
};

代码解释

int lsb = (n == INT_MIN ? n : n & (-n));

可以用n=110来举例,n=6,-n=-6,两者按位与
在这里插入图片描述
可以看到n第一次位为1的位完美,展现出来了。然后可以据此分派。

            if (num & lsb) {
                type1 ^= num;
            }
            else {
                type2 ^= num;

如果num&lsb==1,说明在那位上为1,如果按位与结果是0,说明那位上为0。
两派分别异或,最终得到结果。


相关文章:

  • mysql安装8.0详细操作
  • 《算法竞赛进阶指南》,USACO2007 牛站
  • (附源码)ssm旅游企业财务管理系统 毕业设计 102100
  • 【MindSpore易点通】如何将PyTorch源码转成MindSpore低阶APIP,并在Ascend芯片上实现单机单卡训练
  • vue前端 页面样式强制覆盖
  • WPF 控件专题 ScrollBar控件详解
  • DocuWare 庆祝文档管理云解决方案推出10 周年
  • Busybox实践2:分析busybox文件链接原理并编程模拟实现自己的busybox文件
  • 12030.LMK03000时钟合成器
  • el-table表格进行排序 清除排序和清除排序箭头的高亮图标
  • 5G网络用户面时延测量
  • StreamSets解析MySQL Binlog写入Kafka
  • android开发获取View坐标位置的几种方式
  • antv x6连线与取消连线的操作+自定义连接桩+节点选择/框选
  • TIA博途V17中ProDiag功能的使用方法示例(一)PLC数据类型的监控
  • 《微软的软件测试之道》成书始末、出版宣告、补充致谢名单及相关信息
  • 0基础学习移动端适配
  • 10个最佳ES6特性 ES7与ES8的特性
  • 2017-08-04 前端日报
  • 30秒的PHP代码片段(1)数组 - Array
  • Android组件 - 收藏集 - 掘金
  • AngularJS指令开发(1)——参数详解
  • CentOS7 安装JDK
  • in typeof instanceof ===这些运算符有什么作用
  • JavaScript创建对象的四种方式
  • JavaScript设计模式与开发实践系列之策略模式
  • node入门
  • redis学习笔记(三):列表、集合、有序集合
  • SpiderData 2019年2月16日 DApp数据排行榜
  • 从零开始在ubuntu上搭建node开发环境
  • 多线程 start 和 run 方法到底有什么区别?
  • 回顾2016
  • 类orAPI - 收藏集 - 掘金
  • 如何使用 JavaScript 解析 URL
  • elasticsearch-head插件安装
  • ​如何使用ArcGIS Pro制作渐变河流效果
  • #我与Java虚拟机的故事#连载05:Java虚拟机的修炼之道
  • (10)ATF MMU转换表
  • (C语言)球球大作战
  • (Python) SOAP Web Service (HTTP POST)
  • (Pytorch框架)神经网络输出维度调试,做出我们自己的网络来!!(详细教程~)
  • (大众金融)SQL server面试题(1)-总销售量最少的3个型号的车及其总销售量
  • (附源码)基于SSM多源异构数据关联技术构建智能校园-计算机毕设 64366
  • (转)jdk与jre的区别
  • (转)用.Net的File控件上传文件的解决方案
  • **Java有哪些悲观锁的实现_乐观锁、悲观锁、Redis分布式锁和Zookeeper分布式锁的实现以及流程原理...
  • .net 发送邮件
  • .NET(C#、VB)APP开发——Smobiler平台控件介绍:Bluetooth组件
  • .NET/C# 的字符串暂存池
  • .Net程序猿乐Android发展---(10)框架布局FrameLayout
  • @transaction 提交事务_【读源码】剖析TCCTransaction事务提交实现细节
  • [20190416]完善shared latch测试脚本2.txt
  • [AIGC codze] Kafka 的 rebalance 机制
  • [ComfyUI进阶教程] animatediff视频提示词书写要点
  • [c语言]小课堂 day2