欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

leetcode-169-Majority Element

程序员文章站 2022-05-04 09:27:50
题目描述: Given an array of size n, find the majority element. The majority element is the element that appears more than ⌊ n/2 ⌋ times. You may assume th ......

题目描述:

Given an array of size n, find the majority element. The majority element is the element that appears more than ⌊ n/2 ⌋ times.

You may assume that the array is non-empty and the majority element always exist in the array.

 

要完成的函数:

int majorityElement(vector<int>& nums) 

 

说明:

1、这道题目给定长度为n的数组,要求输出某个元素的值,该元素出现次数超过⌊n/2⌋。

2、不得不说,这道题目有很多种解法,最暴力的莫过于双重循环,统计各个元素的出现次数,统计到出现⌊n/2⌋次的就停止,然后输出。

3、但除此之外,我们还可以尝试快速排序,排序完数组中坐标为⌊n/2⌋的就是我们要输出的元素。快排代码如下:

#include<algorithm>
int majorityElement(vector<int>& nums)
{
    sort(nums.begin(),nums.end());
    return nums[nums.size()/2];
}

sort是头文件<algorithm>中的函数,采用的是类似于快排的方法,该函数一共有三个参数。

第一个参数是数组指针,第二个参数是数组中最后一项的下一项的指针。

最后一个参数是排序方法,可以使用c++内置的比较函数。记得要include<funtional>,然后less<int>()表示从小到大排序,这也是默认排序方法,还可以是greater<int>(),从大到小排序。

最后一个参数还可以是自定义函数,如下是从大到小的排序:

#include<iostream>
#include<vector>
#include<algorithm>
using namespace std;
bool compare(int a,int b)//自定义比较函数
{
    return a>b;
}
int main()
{
    vector<int>v={1,2,3,4,5,6,7,8,9};
    sort(v.begin(),v.end(),compare);//以函数名作为第三个参数,进行快排
    for(vector<int>::iterator iter=v.begin();iter<v.end();iter++)//vector迭代器的使用
    cout<<*iter<<" ";
    return 0;
}

快排算法beats 20% of cpp submissions

 

4、对3中的算法做些优化,我们是否可以只做几遍快速排序,当得到⌊n/2⌋位置的最终值时就停下呢?(我们知道快排每一遍排序都可以得到一个最终值)这样我们可以大大减少所需要的时间。

刚好c++中有nth_element函数,代码如下:

int majorityElement(vector<int>& nums) 
{
        nth_element(nums.begin(),nums.begin()+ nums.size()/2, nums.end());//这个函数进行部分快排,直到第二个参数表示的位置已经得到最终值
        return nums[nums.size()/2];
} 

nth_element原理如下:

首先选定一个基准值,比如我们常用的nums[0],进行第一次快排,nums[0]的值最终放到了mid位置上,这时候mid左边的值都比它小,右边的值都比它大。

接着看一下第二个参数表示的位置的值在mid的哪一边,然后对这一边继续进行快排,mid的另一边就不理了。

重复操作,直到第二个参数表示的位置的值成为最终值。

这种算法beats 79.89% of cpp submissions

 

5、上面我们一共说了三种算法,暴力双重循环,快排,部分快排。除此之外,在评论区还见到了其他三种惊为天人的算法,下面逐一与大家分享。

6、随机算法

代码如下,应该不难看懂~

#include<ctime>
int majorityElement(vector<int>& nums) { int n = nums.size(); srand(unsigned(time(NULL)));//以当前时间作为一个种子,NULL表示产生的时间值不被计算机存储,只被当前语句处理。生成的值要转化为unsigned int类型作为srand的参数。 while (true) { int idx= rand() % n; int candidate = nums[idx]; int counts = 0; for (int i = 0; i < n; i++) if (nums[i] == candidate) counts++; if (counts > n / 2) return candidate; } }

  这个算法虽然最差时间复杂度是无穷大,但是实际过程中由于我们要找的数占据了数组的一半以上,所以随机生成这个数的概率还是很高的。

  实测了五次,最好的是beat 97.24% of cpp submissions,最差的是11.17% of cpp submissions。

  这种做法还是挺新鲜的,碰到做不出的题目也可以用这种方法水过题目测试点。

 

7、成对抵消算法

先附上代码,很简洁的~

int majorityElement(vector<int>& nums) 
{ int major, counts = 0, n = nums.size(); for (int i = 0; i < n; i++)
    { if (!counts) //如果counts为0则改变major值,同时赋counts为1
       { major = nums[i]; counts = 1; } else counts += (nums[i] == major) ? 1 : -1; } return major; }

由于数组中,我们要找的元素出现了⌊n/2⌋次,因此我们可以,从数组中找出能成对抵消的元素,抵消到最后剩下的那一个就是我们要找的数了。

例如:数组为[2,1,2,3,2]

第一次循环,major=2,counts=1

第二次循环,counts=0

第三次循环,major=2,counts=1

第四次循环,counts=0

第五次循环,major=2,counts=1

最后的major就是我们要的数。

 

再举一个例子,数组为[3,3,1,3,1]

第一次循环,major=3,counts=1

第二次循环,counts=2

第三次循环,counts=1

第四次循环,counts=2,

第五次循环,counts=1

因为counts不会退到0,所以major也就不会改变,最后输出major=3

 

这种算法时间复杂度为O(n),是一种极为简单而又精彩的做法。

实测beats 79.89% of cpp submissions。

 

8、位操作算法

看到discuss区中大神提出了7中的成对抵消算法,笔者就想起了前几天做过的异或题目,相同的值可以抵消,但是这道题并不是相同值抵消,而是不同值,所以不能使用异或~

但是我们仍然可以使用位操作来完成这道题目。

我们要找的数出现了⌊n/2⌋次以上,从二进制的角度来看,也就是32位中的某一位出现了⌊n/2⌋次以上,根据这一点我们可以采取如下操作

int majorityElement(vector<int>& nums) 
{
        int major = 0, n = nums.size();
        for (int i = 0, mask = 1; i < 32; i++, mask <<= 1)
        {
            int bitCounts = 0;
            for (int j = 0; j < n; j++)
            {
                if (nums[j] & mask) //在mask代表的这一位上为1
            bitCounts++; if (bitCounts > n / 2) //总的出现次数在n/2以上,则我们要找的数在这一位上有出现,major要“或”上这一位 { major |= mask; break;//跳出内层循环 } } } return major; }

 

总结:

这虽然是一道简单的题目,但发散开来,我们可以总结出这么多种方法:

1、暴力双重循环法

2、快排

3、部分快排

4、随机算法

5、成对抵消算法

6、位操作法

其中的成对抵消算法最稳定,结果也优秀。

部分快排结果也挺优秀,不过花费的时间不会很稳定。

随机算法成绩可以很优秀,也可以很糟糕,实际体验就看命了,不过好处就是很简单地解出一些题目,trick。

快排时间复杂度高一点,传统解法。

暴力双重循环是最简单但是也最高时间复杂度的解法。

位操作也是双重循环,跟暴力解法差不多。

笔者还是最喜欢成对抵消的做法,简单稳定~

 

这道题目还有其他算法,由于时间关系就不介绍了,详见以下两个网址:

https://leetcode.com/problems/majority-element/discuss/51612/6-Suggested-Solutions-in-C++-with-Explanations

https://leetcode.com/problems/majority-element/solution/

本文章几种算法、思路和代码都来源于这两个网页介绍的内容。侵删。