青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

(轉)質數算法大全,及C程序實現優化詳解 (二) 篩選法

質數的定義

一個數,如果只有1和它本身兩個因數,這樣的數叫做質數,又稱素數。

在上文 《素數算法大全,及C程序實現優化詳解 (一) 試除法》中我們已經探討了求解素數的一類算法,并且將試除法從最初的低效版本優化的高效的V2。那么,還有沒有其它更佳算法呢?這就是下面三藏要和大家探討的內容

合數過濾篩選法

算法描述:我們知道,素數N不能被2~(N-1)間的任何數整除;反過來看,只要能被2~(N-1)間的任何數整除的N,都不是素數。所以我們可以采用一個簡單的排除法:就是對N以內的所有數,只要逐個去除值為2~(N-1)的倍數的數,剩下的就是素數。

C語言實現

// 合數過濾篩選法 Ver1 
// 參數:n 求解n以內(包括n)的素數
// 返回值:n以內素數個數
int CompositeNumFilterV1(int n)
{
 int i, j;
 // 素數數量統計
 int count = 0;
 // 分配素數標記空間,結合后文思考為何+1
 char* flag = (char*)malloc( n+1 );
 
 // 初始化素數標記
 for (i=2; i<=n; i++)
 {
  // 為什么*(p+i)要寫成flag[i]呢?可讀性更佳爾
  flag[i] = 1;
 }
 
 // 寫程序要注意排版和留空,方便閱讀,也可減少出錯幾率
 // 以2~(N-1)為因子過濾合數
 for (i=2; i < n; i++)
 {
  for (j=2; i*j <= n; j++)
  {
   // i*j是由i,j兩整數相乘而得,顯然不是素數
   flag[i*j] = 0;
  }
 }
 
 // 統計素數個數
 for (i=2; i<=n; i++)
 {
  // 其實if(flag)就其同樣作用了,但這么寫是有留言的
  // 請參閱《C語言程序設計常見錯誤剖析及解決之道》一文
  if (1 == flag[i]) count++;
 }
  
 // 因輸出費時,且和算法核心相關不大,故略
 
 // 釋放內存,別忘了傳說中的內存泄漏
 free(flag);
 
 return count;
}

在上文給出的main函數中以不同參數調用CompositeNumFilterV1函數,得到執行結果如下:

[100000]以內素數個數:9592, 計算用時:15毫秒
[1000000]以內素數個數:78498, 計算用時:125毫秒
[5000000]以內素數個數:348513, 計算用時:2578毫秒
[10000000]以內素數個數:664579, 計算用時:6281毫秒

注:因程序是非獨占性運行的,所以時間不是完全精確的,但基本能反映實情

顯然,比上文中的試除法要快,而且誰都可以看到上例是一個未經優化的粗陋版本,好多地方是三藏故意采用比較低效做法,為了與后文的優化版比較,凸顯優化之重要,也為了初學者記住別采用類似低效做法,下面我們開始優化之旅

優化分析

上面CompositeNumFilterV1函數存在的問題有:

  1. 在外層循環,需要一直執行到n-1嗎?不要,因為n/2~n-1間的數顯然不能整出n
  2. 在內層循環中重復使用i*j顯然是低效的,考慮到計算機中加減運算速度比乘除快,可以考慮變乘法為加法
  3. 在循環修改flag過程中,其實有很多數會被重復計算若干次,比如6=2*3=3*2,會被重復置0,類似操作很多,所以我們得設法避免或減少flag重復置0

據上述分析,我們可將程序優化如下:

// 合數過濾篩選法 Ver2 
// 參數:n 求解n以內(包括n)的素數
// 返回值:n以內素數個數
int CompositeNumFilterV2(int n)
{
 int i, j;
 // 素數數量統計
 int count = 0;
 // 分配素數標記空間,明白+1原因了吧,因為浪費了一個flag[0]
 char* flag = (char*)malloc( n+1 );
 
 // 初始化素數標記,要高效點咯
 flag[2] = 1;
 // 注意是i<n不是上例中的i<=n了,理由自思
 for (i=3; i<n; i++)
 {
  flag[i++] = 1;
  // 偶數自然不是素數,直接置0好了
  flag[i] = 0;
 }
 // n為奇數
 if (n%2 != 0)
 {
  flag[n] = 1;
 }
 
 // 從3開始filter,因為2的倍數早在初始化時代就干掉了
 // 到n/2止的理由還要說嗎
 for (i=3; i <= n/2; i++)
 {
  // i是合數,請歇著吧,因為您的工作早有您的質因子代勞了
  if (0 == flag[i]) continue;
  
  // 從i的2倍開始過濾,變乘法為加法 
  for (j=i+i; j <= n; j+=i)
  {
   flag[j] = 0;
  }
 }
 
 // 統計素數個數
 for (i=2; i<=n; i++)
 {
  if (flag[i]) count++;
 }
  
 // 因輸出費時,且和算法核心相關不大,故略
 
 // 釋放內存,別忘了傳說中的內存泄漏
 free(flag);
 
 return count;
}

再來調用CompositeNumFilterV2得到執行結果:

[100000]以內素數個數:9592, 計算用時:n太小,時間精度不夠
[1000000]以內素數個數:78498, 計算用時:31毫秒
[5000000]以內素數個數:348513, 計算用時:453毫秒
[10000000]以內素數個數:664579, 計算用時:1062毫秒
[100000000]以內素數個數:5761455, 計算用時:12973毫秒

哇哇,比昨天的試除發快了好多倍,可見算法的威力,值得好好學習,別說學算法沒用咯。

上例著那個計算一億以內的素數只要約13秒,應該算不錯了,今天是否可以休息了呢?No,我們要追求極限!

int CompositeNumFilterV3(int n)
{
 int i, j;
 // 素數數量統計
 int count = 0;
 // 分配素數標記空間,明白+1原因了吧,因為浪費了一個flag[0]
 char* flag = (char*)malloc( n+1 );
 // 干嘛用的,請仔細研究下文
 int mpLen = 2*3*5*7*11*13;
 char magicPattern[mpLen];
 // 奇怪的代碼,why,思考無法代勞,想!
 for (i=0; i<mpLen; i++)
 {
  magicPattern[i++] = 1;
  magicPattern[i++] = 0;
  magicPattern[i++] = 0;
  magicPattern[i++] = 0;
  magicPattern[i++] = 1;
  magicPattern[i] = 0;
 }
 for (i=4; i<=mpLen; i+=5)
  magicPattern[i] = 0;
 for (i=6; i<=mpLen; i+=7)
  magicPattern[i] = 0;
 for (i=10; i<=mpLen; i+=11)
  magicPattern[i] = 0;
 for (i=12; i<=mpLen; i+=13)
  magicPattern[i] = 0;
 
 // 新的初始化方法,將2,3,5,7,11,13的倍數全干掉
 // 而且采用memcpy以mpLen長的magicPattern來批量處理
 int remainder = n%mpLen;
 char* p = flag+1;
 char* pstop = p+n-remainder;
 while (p < pstop)
 {
  memcpy(p, magicPattern, mpLen);
  p += mpLen;
 }
 if (remainder > 0)
 {
  memcpy(p, magicPattern, remainder);
 }
 flag[2] = 1;
 flag[3] = 1;
 flag[5] = 1;
 flag[7] = 1;
 flag[11] = 1;
 flag[13] = 1;
 
 // 從17開始filter,因為2,3,5,7,11,13的倍數早被kill了
 // 到n/13止的,哈哈,少了好多吧
 int stop = n/13;
 for (i=17; i <= stop; i++)
 {
  // i是合數,請歇著吧,因為您的工作早有您的質因子代勞了
  if (0 == flag[i]) continue;
  
  // 從i的17倍開始過濾
  int step = i*2;
  for (j=i*17; j <= n; j+=step)
  {
   flag[j] = 0;
  }
 }
 
 // 統計素數個數
 for (i=2; i<=n; i++)
 {
  if (flag[i]) count++;
 }
  
 // 因輸出費時,且和算法核心相關不大,故略
 
 // 釋放內存,別忘了傳說中的內存泄漏
 free(flag);
 
 return count;
}

再看CompositeNumFilterV3執行結果:

[1000000]以內素數個數:78498, 計算用時:15毫秒
[5000000]以內素數個數:348513, 計算用時:203毫秒
[10000000]以內素數個數:664579, 計算用時:515毫秒
[100000000]以內素數個數:5761455, 計算用時:6421毫秒

再次優化后速度提升了又一倍左右,三藏不禁有點滿足了,睡覺也!

posted on 2009-05-14 15:48 小蟲蟲 閱讀(1051) 評論(0)  編輯 收藏 引用

<2009年5月>
262728293012
3456789
10111213141516
17181920212223
24252627282930
31123456

導航

統計

常用鏈接

留言簿(5)

隨筆分類

隨筆檔案

搜索

最新評論

閱讀排行榜

評論排行榜

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            亚洲欧美日韩在线不卡| 亚洲视频999| 亚洲欧洲在线观看| 在线亚洲一区| 久久狠狠婷婷| 亚洲日本在线观看| 亚洲一区影院| 蜜乳av另类精品一区二区| 欧美日韩一区二区三区在线| 国产精品一卡二卡| 亚洲精品国产精品国自产观看浪潮 | 午夜在线精品偷拍| 老司机午夜精品视频在线观看| 欧美黄色一区二区| 欧美一级黄色录像| 欧美日韩亚洲在线| 亚洲狠狠婷婷| 久久男人资源视频| 国产精品99久久久久久www| 午夜精品久久久久久| 欧美精品v日韩精品v国产精品| 国产视频在线观看一区二区三区| 亚洲精品永久免费| 老司机亚洲精品| 午夜精品一区二区三区电影天堂 | 国产精品一区免费视频| 99精品视频网| 欧美国产第二页| 久久精品成人| 国产日韩欧美自拍| 亚洲综合首页| 日韩亚洲视频在线| 欧美区在线播放| 亚洲欧洲在线一区| 欧美激情精品久久久久久变态| 久久成人这里只有精品| 国产视频在线一区二区| 欧美在现视频| 香蕉久久a毛片| 国产精品自拍在线| 欧美一区二区大片| 亚洲男女自偷自拍| 国产精品视频免费| 久久精品视频在线看| 午夜伦理片一区| 国产亚洲精品久久久| 香蕉成人啪国产精品视频综合网| 99精品热视频只有精品10| 欧美另类人妖| 亚洲一区在线播放| 亚洲视频1区| 国产女同一区二区| 久久久青草青青国产亚洲免观| 亚洲欧美综合精品久久成人| 国产日韩精品入口| 免费日韩成人| 欧美激情第4页| 亚洲香蕉视频| 午夜精品一区二区三区电影天堂 | 在线看一区二区| 欧美华人在线视频| 欧美日韩国产区| 午夜在线精品偷拍| 欧美日韩免费网站| 亚洲一区二区三区中文字幕| 亚洲精品偷拍| 国产麻豆日韩| 欧美激情国产日韩| 欧美午夜a级限制福利片| 久久国产主播| 欧美久久精品午夜青青大伊人| 亚洲一区二区在线| 欧美中文字幕视频在线观看| 亚洲国产精品一区二区第一页| 亚洲精品日韩欧美| 国内精品福利| 亚洲国产二区| 国产精品一级| 亚洲福利视频网| 国产乱码精品一区二区三| 牛人盗摄一区二区三区视频| 欧美日韩国产精品| 久久亚洲春色中文字幕久久久| 欧美电影免费| 久久精品日产第一区二区| 老牛嫩草一区二区三区日本| 亚洲一区二区视频| 免费成人黄色片| 久久国产精品色婷婷| 欧美日韩aaaaa| 欧美.www| 国产一区二区丝袜高跟鞋图片 | 久久久久久久网站| 欧美视频一区二区三区…| 免费日韩成人| 国产亚洲免费的视频看| 亚洲最新视频在线播放| 在线欧美不卡| 久久成人这里只有精品| 小嫩嫩精品导航| 欧美日韩亚洲免费| 亚洲夫妻自拍| 亚洲国产欧美一区| 久久成人免费电影| 久久国产精品黑丝| 国产精品一区视频网站| 亚洲精品久久视频| 亚洲国产综合视频在线观看| 亚洲在线视频网站| 午夜精品成人在线视频| 欧美日韩喷水| 亚洲裸体在线观看| 一本色道久久| 欧美日韩国产一中文字不卡| 欧美激情一区二区三级高清视频| 狠狠色狠狠色综合日日小说| 欧美一级久久| 久久一区二区精品| 国内一区二区三区在线视频| 午夜精品国产精品大乳美女| 午夜综合激情| 国产欧美日韩一区二区三区在线观看 | 在线欧美影院| 久久久噜噜噜久噜久久 | 亚洲一区二区三区中文字幕在线| 亚洲国产视频一区| 欧美成人免费全部观看天天性色| 你懂的一区二区| 亚洲三级性片| 欧美无乱码久久久免费午夜一区 | 欧美成人亚洲| 亚洲美女毛片| 国产精品sm| 午夜日韩av| 蜜桃伊人久久| 亚洲破处大片| 欧美日韩亚洲一区| 亚洲制服欧美中文字幕中文字幕| 亚洲欧美变态国产另类| 国产一区二区三区成人欧美日韩在线观看 | avtt综合网| 国产精品草草| 久久精品成人一区二区三区蜜臀| 欧美成人激情在线| 亚洲视频在线一区| 国产一区av在线| 欧美二区乱c少妇| 亚洲午夜精品网| 免费观看成人网| 亚洲特级片在线| 国产亚洲精品久久久久动| 免播放器亚洲| 亚洲一区中文字幕在线观看| 久久影院亚洲| 亚洲婷婷综合色高清在线| 国产一区二区三区直播精品电影| 免费观看一级特黄欧美大片| 99精品国产高清一区二区| 久久久久久久久久码影片| 99re66热这里只有精品3直播| 国产精品中文在线| 欧美精品大片| 欧美在线视频不卡| 一道本一区二区| 你懂的成人av| 久久精品国产69国产精品亚洲| 亚洲精品视频在线播放| 国产欧美日韩综合| 欧美日韩理论| 美日韩免费视频| 久久福利精品| 亚洲一二区在线| 亚洲精品欧美精品| 欧美大秀在线观看| 久久九九国产精品怡红院| 亚洲一区二区三区高清| 亚洲日本va午夜在线影院| 国内成+人亚洲| 国产精品一区三区| 国产精品成人观看视频免费| 欧美国产综合一区二区| 久久久久久久久久久久久9999| 亚洲一级在线| 99视频热这里只有精品免费| 亚洲高清久久久| 欧美国产先锋| 欧美bbbxxxxx| 免费黄网站欧美| 久久综合九色综合网站| 欧美亚洲日本网站| 午夜精品久久久久久久99水蜜桃 | 亚洲一品av免费观看| 欧美成人一区二区| 欧美在线视频免费播放| 亚洲女性裸体视频| 亚洲免费在线电影| 亚洲午夜激情| 亚洲综合大片69999| 亚洲视频一区在线观看| 亚洲午夜久久久|