数组中重复的数字

最近在刷题,准备记录一些值得参考的题目,也希望通过这种方式可以让自己印象更加深刻,本篇文章将讨论 数组中重复的数字 相关的一些题目。

题目

在一个长度为 n 的数组里,所有的数字都在 0~n-1 的范围内。数组中某些数字是重复的,但不知道有几个数字重复了,也不知道每个数字重复了多少次。请找出数组中任意一个重复的数字。

例如,如果输入长度为7的数组 {5,0,3,6,5,2,3},那么对应的输出是重复的数字 3 或者 5。

思路一 —— 排序
  1. 将输入的数组排序(升序)
  2. 从头到尾扫描排序后的数组找出重复的数字即可

该方法实现简单,但是对于一个长度为 n 的数组,排序该数组的时间复杂度为 O(nlogn)

思路二 —— 哈希表
  1. 从头到尾扫描数组每个数字
  2. 判断哈希表里面是否已经包含该数字
  3. 如果没有该数字,则把该数字放入哈希表
  4. 如果哈希表中已经存在该数字,则找到一个重复的数字

该算法的时间复杂度为 O(n),但是提高时间效率是以一个大小为 O(n) 的哈希表为代价的。

思路三 —— 推荐

我们注意到数组中的数字都在 0~n-1 的范围内。如果输入的数组中没有重复的数字,那么当数组排序后数字 i 将出现在下标为 i 的位置。

  1. 从头到尾扫描数组中每个数字
  2. 当扫描到下标为 i 的数字 m 时,首先比较这个数字 m 是不是等于 i
    1. 如果是,则接着扫描下一个数字(说明数字本身在应该在的位置上)
    2. 如果不是,则那数字 m 和在第 m 个位置上的数字进行比较
      1. 如果和第 m 个数字相等,则找到了重复的数字(第i个和第m个是重复的数字)
      2. 如果和第 m 个数字不相等,就把第 i 个数字和第 m 个数字交换,把数字 m 放到属于它的位置上,然后继续重复比较,交换的过程,直到发现一个重复的数字,或者扫描结束(没有重复的数字)

我们以题目中的例子来进行说明

{5,0,3,6,5,2,3}

  • 从该数组的第一个数字(位置0)开始扫描
  • 第 0 个数字是 5, 0 和 5 不相等,则比较数字 5 和第 5 个位置上的数(数字2)
  • 数字 5 和第 5 个位置上的数字 2 也不相等,则交换第 0 个位置和第 5 个位置上的数字,得到

{2,0,3,6,5,5,3}

  • 交换后第 0 个位置的数字是 2,仍然不等于 0 ,继续比较、交换的过程
  • 2 和第 2 个位置的数字 3 不相等,继续交换,交换后得到

{3,0,2,6,5,5,3}

  • 交换后的 3 仍然不等于 0,继续比较、交换步骤
  • 3 和第 3 个位置的数字 6 不相等,继续交换,交换后得到

{6,0,2,3,5,5,3}

  • 交换后的 6 仍然不等于 0,继续比较、交换步骤
  • 6 和第 6 个位置的数字 3 不相等,继续交换,交换后得到

{3,0,2,3,5,5,6}

  • 交换后的 3 仍然不等于 0,继续比较、交换步骤
  • 到这里 第 0 个位置的数字 3 和第 3 个位置的数字 3 出现了重复,即找到了重复的数字

该算法中每个数字最多只要交换两次就能找到属于它自己的位置,因此总的时间复杂度是 O(n) ,由于是直接在输入数组上进行的操作,没有分配额外的内存空间,因此空间复杂度为 O(1)

下面是该算法的 JAVA 代码测试用例和实现:

测试用例
public class DuplicateNumberArrayTest {

    @Test
    public void testNormalArray() {
        assertEquals(3, new DuplicateNumberArray().findDuplicationNumber(new int[]{5, 0, 3, 6, 1, 2, 3}));
        assertEquals(2, new DuplicateNumberArray().findDuplicationNumber(new int[]{5, 0, 2, 6, 1, 2, 3}));
        assertEquals(5, new DuplicateNumberArray().findDuplicationNumber(new int[]{5, 0, 2, 6, 5, 1, 3}));
        assertEquals(4, new DuplicateNumberArray().findDuplicationNumber(new int[]{4, 0, 2, 6, 5, 4, 3}));
    }

    @Test
    public void testMultipleDuplicationNumber() {
        assertEquals(5, new DuplicateNumberArray().findDuplicationNumber(new int[]{5, 0, 3, 5, 1, 1, 3}));
        assertEquals(1, new DuplicateNumberArray().findDuplicationNumber(new int[]{1, 0, 3, 5, 1, 1, 3}));
    }

    @Test
    public void testNoDuplicationNumber() {
        assertEquals(-1, new DuplicateNumberArray().findDuplicationNumber(new int[]{5, 0, 3, 2, 1, 6, 4}));
    }

    @Test
    public void testInvalidArray() {
        assertEquals(-1, new DuplicateNumberArray().findDuplicationNumber(new int[]{5, 0, -1, 2, 1, 6, 4}));
    }

    @Test
    public void testNullArray() {
        assertEquals(-1, new DuplicateNumberArray().findDuplicationNumber(null));
    }
}
实现
/**
 * 在一个长度为 n 的数组里,所有的数字都在 0~n-1 的范围内。数组中某些数字是重复的,但不知道有几个数字重复了,也不知道每个数字重复了多少次。请找出数组中任意一个重复的数字。
 *
 * @author Aaric
 */
public class DuplicateNumberArray {

    /**
     * 找出数组中任意一个重复的数字
     *
     * @param array 输入的数组
     * @return 数组中重复的数字或者 -1 (如果输入无效、或者没有重复的数字)
     */
    public int findDuplicationNumber(int[] array) {
        if (array == null) {
            return -1;
        }
        for (int i = 0; i < array.length; i++) {
            if (array[i] < 0 || array[i] >= array.length) {
                return -1;
            }
            while (array[i] != i) {
                if (array[i] == array[array[i]]) {
                    return array[i];
                } else {
                    int temp = array[i];
                    array[i] = array[array[i]];
                    array[temp] = temp;
                }
            }
        }
        return -1;
    }
}

记住先写测试用例再写代码!

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

推荐阅读更多精彩内容