寻找数组中出现次数超过一半以上的元素

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

http://www.nowcoder.com/questionTerminal/e8a1b01a2df14cb2b228b30ee6a92163

这道题,最直白的解法是使用一个 map 来记录各个数字出现的次数,最后取出现次数最多的作为解。但这个方法需要消耗额外的空间,不是最优。下面是最优解法:

我们做这样的想象,现在有来自不同阵营的多支部队,他们互为敌人。每个士兵都容不得敌人,宁愿与敌人同归于尽。可以想象,如果某个阵营的士兵数量超过所有阵营士兵总数的一半,该阵营士兵一换一带走一个其他阵营的,最终剩下的就是该阵营的士兵了,该阵营就获胜了。

这和本题有什么关系呢?且看下面的故事:

现在要打仗了,所有士兵依次进入战场,如果战场上有其他阵营的士兵,他就与其中一个同归于尽。否则,他就留在战场上。容易想象,战场上要么没有人,要么是同一阵营的。所有士兵都进入战场后,最终战场上剩下的只会是同一阵营的,而该阵营就是人数过半的那个。

我们要模拟这个过程,找到那个获胜的阵营。数组中每个元素就是士兵,元素的值,就是这个士兵所属阵营。

注意:这个题解是我在 leetcode https://leetcode-cn.com/problems/majority-element 上的题解,我把它贴过来了,在牛客网上需要做些许修改。

class Solution {
public:
    int majorityElement(vector<int> nums) {
        int count = 0;
        int win;

        for(int n: nums){
            // 战场上没人
            if(count == 0){
                win = n;
                count = 1;
            }else{
                // 同一阵营
                if(win == n){
                    count++;
                }
                // 不同阵营
                else{
                    count--;
                }
            }
        }
        return win;
    }
};

补充说明

如果可以保证某个阵营的人数过半,那么最后留下的就一定是这个阵营的士兵。但如果没有任何一个阵营人数过半,那么某个阵营可以猫在最后,在其他阵营都厮杀结束后进入战场,坐收渔翁之利。

因此,如果不存出现过半的数,上面的算法是错误的,需要加一个检验判断一下 win 是否出现了一般以上。

全部评论
谢谢,这种解释很形象易懂!
3 回复 分享
发布于 2020-02-10 20:34
有最后一个循环岂不是脱裤子放屁了
1 回复 分享
发布于 2020-12-17 15:15
学到了,老哥
1 回复 分享
发布于 2020-03-07 21:28
???
1 回复 分享
发布于 2020-02-12 20:07
问一下博主 这句话: for(int n: nums){} 能在C++下跑通吗?
点赞 回复 分享
发布于 2021-09-30 17:41
绝了。。
点赞 回复 分享
发布于 2021-06-05 16:44
正反各来一次,判断最后是否一样就行
点赞 回复 分享
发布于 2021-04-01 08:54
这个解法没有考虑数组个数是奇数的情况,类似的[1,2,3,2,4,2,5,2,3],会输出最后一个字符
点赞 回复 分享
发布于 2021-03-30 07:10
牛逼啊,对于本题需要最后加个循环判断, // 最后在判断留下的数是否超过一半 count = 0; for(int arr:array) { if(arr == win) { count ++; } } return count > array.length / 2 ? win : 0;
点赞 回复 分享
发布于 2020-10-02 17:03
所以应该像官方题解那样,最后再统计一下出现最后一个数的出现次数。
点赞 回复 分享
发布于 2020-07-11 21:59
牛皮 代码界的霍金
点赞 回复 分享
发布于 2020-07-06 23:33
1 2 1 2 1 2 3 3,输出3?
点赞 回复 分享
发布于 2020-05-24 14:54
牛批学到了
点赞 回复 分享
发布于 2020-03-23 22:43

相关推荐

会飞的猿:我看你想进大厂,我给你总结一下学习路线吧,java语言方面常规八股要熟,那些java的集合,重点背hashmap八股吧,jvm类加载机制,运行时分区,垃圾回收算法,垃圾回收器CMS、G1这些,各种乐观锁悲观锁,线程安全,threadlocal这些。在进阶一些的比如jvm参数,内存溢出泄漏排查,jvm调优。我这里说的只是冰山一角,详细八股可以去网上找,这不用去买,都免费资源。mysql、redis可以去看小林coding,我看你简历上写了,你一定要熟,什么底层b+树、索引结构、innodb、mvcc、undo log、redo log、行级锁表级锁,这些东西高频出现,如果面试官问我这些我都能笑出来。消息队列rabbitmq也好kafka也好,学一种就行,什么分区啊副本啊确认机制啊怎么保证不重复消费、怎么保证消息不丢失这些基本的一定要会,进阶一点的比如LEO、高水位线、kafka和rocketmq底层零拷贝的区别等等。计算机网络和操作系统既然你是科班应该理解起来问题不大,去看小林coding这两块吧,深度够了。spring boot的八股好好看看吧,一般字节腾讯不这么问,其他的java大厂挺爱问的,什么循环依赖啥的去网上看看。数据结构的话科班应该问题不大,多去力扣集中突击刷题吧。项目的话其实说白了还是结合八股来,想一想你写的这些技术会给你挖什么坑。除此之外,还有场景题、rpc、设计模式、linux命令、ddd等。不会的就别往简历上写了,虽然技术栈很多的话好看些,但背起来确实累。总结一下,多去实习吧,多跳槽,直到跳到一个不错的中厂做跳板,这是一条可行的进大厂的路线。另外,只想找个小厂的工作的话,没必要全都照这些准备,太累了,重点放在框架的使用和一些基础八股吧。大致路线就这样,没啥太多难度,就是量大,你能达到什么高度取决于你对自己多狠,祝好。
点赞 评论 收藏
分享
Aaso:挺好的,早挂早超生
点赞 评论 收藏
分享
评论
69
9
分享

创作者周榜

更多
牛客网
牛客企业服务