从零学算法 (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};
  }
相关推荐
C雨后彩虹28 分钟前
任务总执行时长
java·数据结构·算法·华为·面试
风筝在晴天搁浅31 分钟前
代码随想录 463.岛屿的周长
算法
柒.梧.1 小时前
数据结构:二叉排序树构建与遍历的解析与代码实现
java·开发语言·数据结构
一个不知名程序员www1 小时前
算法学习入门---priority_queue(C++)
c++·算法
zhuzewennamoamtf1 小时前
Linux内核platform抽象、数据结构、内核匹配机制
linux·运维·数据结构
TL滕1 小时前
从0开始学算法——第十八天(分治算法)
笔记·学习·算法
LYFlied2 小时前
【每日算法】LeetCode 84. 柱状图中最大的矩形
前端·算法·leetcode·面试·职场和发展
CoderCodingNo2 小时前
【GESP】C++三级真题 luogu-B4414 [GESP202509 三级] 日历制作
开发语言·c++·算法
Liangwei Lin2 小时前
洛谷 P1955 [NOI2015] 程序自动分析
算法