找一个数组中的主元素

问题:在一个规模为N的数组array[N]中,所谓主元素就是出现次数大于N/2的元素,例如 

                                           3.3.4.2.4.4.2.4.4  有一个主元素为4。

          给出一个算法,如果过半元素存在,就找出来,否则给出报告,要求给出O(N)的算法。

 

 

常规想法

 

(1) 穷举:找出元素中每一个数在数据中的数量。时间复杂度O(N^2)

(2) 排序:先对数组快排,然后重头开始遍历一遍计算每个数的数量。时间复杂度O(N*logN+N)

 

经典算法    裁剪数组算法, 时间复杂度为O(2N ) 

     思想: 如果一个数组array[N],其中有两个元素e1和e2。

      (1) e1不等于e2

      假如e1是数组array[N]的主元素,e2不是。那么e1在array[N]中的数量en>N/2。此时去掉array[N]中的e1和e2两个元素(e1!=e2)。那么新数组array[N-2]中e1的数量为en-1,并且en-1>N/2-1=(N-2)/2。即e1还是新数组array[N-2]的主元素。

      假如e1和e2都不是数组array[N]的主元素,那么去掉e1、e2以后。那么新数组的大小将变成N-2。此时很有可能出现一个新数组的主元素X,此主元素X的数量正好等于X=(N-2)/2+1。但是该主元素就不是原数组的主元素了,因为X= (N-2)/2+1=N/2。那么这样的主元素X我们叫做伪主元素。因此需要通过和裁剪掉元素比较,来确定是否是真正的主元素。

      (2) e1等于e2

      这种情况下不能进行裁剪。只能继续找不同的两个数裁减掉

 

C代码   收藏代码
  1. #include<stdio.h>    
  2. #include<malloc.h>  
  3.  // 算法源代码,裁剪当前数组中相邻的不同元素    
  4.  void main(){    
  5.     int pArr[6]={4,4,4,4,6,6};    
  6.     int arrLength=6;  //数组长度  
  7.     int element=pArr[0];    
  8.     int value=1;  //记录剪裁过程中遇到相同元素的个数  
  9.     int delNum=0; //记录裁剪数组的元素个数  
  10.     int *dArr=(int *)malloc(arrLength*sizeof(int)); //记录被剪裁的数组元素  
  11.     int dTop=0; //当前剪裁数组的索引位置  
  12.        
  13.     for(int i=1;i<arrLength;i++){    
  14.        if(value==0){    
  15.            element=pArr[i];    
  16.        }    
  17.        if(pArr[i]==element) //如果当前数组相邻的元素相等    
  18.            value++;    
  19.        else if(value>0){//如果当前数组相邻的元素不等,则需要裁剪得到新的数组    
  20.            dArr[dTop++]=element;  
  21.            dArr[dTop++]=pArr[i];  
  22.            delNum+=2;    
  23.            value--;    
  24.        }    
  25.     }    
  26.     //如果裁剪之后出现了主元素,那么这个主元素有可能是个伪主元素  
  27.     if(value>(arrLength-delNum)/2){  
  28.         //与裁剪掉的数组元素一一比较  
  29.         for(int j=0;j<delNum;j++)  
  30.             if(element==dArr[j]) value++;             
  31.         //确定真正的主元素  
  32.         if(value>arrLength/2)  
  33.             printf("主元素为:%d\n",element);    
  34.     }    

 

 

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值