从零学算法 (LCR 177. 撞色搭配)

整数数组 sockets 记录了一个袜子礼盒的颜色分布情况,其中 sockets[i] 表示该袜子的颜色编号。礼盒中除了一款撞色搭配的袜子,每种颜色的袜子均有两只。请设计一个程序,在时间复杂度 O(n),空间复杂度O(1) 内找到这双撞色搭配袜子的两个颜色编号。

示例 1:

输入:sockets = [4, 5, 2, 4, 6, 6]

输出:[2,5] 或 [5,2]

示例 2:

输入:sockets = [1, 2, 4, 1, 4, 3, 12, 3]

输出:[2,12] 或 [12,2]

  • 如果你对位运算有了一定的了解,你肯定会第一时间想到异或,因为异或运算有个特性就是 a^a = 0,所以就像连连看一样,遍历异或 [1,1,2,2,3] 以后会剩下 3。但是这一题就难在他并非只有一个单色袜,而是有两个,设他们为 x,y ,遍历异或完数组以后会剩下 x^y,记作 z,根据异或的特点可知,如果 z 的某一位为 1,那就表示 x 和 y 在那一位一个为 0 一个为 1。这也就是我们分出 x 和 y 的关键。我们就从右往左不断取 z 的某一位,看哪个是 1。用一个 m 记录,初始化为 0001,然后为 0010, 0100...。那么 x 和 y 与 m 进行 & 运算会得到不同结果,我们再遍历一遍数组就能以此得到 x 和 y 了。
java 复制代码
  public int[] sockCollocation(int[] nums) {
      int x=0,y=0,z=0,m=1;
      // 遍历异或得到 z
      for(int n:nums){
          z ^=n;
      }
      // 得到用来区分 x 和 y 的 m
      while((z&m)==0)m<<=1;
      // 反正最后和 x 异或的同色袜子会被消除,所以一直异或就行了
      // 比如 m 为 0010,就把 xx0x 的同色袜子都交给 x 消除了,剩下为 xx0x 的 x 袜子
      // 把 xx1x 的同色袜子都异或给 y 消除了,剩下为 xx1x 的 y 袜子
      for(int n:nums){
          if((n&m)==0)x^=n;
          else y^=n;
      }
      return new int[]{x,y};
  }
相关推荐
张二娃同学7 分钟前
03_变量常量与输入输出_printf与scanf详解
算法
江南十四行1 小时前
并发编程(一)
java·jvm·算法
z200509301 小时前
今日算法(依旧二叉树)
算法·leetcode·职场和发展
Zxc_1 小时前
《遗传算法:从自然选择到Rastrigin函数优化,手写一个完整的进化求解器》
算法
阿Y加油吧2 小时前
两道经典动态规划题:乘积最大子数组 & 分割等和子集 复盘笔记
笔记·算法·动态规划
三品吉他手会点灯2 小时前
C语言学习笔记 - 33.数据类型 - printf函数的详细用法
c语言·开发语言·笔记·学习·算法
NashSKY2 小时前
PnP 问题:数学描述与 DLT 算法推导
算法·矩阵分解·多视图几何·射影几何
csdn_aspnet3 小时前
C++ Lomuto分区算法(Lomuto Partition Algorithm)
开发语言·c++·算法
ZPC82103 小时前
Open3D 与yolo-3d 那个更适合生成物体3d 包围盒
人工智能·算法·计算机视觉·机器人
行走的陀螺仪3 小时前
JavaScript 算法详解:10大经典算法,通俗易懂,从入门到精通
开发语言·javascript·算法