百度實習生招聘的一道大數據處理題目(上)

 題目爲:兩個200G大小的文件ABAB文件裏內容均爲無序的一行一個正整數字(不超過2^63),請設計方案,輸出兩個文件中均出現過的數字,使用一臺內存不超過16G、磁盤充足的機器。方案中指明使用java編程時使用到的關鍵工具類,以及爲什麼?

對於這種大數據量問題(至少對於一臺機器來說算是大數據了),使用MapReduce是最簡單的方式了。現在開源的最好的支持MapReduce的分佈式計算框架軟件就是Hadoop,而Hadoop是用Java寫的,整個運行時系統也是需要Java虛擬機的支持。所以這個問題放在了Java組,當然這只是我的猜測。

改日我再把MapReduce的代碼和實驗結果發上來,今天主要不是討論MapReduce。而是討論使用共享內存的編程模型解決這個問題的辦法。共享內存的並行編程模型現在比較常用的是多線程,GPU,還有OpenMP等。這裏使用多線程的方式來解決這個問題。

以下分析對文件的數據存儲重新做一個假設,假設數據都是以二進制的形式存儲,每個元素的數據類型爲uint64_t,即佔8個字節,所以一共有25Guint64_t類型的元素。

首先從數據結構的角度去考慮這個問題,對於一般的尋找兩個列表listAlistB中相同元素的問題,我們可以使用的最簡單的算法如下:

 

  1. foreach itemA in listA 
  2.  
  3. foreach itemB in listB 
  4.  
  5.     if(itemB == itemA) 
  6.  
  7.        print itemA; 
  8.  
  9.        break; 
  10.  
  11.     end if; 
  12.  
  13. end foreach; 
  14.  
  15. end foreach; 

len(listA)Mlen(listB)N。算法的時間複雜度爲OMN)。這裏假設M=N,則時間複雜度爲OM^2)。對於200G的文件,就意味着對於A文件的每一個數,都需要對B文件掃描一次。假設磁盤的讀取速度爲50MB/s,需要的時間爲:25G*200G/50MB/s =~ 10^14s =~ 3170979 年。也就是說如果用這種方法的話,大致需要300萬年。

當然沒有人會願意這麼做的,除非N的規模很小的數據才考慮這麼做。學過點數據結構的人都知道應該用二叉平衡樹,二叉查找樹,B+B-之類的樹形結構來存儲數據。這樣對第一個文件首先建立樹的時間複雜度爲NlogN,對第二個文件中的每個元素,在樹中查找是否有相同的數,查找複雜度爲logN。對所有元素查找的時間複雜度就爲NlogN。所以總時間複雜度爲2NlogN,即:ONlogN)。使用Big O這種時間複雜度分析方法一般是針對於數據都能夠放入內存的算法。對於這個問題,這樣分析就太樂觀了。首先,建立平衡樹的時候,內存會爆掉,而且可以爆掉幾十次了。就算是用硬盤索引技術,把一部分樹放在磁盤上,這樣在查找的時候會更加痛苦。因爲要根據樹的查找路徑不斷的移動磁頭需找新的索引位置,對於大規模數據,你要想不停的移動磁頭,那你就輸了。

還有一種方式就是借鑑MapReduce這種流處理方式,對兩個文件的數據進行排序,然後再對比是否有相同的數據。使用Big O的分析這個算法的時間複雜度,應該爲:NlogN+NlogN+N+N.所以時間複雜度爲:O(NlogN).這樣看起來和使用樹形數據結構的效果差不多。但是,這有一個顯著的優點就是你的數據可以從磁盤上“大塊大塊連續連續”的讀取,磁頭不需要跳動,這對於這種對上世紀大機械時代有懷舊感的老技術還是非常划得來的。

所以大致的處理方式如下,首先對200G文件A進行分塊排序,排序後每塊放入一個文件中,然後對文件B做同樣的處理。文件A對應的分塊有序文件分別爲A1A2A3…,AN(如圖1所示)。文件B對應的分塊有序文件分別爲B1B2B3…,BN(如圖2所示)。然後使用文件指針分別指向A1A2A3…ANB1B2B3…BN。接下來的方法就很類似於歸併排序的方式了,從A1AN所指向的文件中取出一個最小minA,然後從B1-BN所指向的文件中也取出一個最小值minB,判斷兩者是否相等,若minA==minB,則將這個值寫入到結果文件(指針fpResult所指向的文件),然後兩者指針都向後移動,minA>minB.則從取出minB對應的那個文件指針向後移動一個單元,接着再從B1-BN中輸出一個最小值。minA<minB的處理方式也是一樣的,如圖3所示。

 

李超

 

1 文件A分塊排序

 

李超

2 文件B分塊排序

 

李超

 

3 歸併尋找相同值

 

歸併的算法可以使用錦標賽算法,而不是用普通的在N個元素中尋找最小值的算法。

接下來詳細分析下,對每一塊分塊排序的方法,假設塊的大小爲1G,則需要以下三個步驟:

1.1G數據從磁盤中載入內存

2.1G數據進行排序

3.將排序後的結果寫入磁盤

首先我們需要一個計時程序,用來統計程序各個模塊運行時所耗費的時間,這個程序由timer.htimer.c構成:

/*timer.h*/

 

  1. /* 
  2.  
  3. * Author: Chaos Lee 
  4.  
  5. * Date: 2012-06-30 
  6.  
  7. * Description: interfaces for public to use timer 
  8.  
  9. */ 
  10.  
  11. #ifndef __TIMER_H_ 
  12.  
  13. #define __TIMER_H_ 
  14.  
  15. #include<stdio.h> 
  16.  
  17. #include<sys/time.h> 
  18.  
  19. void start_timer(); 
  20.  
  21. int get_elapsed_time(); 
  22.  
  23. #endif 

 

/*timer.c*/

 

  1. /* 
  2.  
  3. * Author: Chaos Lee 
  4.  
  5. * Date: 2012-06-30 
  6.  
  7. * Description: implementation of the timer 
  8.  
  9. */ 
  10.  
  11. #include "timer.h" 
  12.  
  13. static struct timeval start,end; 
  14.  
  15.   
  16.  
  17. void start_timer() 
  18.  
  19.  
  20.         gettimeofday(&start,NULL); 
  21.  
  22.  
  23.   
  24.  
  25. void restart_timer() 
  26.  
  27.  
  28.         start_timer(); 
  29.  
  30.  
  31.   
  32.  
  33. int get_elapsed_time() 
  34.  
  35.  
  36.         gettimeofday(&end,NULL); 
  37.  
  38.         return end.tv_sec-start.tv_sec + (end.tv_usec-start.tv_usec)/1000000; 
  39.  

其次,我們還需要一個程序用來產生隨機數文件,以下該程序的源代碼,使用的時候需要傳入一個參數N,則產生大小爲2^N 字節的文件。

/*random_generator.c*/

 

  1. /* 
  2.  
  3. * Author: Chaos Lee 
  4.  
  5. * Date: 2012-06-30 
  6.  
  7. * Description: Generating a file containing a given number of random elements whose type are uint64_t 
  8.  
  9. */ 
  10.  
  11. #include<stdio.h> 
  12.  
  13. #include<stdlib.h> 
  14.  
  15. #include<time.h> 
  16.  
  17. #include<stdint.h> 
  18.  
  19. #include<sys/time.h> 
  20.  
  21.   
  22.  
  23. #include "timer.h" 
  24.  
  25.   
  26.  
  27. int main(int argc,char *argv[]) 
  28.  
  29.  
  30.         int shift,tmp[2]; 
  31.  
  32.         FILE * fp; 
  33.  
  34.         int64_t size,i; 
  35.  
  36.         int elapsed_seconds; 
  37.  
  38.         start_timer(); 
  39.  
  40.         if(2 > argc) 
  41.  
  42.         { 
  43.  
  44.                 fprintf(stderr,"Usage:%s NUMBER\n",argv[0]); 
  45.  
  46.                 exit(1); 
  47.  
  48.         } 
  49.  
  50.         shift = atoi(argv[1]); 
  51.  
  52.         shift -= 3; 
  53.  
  54.         if(0 > shift) 
  55.  
  56.         { 
  57.  
  58.                 fprintf(stderr,"too small\n"); 
  59.  
  60.                 exit(1); 
  61.  
  62.         } 
  63.  
  64.         size = 1 << shift; 
  65.  
  66.         srand(time(NULL)); 
  67.  
  68.         fp = fopen("data.dat","wb"); 
  69.  
  70.         if(NULL == fp) 
  71.  
  72.         { 
  73.  
  74.                 fprintf(stderr,"file open error."); 
  75.  
  76.                 exit(1); 
  77.  
  78.         } 
  79.  
  80.         for(i=0;i<size;i++) 
  81.  
  82.         { 
  83.  
  84.                 tmp[1] = rand(); 
  85.  
  86.                 tmp[2] = rand(); 
  87.  
  88.                 if( 2 != fwrite(&tmp[0],sizeof(int),2,fp)) 
  89.  
  90.                 { 
  91.  
  92.                         fprintf(stderr,"writing file failure...\n"); 
  93.  
  94.                         exit(1); 
  95.  
  96.                 } 
  97.  
  98.         } 
  99.  
  100.         elapsed_seconds = get_elapsed_time(); 
  101.  
  102.         fprintf(stdout,"generating cost %d seconds.\n",elapsed_seconds); 
  103.  
  104.         fclose(fp); 
  105.  

對於數據排序可以使用單線程的排序方法或者多線程的排序方法。單線程的排序版本的源代碼如下:

 

/* single_thread_sort.c */

 

  1. /* 
  2.  
  3. * Author: Chaos Lee 
  4.  
  5. * Date: 2012-06-30 
  6.  
  7. * Description: load,sort,store data with single core 
  8.  
  9. */ 
  10.  
  11.   
  12.  
  13. #include<stdio.h> 
  14.  
  15. #include<stdlib.h> 
  16.  
  17. #include<stdint.h> 
  18.  
  19. #include<sys/types.h> 
  20.  
  21. #include<sys/stat.h> 
  22.  
  23.   
  24.  
  25. #include "../error.h" 
  26.  
  27. #include "timer.h" 
  28.  
  29.   
  30.  
  31. int uint64_compare(const void * ptr1,const void * ptr2) 
  32.  
  33.  
  34.         return  *((uint64_t *)ptr1) > *((uint64_t *)ptr2) ? 1 : *((uint64_t *)ptr1) < *((uint64_t *)ptr2) ? -1 : 0; 
  35.  
  36.  
  37.   
  38.  
  39. int main(int argc,char * argv[]) 
  40.  
  41.  
  42.         struct stat data_stat; 
  43.  
  44.         int status,elapsed_seconds;; 
  45.  
  46.         uint64_t size; 
  47.  
  48.         uint64_t *buffer; 
  49.  
  50.         FILE * fp; 
  51.  
  52.         FILE * fp_result; 
  53.  
  54.         status = stat("data.dat",&data_stat); 
  55.  
  56.         if(0 != status) 
  57.  
  58.                 error_abort("stat file error.\n"); 
  59.  
  60.         size = data_stat.st_size; 
  61.  
  62.         buffer = (uint64_t *) malloc(size); 
  63.  
  64.         if(NULL == buffer) 
  65.  
  66.         { 
  67.  
  68.                 fprintf(stderr,"mallocing error."); 
  69.  
  70.                 exit(1); 
  71.  
  72.         } 
  73.  
  74.         fp = fopen("data.dat","rb"); 
  75.  
  76.         if(NULL == fp) 
  77.  
  78.         { 
  79.  
  80.                 fprintf(stderr,"file open error."); 
  81.  
  82.                 exit(1); 
  83.  
  84.         } 
  85.  
  86.         start_timer(); 
  87.  
  88.         fread(buffer,size,1,fp); 
  89.  
  90.         elapsed_seconds = get_elapsed_time(); 
  91.  
  92.         fprintf(stdout,"loading cost %d seconds\n",elapsed_seconds); 
  93.  
  94.         restart_timer(); 
  95.  
  96.         qsort(buffer,size/sizeof(uint64_t),sizeof(uint64_t),uint64_compare); 
  97.  
  98.         elapsed_seconds = get_elapsed_time(); 
  99.  
  100.         fprintf(stdout,"sorting cost %d seconds\n",elapsed_seconds); 
  101.  
  102.         fp_result = fopen("single_result.dat","wb"); 
  103.  
  104.         if(NULL == fp_result) 
  105.  
  106.         { 
  107.  
  108.                 fprintf(stderr,"open result file error.\n"); 
  109.  
  110.                 exit(1); 
  111.  
  112.         } 
  113.  
  114.         restart_timer(); 
  115.  
  116.         fwrite(buffer,sizeof(uint64_t),size/sizeof(uint64_t),fp_result); 
  117.  
  118.         elapsed_seconds = get_elapsed_time(); 
  119.  
  120.         fprintf(stdout,"writing results cost %d seconds\n",elapsed_seconds); 
  121.  
  122.         free(buffer); 
  123.  
  124.         fclose(fp); 
  125.  
  126.         return 0; 
  127.  

單線程版本的運行時間和測試方法如下:

 

  1. [lichao@sg01 thread_power]$ gcc -c timer.c -o timer.o 
  2.  
  3. [lichao@sg01 thread_power]$ gcc random_generator.c -o random_generator timer.o 
  4.  
  5. [lichao@sg01 thread_power]$ ./random_generator 30 
  6.  
  7. generating cost 36 seconds. 

由此可見,創建1GB的文件耗時36秒,即寫的速度爲: 29826161B/s.差不多爲30MB/s。下面編譯單線程版本的代碼,並運行測試下時間:

 

  1. [lichao@sg01 thread_power]$ gcc single_thread_sort.c -o single_thread_sort timer.o -lpthread 
  2.  
  3. [lichao@sg01 thread_power]$ ./single_thread_sort 
  4.  
  5. loading cost 44 seconds 
  6.  
  7. sorting cost 85 seconds 
  8.  
  9. writing results cost 81 seconds 

 

 

李超

 

4 排序階段CPU使用率

由於篇幅所限,接下來的內容請看下一篇博文:百度實習生招聘的一道大數據處理題目(下)

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章