算法 | 如何用位运算从一堆数中寻找出现K次的数?

前言
本文主要介绍如何使用位运算从一堆数中寻找出现K次的数,这一堆数中有很多数出现了M次,只有一个数出现了K次,并且K是小于M的,另外要求额外空间复杂度为O(1),时间复杂度为O(n),那么该如何寻找出这个数呢?
例如,给出一个数组arr=[1, 1, 2, 2, 2, 3, 3, 3, 4, 4, 4],K = 2,M = 3。在这个数组中,只有1这个元素出现了2次,其他的元素都出现了3次,现在要求使用位运算把1这个元素找出来。
思路分析
要求额外空间复杂度为O(1),也就是说,我们只能创建有限的、固定的几个变量。
之前的几个寻找元素的题目都用到了异或运算来解决的,这个题目还能用异或运算吗?这个就不可以了,如果K是个偶数的话,用异或就直接给干废了。
我们知道,在Java中,int型的二进制的长度为32位。
如果把这个数组的所有元素都转成二进制,然后把二进制每个位置的数据相加,存放到一个长度为32位的数组中呢?
步骤
假如数组arr=[a,a,b,b,b,c,c,c,d,d,d],K = 2,M = 3,要找出a这个元素。

先声明一个32位长度的数组tmp,里面的元素都默认为0。
循环遍历数组,并把数组中的每一个元素都转为二进制形式。
把每一个元素的二进制的值都累加到第1步声明的32位长度的数组中。
循环遍历这个32位长度的数组,判断数组中每个元素的数值和M的关系。
如果数组中的元素是M的整数倍,那么说明这个位置是没有a这个元素的。

循环arr数组过程如下:
第1个元素a的二进制假设为01111,累加到tmp数组中为,01111。
第2个元素a的二进制假设为01111,累加到tmp数组中为,02222。
第3个元素b的二进制假设为01000,累加到tmp数组中为,03222。
第3个元素b的二进制假设为01000,累加到tmp数组中为,04222。
......
以此类推,直到把所有的元素全部累加。
还原出现K次的数
tmp数组中的元素,假如为T,代表着arr数组中元素二进制位为1数出现的次数,他可能出现这几种组合:

T = 1 * K + 1 * M,所有元素在这个二进制位都为1。
T = 0 * K + 1 * M,只有出现M次的数,二进制位为1。
T = 0 * K + 0 * M,所有元素在这个二进制位都为0。
T = 1 * K + 0 * M,只有出现K次的数,二进制位为1。

从上面4种情况来看,因为 K < M,如果说T对M进行取模运算,取模为0的话,说明这个位置出现K次这个数的二进制位一定为0,否则为1。
所以,让tmp数组中的元素与M进行取模运算,就能识别出来出现K次的数的二进制位是什么情况,然后把二进制转成十进制就可以了,或者直接与0进行或运算,也能得到结果。
代码实现
经过上面的分析,来看下代码是如何实现的吧。
public class Code18_FindKTimes {
public static int findKTimes(int[] arr, int k, int m) {
int[] tmp = new int[32];
for (int i = 0; i < arr.length; i++) {
for (int j = 0; j <= 31; j++) {
tmp[j] += (arr[i] >> j) & 1;
}
}
int res = 0;
for (int i = 0; i < 32; i++) {
if (tmp[i] % m != 0) {
res |= (1 << i);
}
}
return res;
}
public static void main(String[] args) {
int[] arr = {11, 11, 11, 2, 2, 2, 2, 3, 3, 3, 3, 4, 4, 4, 4};
int kNum = findKTimes(arr, 3, 4);
System.out.println(kNum);
}
}
复制代码
运行输出结果为:11。
(arr[i] >> j) & 1代码含义为判断arr[i]元素在第j位置的值是0还是1。
总结
本文主要介绍如何使用位运算从一堆数中寻找出现K次的数,利用了二进制、或运算、以及题目中M次的逻辑关系。
如果你有更好的办法,欢迎在评论区留言交流。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,884评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,755评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,369评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,799评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,910评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,096评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,159评论 3 411
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,917评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,360评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,673评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,814评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,509评论 4 334
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,156评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,882评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,123评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,641评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,728评论 2 351

推荐阅读更多精彩内容