剑指offer第二版-总结:元素出现次数

本系列导航:剑指offer(第二版)java实现导航帖

说明

本篇是在56.数组中只出现一次的两个数字56.2.数组中唯一只出现一次的数字两道题目之上进行的总结与提炼,最好先看懂以上两个题目再来看此总结。

问题描述

                        1.所有元素都出现了2次,除了一个出现1次
                               /                       \
2.所有元素都出现了2次,除了两个数各出现1次       3.所有元素都出现了k次,只有一个出现了1次
                                                       |
                                              4.所有元素都出现了K次,只有一个不是K次

概括分析

对于问题1

可用亦或解决。因为亦或有如下两条关键性质:
(1)a^b=b^a,意味着对一个无序数组进行亦或等价于对排序后的数组进行疑惑
(2)a^b^a=b,意味着出现两次的数字可以抵消掉。
因此,将数组中所有元素进行亦或,即可得到那个只出现一次的一个数字。

对于问题2

它可以通过分组转化为问题1,分析过程见56.数组中只出现一次的两个数字

对于问题3

可以使用一个长度为32的整型数组bitsum记录数据数组中所有元素的每个bit的状态,分析过程见56.2.数组中唯一只出现一次的数字

对于问题4

与问题3的区别在于没有告诉那个特殊的数字出现的次数。发现问题4与问题3的相似性,我们可以尝试借助问题3的解法解决问题4。

问题3的题目要求中的“三次”如果改成其他的数字,该解法依旧是有效的。或者说,该解法可以解决“除了一个数字出现一次外,其他数字都出现k次”这个问题,修改代码中的int k的数值即可。

更近一步,对于“除了一个数字出现M次外,其他数字都出现K次(0<M<K)”这个问题,其实此题的解法也是有效的,只需将

int result = 0;
for(int i=0;i<32;i++){
    result<<=1; 
    result+=bitSum[i]%k;
}

修改为如下代码即可。因为在那个特殊数字出现M次时,bitSum[i]%k不是等于0,就是等于M。

int result = 0;
for(int i=0;i<32;i++){
    result<<=1; 
    result+=bitSum[i]%k>0?1:0;
}

问题4更加凝练的解法:借助电路设计知识解决

为了便于理解,我们先从这个具体题目入手分析面试题56.2:数组中唯一只出现一次的数字

题目要求:

在一个整数数组中除了一个数字只出现一次外,其他数字都出现三次。找出那个出现一次的数字。

思路分析:

之前我们使用长度为32的int数组记录每个bit的状态,但其实是有很多空间浪费的,我们只需要记录出现次数为0次、1次或是2次这三个状态即可,因为3次、4次...等效于0次、1次...

那么int bitsumOld[32]可以替换为boolean bitsum[32][2]。也就是bitsumbitsumOld[i] 被替换成了bitsum[i][],bitsumOld中的每一个整数元素被换成了一个长度为2的boolean型数组,这样之后申请的空间仅为原来的1/16。

更进一步,我们将boolean bitsum[32][0]记为int a,boolean bitsum[32][1]记为int b,待计入的数字c的为1的二进制位会影响bitsum的值,其实也就是在影响a,b的值,只不过a,b将bitsum表示成了两个整数而已。而a,b的数值变化规则(即真值表)是

a的第i个bit  b的第i个bit     |  c的第i个bit   ->     a的第i个bit   b的第i个bit
--------------------------------------------------------------------------
0            0                 0                    0            0
0            1                 0                    0            1
1            0                 0                    1            0
0            0                 1                    0            1
0            1                 1                    1            0
1            0                 1                    0            0

上述表格可用卡诺图表示如下:


卡诺图表示.gif

此图表示的是a,b每个对应bit的变化,由于不同bit之间是互不影响的,因此可以整体表示,即
newa = a&~b&~c+~a&b&c
newb = ~a&b&~c+~a&~b&c

因此,面试题56.2:数组中唯一只出现一次的数字的另一种解法如下:

public class P278_NumberAppearOnce {
    public static int findNumberAppearOnce2(int[] data){
        int a = 0, b = 0,aTemp = 0;
        for(int c:data){
            aTemp = a&~b&~c | ~a&b&c;
            b = ~a&b&~c | ~a&~b&c;
            a = aTemp;
        }
        return b;
    }
}

与之前的解法相比,不仅节省空间,更大大降低了代码量。如果题目是所有数字出现了3次,除了一个数字例外,即不确定是1次还是2次,那么只需要将

return b

修改为

return a|b

如果题目中的3次更改为k次,那么需要⌈log2k⌉个int变量储存状态,然后绘制相应的真值表与卡诺图,写出逻辑表达式即可。

引用与参考

https://discuss.leetcode.com/topic/22821/an-general-way-to-handle-all-this-sort-of-questions/12
http://m.blog.csdn.net/smile_watermelon/article/details/47748227

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

推荐阅读更多精彩内容

  • 背景 一年多以前我在知乎上答了有关LeetCode的问题, 分享了一些自己做题目的经验。 张土汪:刷leetcod...
    土汪阅读 12,663评论 0 33
  • 1.把二元查找树转变成排序的双向链表 题目: 输入一棵二元查找树,将该二元查找树转换成一个排序的双向链表。 要求不...
    曲终人散Li阅读 3,243评论 0 19
  • 【程序1】 题目:古典问题:有一对兔子,从出生后第3个月起每个月都生一对兔子,小兔子长到第三个月后每个月又生一对兔...
    叶总韩阅读 5,068评论 0 41
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,296评论 18 399
  • 5/21 周卫 河南【每日一结构】结构思考力21天思维改善训练营 G:【专注力的修炼】感觉自己一直都在忙着学习,其...
    心理实践者周周阅读 132评论 0 0