日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

程序员面试题精选100题(63)-数组中三个只出现一次的数字[算法]

發布時間:2025/3/21 编程问答 18 豆豆
生活随笔 收集整理的這篇文章主要介紹了 程序员面试题精选100题(63)-数组中三个只出现一次的数字[算法] 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

題目:一個數組中有三個數字a、b、c只出現一次,其他數字都出現了兩次。請找出三個只出現一次的數字。

分析:在博客http://zhedahht.blog.163.com/blog/static/2541117420071128950682/中我們討論了如何在一個數組中找出兩個只出現一次的數字。在這道題中,如果我們能夠找出一個只出現一次的數字,剩下兩個只出現一次的數字就很容易找出來了。

如果我們把數組中所有數字都異或起來,那最終的結果(記為x)就是a、b、c三個數字的異或結果(x=a^b^c)。其他出現了兩次的數字在異或運算中相互抵消了。

我們可以證明異或的結果x不可能是a、b、c三個互不相同的數字中的任何一個。我們用反證法證明。假設x等于a、b、c中的某一個。比如x等于a,也就是a=a^b^c。因此b^c等于0,即b等于c。這與a、b、c是三個互不相同的三個數相矛盾。

由于x與a、b、c都各不相同,因此x^a、x^b、x^c都不等于0。

我們定義一個函數f(n),它的結果是保留數字n的二進制表示中的最后一位1,而把其他所有位都變成0。比如十進制6表示成二進制是0110,因此f(6)的結果為2(二進制為0010)。f(x^a)、f(x^b)、f(x^c)的結果均不等于0。

接著我們考慮f(x^a)^f(x^b)^f(x^c)的結果。由于對于非0的n,f(n)的結果的二進制表示中只有一個數位是1,因此f(x^a)^f(x^b)^f(x^c)的結果肯定不為0。這是因為對于任意三個非零的數i、j、k,f(i)^f(j)的結果要么為0,要么結果的二進制結果中有兩個1。不管是那種情況,f(i)^f(j)都不可能等于f(k),因為f(k)不等于0,并且結果的二進制中只有一位是1。

于是f(x^a)^f(x^b)^f(x^c)的結果的二進制中至少有一位是1。假設最后一位是1的位是第m位。那么x^a、x^b、x^c的結果中,有一個或者三個數字的第m位是1。

接下來我們證明x^a、x^b、x^c的三個結果第m位不可能都是1。還是用反證法證明。如果x^a、x^b、x^c的第m位都是1,那么a、b、c三個數字的第m位和x的第m位都相反,因此a、b、c三個數字的第m位相同。如果a、b、c三個數字的第m位都是0,x=a^b^c結果的第m位是0。由于x和a兩個數字的第m位都是0,x^a結果的第m位應該是0。同理可以證明x^b、x^c第m位都是0。這與我們的假設矛盾。如果a、b、c三個數字的第m位都是1,x=a^b^c結果的第m位是1。由于x和a兩個數字的第m位都是1,x^a結果的第m位應該是0。同理可以證明x^b、x^c第m位都是0。這還是與我們的假設矛盾。

因此x^a、x^b、x^c三個數字中,只有一個數字的第m位是1。于是我們找到了能夠區分a、b、c三個數字的標準。這三個數字中,只有一個數字滿足這個標準,而另外兩個數字不滿足。一旦這個滿足標準數字找出來之后,另外兩個數字也就可以找出來了。

這種思路的C++代碼如下:

void getThreeUnique(vector<int>& numbers, vector<int>& unique) {if(numbers.size() < 3)return;int xorResult = 0;vector<int>::iterator iter = numbers.begin();for(; iter != numbers.end(); ++iter)xorResult ^= *iter;int flags = 0;for(iter = numbers.begin(); iter != numbers.end(); ++iter)flags ^= lastBitOf1(xorResult ^ *iter);flags = lastBitOf1(flags);// get the first unique numberint first = 0;for(iter = numbers.begin(); iter != numbers.end(); ++iter){if(lastBitOf1(*iter ^ xorResult) == flags)first ^= *iter;}unique.push_back(first);// move the first unique number to the end of arrayfor(iter = numbers.begin(); iter != numbers.end(); ++iter){if(*iter == first){swap(*iter, *(numbers.end() - 1));break;}}// get the second and third unique numbersgetTwoUnique(numbers.begin(), numbers.end() - 1, unique); }int lastBitOf1(int number) {return number & ~(number - 1); }void getTwoUnique(vector<int>::iterator begin, vector<int>::iterator end, vector<int>& unique) {int xorResult = 0;for(vector<int>::iterator iter = begin; iter != end; ++iter)xorResult ^= *iter;int diff = lastBitOf1(xorResult);int first = 0;int second = 0;for(vector<int>::iterator iter = begin; iter != end; ++iter){if(diff & *iter)first ^= *iter;elsesecond ^= *iter;}unique.push_back(first);unique.push_back(second); }

上文中getThreeUnique從數組中找出三個只出現一次的數字,而getTwoUnique從數組中找出兩個只出現一次的數字。lastBitOf1實現分析中的函數f(n)的功能,它只保留數字n的二進制表示中的最后一位1,而把其他所有位都變成0。

在函數getThreeUnique中,我們通過第一個for循環把a、b、c三個數字異或的結果保存到xorResult中,接著在第二個for循環中求出f(x^a)^f(x^b)^f(x^c)并保存到變量flags中。在語句flags=lastBitOf1(flags)求出f(x^a)^f(x^b)^f(x^c)結果的二進制中最后一位是1的位。并根據這一數位求出第一個只出現一次的數字first。接著把first交換到數組的最后,并在數組的前n-1個數字中求出另外兩個只出現一次的數字。


博主何海濤對本博客文章享有著作權。網絡轉載請注明出處http://zhedahht.blog.163.com/。整理出版物請和作者聯系。

《新程序員》:云原生和全面數字化實踐50位技術專家共同創作,文字、視頻、音頻交互閱讀

總結

以上是生活随笔為你收集整理的程序员面试题精选100题(63)-数组中三个只出现一次的数字[算法]的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。