码迷,mamicode.com
首页 > 编程语言 > 详细

面试40-一个数组,有2个数字出现奇数次,其余都是偶数次,求这两个数字O(n) O(1)

时间:2015-09-10 17:21:17      阅读:198      评论:0      收藏:0      [点我收藏+]

标签:

#include<iostream>
using namespace std;
// 题目:数组中只有不多于两个数字出现次数是奇数次,其他都是偶数次,求出出现奇数次的数字(不含0的数组)

//思想:
/*
(1)如果只有一个数字是奇数次,直接对数组进行按位异或运算,得到的结果就是该数
(2)如果有俩个,可以先对数组异或,得到的结果(就是两个奇数次的数字异或的结果),必定至少包含一个1,可以根据这个1在的位置,把数组分为两个部分
     则两个奇数次的数字必定分别在两个部分,而相同的数次必定在同一组,则可以对两个部分分别求异或得到答案
*/
int findNumsAppear(int arr[],int len,pair<int,int>& st);// 返回0表示没有,1表示一个,2表示2个,-1表示参数错误,没有考虑更多的情况
int count1(int num);// 统计int在内存中1的个数
int main()
{
    pair<int,int>st;
    st.first=0x80000000;
    st.second=0x80000000;
    int arr[]={2,2,2,1,1,4,4,4,4,3,5,5,6,6,6,6,7,7,8,8,9,9,10,10,11,11};
    cout<<findNumsAppear(arr,sizeof(arr)/sizeof(int),st)<<endl;
    cout<<st.first<<"   "<<st.second<<endl;
    return 0;
}
int findNumsAppear(int arr[],int len,pair<int,int>& st)
{
    if(arr==NULL||len<=0)
        return -1;
    int num=arr[0];
    int count=0;
    for(int i=1;i<len;i++)
        num^=arr[i];
    if(num==0)
        return 0;
    else  // 找到num 中第一个1,让其他位置1为0   此处参考count1的方法
    {
        while(num)
        {
            if((num&(num-1))==0)
                break;
            else
                num=num&(num-1);   // 最终会保留下最高位的一个1,用来划分数组
        }
    }
    int i=0;int j=len-1;
    int tmp;
    while(i<j)   // 循环最终结果为i==j 左半边数组的最后个元素下标
    {
        while(i<j&&(arr[j]&num)==0)
            j--;
        while(i<j&&(arr[i]&num)==num)
            i++;
        if(i<j)
        {
            tmp=arr[i];
            arr[i]=arr[j];
            arr[j]=tmp;
        }
    }
    int num1=arr[0];
    for(int k=1;k<=i;k++)
        num1^=arr[k];

    int num2=arr[len-1];
    for(int k=len-2;k>i;--k)
        num2^=arr[k];
    if(num1!=0)
        st.first=num1;
    if(num2!=0)
        st.second=num2;
    if(num1!=0&&num2!=0)
        return 2;
    else return 1;
}
int count1(int num) // 如果不许用位运算,直接 unsigned int n=(unsigned int)num; 用%  / 计算    -1内存全部为FFFFFFFF 转为最大的unsigned int
{
    int count=0;
    while(num)
    {
        ++count;
        num=num&(num-1);
    }
    return count;
}

 

面试40-一个数组,有2个数字出现奇数次,其余都是偶数次,求这两个数字O(n) O(1)

标签:

原文地址:http://www.cnblogs.com/fchy822/p/4798414.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!