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

(轉)質數算法大全,及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>
            亚洲欧美影音先锋| 欧美亚洲网站| 亚洲免费影视第一页| 亚洲激情偷拍| 在线观看日韩精品| 亚洲国产91| 亚洲精品乱码久久久久久久久| 黄色成人av网站| 91久久夜色精品国产九色| 亚洲人成毛片在线播放| 亚洲伦理在线| 亚洲自拍偷拍视频| 久久精品盗摄| 欧美国产亚洲另类动漫| 最新国产乱人伦偷精品免费网站| 麻豆精品视频在线观看视频| 亚洲高清av在线| 一区二区三区欧美日韩| 欧美一区二区三区精品电影| 久久久久久夜精品精品免费| 欧美成人精品| 国产精品色午夜在线观看| 激情成人av在线| 在线一区亚洲| 久久午夜电影网| 日韩午夜电影av| 欧美一区三区三区高中清蜜桃 | 亚洲欧美日韩国产成人精品影院| 欧美在线视屏 | 在线成人激情黄色| 亚洲视频日本| 免费看的黄色欧美网站| 一区二区久久| 久久精品亚洲精品国产欧美kt∨| 久久免费99精品久久久久久| 99热免费精品| 激情视频亚洲| 国产欧美一区二区精品性色| 国产精品国产自产拍高清av| 免费成人高清在线视频| 国产精品久久久久久av福利软件| 欧美国产日韩一区二区三区| 美女精品在线观看| 欧美视频你懂的| 国产性色一区二区| 亚洲精品乱码久久久久| 一本一道久久综合狠狠老精东影业 | 蜜臀a∨国产成人精品| 日韩五码在线| 久久综合电影一区| 国产亚洲欧美激情| 午夜在线电影亚洲一区| 最新日韩在线视频| 亚洲欧美精品在线观看| 欧美三级视频| 久久色在线播放| 国产精品羞羞答答xxdd| 宅男噜噜噜66一区二区66| 亚洲国产经典视频| 久久精品国产免费| 国产在线视频欧美| 亚洲欧美在线磁力| 亚洲综合国产精品| 国产欧美日韩综合精品二区| 在线视频你懂得一区二区三区| 欧美韩国日本综合| 裸体歌舞表演一区二区| 亚洲国产精品第一区二区三区| 久久一区二区三区四区| 久久精品国产2020观看福利| 在线成人国产| 91久久国产精品91久久性色| 久久免费高清| 久久尤物电影视频在线观看| 极品中文字幕一区| 欧美激情国产高清| 欧美精品久久一区| 亚洲欧美日韩国产| 午夜精品久久久久久久久久久 | 欧美国产日韩a欧美在线观看| 欧美一区二区视频在线观看| 国产一二三精品| 欧美aⅴ99久久黑人专区| 久久精品国产免费观看| 激情文学综合丁香| 亚洲国产裸拍裸体视频在线观看乱了中文| 奶水喷射视频一区| 亚洲一区二区高清视频| 亚洲欧美视频一区二区三区| 国模大胆一区二区三区| 欧美大片免费观看在线观看网站推荐| 欧美mv日韩mv国产网站| 亚洲午夜精品17c| 午夜精品一区二区三区在线| 曰本成人黄色| 夜夜嗨av一区二区三区网页| 国产亚洲综合在线| 欧美激情第三页| 国产精品色在线| 欧美+日本+国产+在线a∨观看| 欧美日韩 国产精品| 久久成人18免费观看| 母乳一区在线观看| 久久高清免费观看| 亚洲国产综合在线看不卡| 欧美另类亚洲| 久久国产精品久久久| 欧美91大片| 久久成人18免费观看| 欧美理论在线播放| 久久精品一区| 欧美丝袜第一区| 免费一级欧美片在线播放| 国产精品美女午夜av| 亚洲福利视频三区| 国产日韩欧美一二三区| 亚洲精一区二区三区| 在线精品观看| 午夜精品久久久久久| 亚洲最新中文字幕| 麻豆精品国产91久久久久久| 久久av一区二区| 欧美午夜片欧美片在线观看| 亚洲国产视频直播| 今天的高清视频免费播放成人| 亚洲欧美激情四射在线日| 99视频+国产日韩欧美| 久久午夜影视| 久久久久久午夜| 国产精品日韩欧美| 日韩一区二区精品葵司在线| 亚洲日本电影在线| 蜜臀av性久久久久蜜臀aⅴ四虎 | 性欧美1819性猛交| 亚洲一区二区三区高清不卡| 欧美精品激情在线观看| 欧美激情精品久久久六区热门| 国产真实乱子伦精品视频| 亚洲欧美日韩另类| 午夜国产精品视频免费体验区| 篠田优中文在线播放第一区| 亚洲——在线| 欧美揉bbbbb揉bbbbb| 亚洲另类春色国产| 亚洲少妇一区| 欧美视频一区二区| 一区二区三区你懂的| 亚洲一区二区三区午夜| 国产精品国产三级国产aⅴ浪潮| 一区二区三区免费网站| 亚洲欧美一级二级三级| 国产精品自拍在线| 欧美一级淫片播放口| 久久手机精品视频| 亚洲国产精品小视频| 欧美国产日韩一区二区在线观看| 欧美激情一区二区三区蜜桃视频| 亚洲国产欧美另类丝袜| 欧美3dxxxxhd| 一本到高清视频免费精品| 欧美理论电影在线播放| 一本一本久久| 午夜精品久久久久久久久久久| 国产精品日韩在线观看| 欧美亚洲网站| 久久伊人亚洲| 欧美午夜无遮挡| 午夜亚洲视频| 欧美高清你懂得| 亚洲天堂网站在线观看视频| 国产日本欧美一区二区三区| 米奇777在线欧美播放| 一本大道久久a久久精二百| 久久久久久日产精品| 99精品国产一区二区青青牛奶| 国产精品视频观看| 另类激情亚洲| 亚洲欧美激情诱惑| 91久久久亚洲精品| 久久久国产一区二区| 亚洲免费观看在线观看| 国产欧美综合在线| 欧美激情综合五月色丁香| 亚洲一区二区三区四区五区黄 | 亚洲欧洲偷拍精品| 欧美一区二区三区在线免费观看 | 一区二区三区久久久| 久久都是精品| 99精品久久| 韩国成人理伦片免费播放| 欧美日韩日本网| 久久视频在线看| 亚洲一区二区三区国产| 亚洲国产精品va在线观看黑人| 午夜精品在线视频| 亚洲啪啪91| 美腿丝袜亚洲色图| 欧美在线一区二区| av不卡免费看| 亚洲国产va精品久久久不卡综合|