数组中出现次数超过一半的数字(C++剑指Offer详解)


第一次见到这个题,感觉简单到爆,不就是sort函数排个序,取中位数,遍历一遍计个数不就OK了吗?但是面试时,sort可能会让你自己去实现,或者说,题目要求:不能修改输入的数组呢???
第一种方法(基于Partition函数)
注意:此方法修改了输入的数组

核心:出现次数超过一半的数字,排序后中位数肯定是这个数字

解题思路:通过Partition函数,随机选出数组中的一个元素(我的代码中默认是数组中第一个元素),并且通过快排的思想,小的元素移到随机值前面,大的元素移到随机值的后面,然后返回随机值最后的位置。然后通过调用Partition函数,用返回的index值和中值判等,如果index>mid,说明中位数在左边,end=mid-1,如果index<mid,说明中位数在右边,start=mid+1,直到最后等于mid,然后numbers[mid]作为出现最多的数字,最后通过Check函数,判断出现次数是否超过数组的一半

class Solution {
public:
    int MoreThanHalfNum_Solution(vector<int> numbers)
    {
        int length = numbers.size();
        if (length <= 0)//判空
            return 0;
        int start = 0;
        int end = length - 1;
        int index = Partition(numbers, start, end);//Partition函数随机值在数组中的位置
        int mid = length >> 1;//统计学的中位数
        while (index != mid)//随机值与中位数位置的判等
        {
            if (index>mid)
            {
                end = index - 1;
                index = Partition(numbers, start, end);
            }
            else if (index<mid)
            {
                start = index + 1;
                index = Partition(numbers, start, end);
            }
        }
        if (!Check(numbers, numbers[index]))//判断次数是否超过一半
            return 0;
        return numbers[index];
    }
    int Partition(vector<int> num, int start, int end)
    {
        int index = start;
        int sum = num[start];//默认随机值为首位,这里可以用“随机函数”
        //从第二位开始
        for (int i = start + 1; i<end; ++i)
        {
            if (num[i]<sum)
            {
                ++index;
                swap(num[i], num[index]);
            }
        }
        swap(num[start], num[index]);//随机值居中(左小右大)
        return index;
    }
    bool Check(vector<int> num, int result)
    {
        int count = 0;
        for (int i = 0; i<num.size(); ++i)
        {
            if (num[i] == result)
                ++count;
        }
        if (count * 2>num.size())
            return true;
        return false;
    }
};

第二种方法(计数法)
注意:此方法没有修改输入的数组

核心:出现次数超过一半的数字个数比其他所有数字个数至少多一

解题思路:从首位开始(result==numbers[0]),遇到相等的元素,count++,遇到不相等的元素,count--,当count==0时,result重新赋值为当前元素,重新给count赋值为1,直到数组遍历完之后,返回result

疑惑???既然,判断出来了,为什么还要调用Check函数,因为我们无法判断result是否为出现次数最多的元素,就算是出现次数最多的元素,不知是否超过数组一半,例如:{5,5,5,5,5,3,3,3,4,4,4,4} {3,3,3,4,4,4,4,5,5,5,5,5}

class Solution {
public:
    int MoreThanHalfNum_Solution(vector<int> numbers) {
        if(numbers.size()<=0)
            return 0;
        int result=numbers[0];
        int i=1;
        int count=1;
        for(;i<numbers.size();++i)
        {
            if(count==0)
            {
                result=numbers[i];
                count=1;
            }
            else if(result==numbers[i])
                ++count;
            else
                --count;
        }
        if(!Check(numbers,result))
                result=0;
        return result;
    }
    bool Check(vector<int> num,int result)
    {
        int count=0;
        for(int i=0;i<num.size();++i)
        {
            if(num[i]==result)
                ++count;
        }
        if(count*2>num.size())
            return true;
        return false;
    }

原文地址:https://blog.51cto.com/14233078/2469750

时间: 2024-08-04 08:28:52

数组中出现次数超过一半的数字(C++剑指Offer详解)的相关文章

编程算法 - 数组中出现次数超过一半的数字 代码(C)

数组中出现次数超过一半的数字 代码(C) 本文地址: http://blog.csdn.net/caroline_wendy 题目: 数组中有一个数字出现的次数超过数组长度的一半, 请找出这个数字. 1. 使用高速排序(QuickSort)的方法, 把中值(middle)和索引(index)匹配, 输出中值, 并检測是否符合要求. 2. 使用计数方法依次比較. 代码:  方法1: /* * main.cpp * * Created on: 2014.6.12 * Author: Spike */

【剑指offer】Q29:数组中出现次数超过一半的数字

就本题而言,个人觉得练习下partition函数是有必要的,毕竟它是快速排序的核心,是基础性的东西,也是必须要掌握的,至于书中给出的"取巧"性解法,是属于个人思维能力的考察,是一种考虑问题的思路,不是一两个问题就能练就的. partition函数,包括快速排序,是一定要信手拈来的,必须的. import random def MoreThanHalf(array): if len(array) == 0: return 0 start = 0 end = len(array) - 1

47. 数组中出现次数超过一半的数字[Number appears more than half times]

[题目]:数组中有一个数字出现的次数超过了数组长度的一半,找出这个数字. 例如长度为9的数组{1,2,3,2,2,2,5,4,2}中次数超过了数组长度的一半的数字为2,而长度为8的数组{1,2,3,2,2,2,5,4}则为非法输入. [思路一]:先对数组进行排序,再遍历排序后的数组,统计每个数的次数,出现次数最大的数即为要找的数. 时间复杂度:O(nlgn)+ O(n)= O(nlgn):空间复杂度:O(1). [思路二]:先对数组进行排序,出现次数超过数组长度的一半的数必然是数组中间的那个数.

数据结构-数组中出现次数超过一半的数字

题目:数字中有一个数字出现的次数超过数组长度的一半,请找出这个数字.例如输入一个长度为9的数组{1,2,3,2,2,2,5,4,2}.由于数字2在数组中出现了5次,超过数组长度的一半,因此输出2. 分析:首先进行排序,因为大于一半,所以说经过排序之后的数组,如有次数大于的话说明那个数肯定是数组中间的那个数. #include <iostream> using namespace std; void QickSort(int *a,int n,int low,int high){ int i=l

【剑指offer】十九,数组中出现次数超过一半的数字

题目描述 数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字.例如输入一个长度为9的数组{1,2,3,2,2,2,5,4,2}.由于数字2在数组中出现了5次,超过数组长度的一半,因此输出2. 分析:找数组中出现次数超过一半的数字,我们一半可以采用两种方法,一是使用快排,对数组进行排序,然后直接输出排序数组中间位置的数.第二种是使用HashMap<Integer,Integer>,key是数组中的数字,value为其在数组中出现的次数,顺序扫描数组,记录下数组出现的次数,输出大于数组长

数组中出现次数超过一半的数字-剑指Offer

数组中出现次数超过一半的数字 题目描述 数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字.例如输入一个长度为9的数组{1,2,3,2,2,2,5,4,2}.由于数字2在数组中出现了5次,超过数组长度的一半,因此输出2.如果不存在则输出0. 思路 若一个数出现的次数超过长度的一半,那么这个数出现的次数比数组里的其他数都多 用num来存数组的数,times来存次数,遍历数组,遇到一个数若跟num相等,times++,若不相等,times--,若times为零,则num存入下一个数,最后n

剑指offer系列源码-数组中出现次数超过一半的数字

题目1370:数组中出现次数超过一半的数字 时间限制:1 秒内存限制:32 兆特殊判题:否提交:2844解决:846 题目描述: 数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字.例如输入一个长度为9的数组{1,2,3,2,2,2,5,4,2}.由于数字2在数组中出现了5次,超过数组长度的一半,因此输出2. 输入: 每个测试案例包括2行: 第一行输入一个整数n(1<=n<=100000),表示数组中元素的个数. 第二行输入n个整数,表示数组中的每个元素,这n个整数的范围是[1,10

剑指OFFER之数组中出现次数超过一半的数字(九度OJ1370)

题目描述: 数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字.例如输入一个长度为9的数组{1,2,3,2,2,2,5,4,2}.由于数字2在数组中出现了5次,超过数组长度的一半,因此输出2. 输入: 每个测试案例包括2行: 第一行输入一个整数n(1<=n<=100000),表示数组中元素的个数. 第二行输入n个整数,表示数组中的每个元素,这n个整数的范围是[1,1000000000]. 输出: 对应每个测试案例,输出出现的次数超过数组长度的一半的数,如果没有输出-1. 样例输入:

25.数组中出现次数超过一半的数字

数组中出现次数超过一半的数字 参与人数:3419时间限制:1秒空间限制:32768K 本题知识点: 数组 算法知识视频讲解 题目描述 数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字.例如输入一个长度为9的数组{1,2,3,2,2,2,5,4,2}.由于数字2在数组中出现了5次,超过数组长度的一半,因此输出2.如果不存在则输出0. 这道题很简单,可是我却做了很久,中间想到了用multimap记录键值对(以数组的元素做键,元素出现的次数做值),然后合并键,可是multimap好像没有这