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

(轉)質數算法大全,及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 小蟲蟲 閱讀(1054) 評論(0)  編輯 收藏 引用


只有注冊用戶登錄后才能發表評論。
網站導航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理


<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>
            欧美成人性网| 欧美成人xxx| 久久久综合香蕉尹人综合网| 99ri日韩精品视频| 亚洲激情在线播放| 亚洲国产影院| 最新精品在线| 亚洲作爱视频| 亚洲欧美日韩在线观看a三区| 亚洲一区二区三区午夜| 亚洲在线视频| 久久狠狠亚洲综合| 欧美大片18| 亚洲欧洲精品天堂一级| 欧美黄色视屏| 99精品热6080yy久久| 一区二区三区四区五区在线| 一本色道久久综合| 亚洲自拍高清| 裸体素人女欧美日韩| 欧美日韩不卡视频| 国产亚洲激情在线| 亚洲精品视频一区| 香蕉免费一区二区三区在线观看 | 国产精品v欧美精品∨日韩| 欧美日韩在线一区二区| 国产精品国码视频| 国精品一区二区| 亚洲高清二区| 亚洲一卡二卡三卡四卡五卡| 亚洲欧美成人网| 欧美一区二区三区在线观看| 老牛嫩草一区二区三区日本 | 欧美一区二区在线| 欧美激情视频给我| 一本一本久久a久久精品综合妖精| 999亚洲国产精| 久久国产66| 亚洲美女诱惑| 久久高清免费观看| 欧美成人蜜桃| 欧美成年人视频网站| 国产欧美一区二区在线观看| 亚洲国产成人精品久久| 亚洲性视频h| 国产偷国产偷亚洲高清97cao | 国产精品美女www爽爽爽| 国产亚洲精品v| 一区二区免费看| 亚洲福利国产| 老司机精品久久| 伊人久久综合97精品| 亚洲欧美中文另类| 亚洲视频在线观看三级| 国产精品久久久久久久久久ktv | 亚洲国产裸拍裸体视频在线观看乱了中文 | 亚洲影院免费观看| 美女999久久久精品视频| 国产婷婷色综合av蜜臀av| 亚洲欧美在线一区二区| 一个色综合av| 99精品国产高清一区二区| 欧美日韩福利在线观看| 雨宫琴音一区二区在线| 欧美亚洲一区二区三区| 国产亚洲激情在线| 久久久国产视频91| 欧美在线视频观看免费网站| 国产视频综合在线| 亚洲激情影院| 一本色道久久综合亚洲二区三区| 美女在线一区二区| 在线成人小视频| 日韩一级欧洲| 久久久久久尹人网香蕉| 亚洲国产精品精华液网站| 亚洲免费在线视频一区 二区| 久久在精品线影院精品国产| 午夜综合激情| 久久一区二区三区国产精品 | 久久综合成人精品亚洲另类欧美| 中日韩高清电影网| 亚洲女同同性videoxma| 免费一级欧美在线大片| 国产欧美日韩91| 久久精品国产久精国产一老狼 | 久久综合狠狠综合久久综合88 | 亚洲欧洲午夜| 亚洲精品网址在线观看| 久久久中精品2020中文| 99精品久久| 欧美大胆成人| 国产精品日韩欧美一区| 一本久道综合久久精品| 亚洲欧美一区二区视频| 欧美日韩视频专区在线播放 | 日韩视频三区| 午夜免费日韩视频| 久久综合九色欧美综合狠狠| 国产欧美亚洲视频| 亚洲欧洲日产国产网站| 久久久噜噜噜久久狠狠50岁| 亚洲另类一区二区| 91久久精品网| 校园春色国产精品| 日韩五码在线| 91久久精品日日躁夜夜躁国产| 国产欧美一区二区精品忘忧草| 亚洲一区二区三区精品动漫| 久久国产精品久久精品国产| 国内精品久久久久久久影视蜜臀 | 久久婷婷人人澡人人喊人人爽| 国产日韩成人精品| 欧美一区二区三区日韩视频| 亚洲一区二区在线看| 欧美日韩亚洲一区| 国语自产精品视频在线看一大j8| 日韩亚洲精品电影| 欧美日韩国内自拍| 欧美成年人网站| 欧美电影免费观看高清完整版| 午夜老司机精品| 久久不射电影网| 欧美成人精品高清在线播放| 久久亚裔精品欧美| 久久精品国产69国产精品亚洲| 亚洲欧美日韩国产综合在线 | 午夜精品福利在线观看| 亚洲视频福利| 国产婷婷色一区二区三区在线| 久久一区免费| 欧美大秀在线观看| 亚洲麻豆视频| 激情亚洲成人| 久久久久九九九九| 国产欧美日韩不卡| 美女诱惑黄网站一区| 99re热这里只有精品视频| 亚洲第一福利视频| 亚洲精品小视频在线观看| 国产精品美女久久久久aⅴ国产馆| 好吊妞**欧美| 欧美风情在线观看| 亚洲国产婷婷综合在线精品 | 免费国产一区二区| 免费h精品视频在线播放| 欧美风情在线观看| 亚洲激情在线| 国产日产欧美一区| 欧美高清视频在线| 99精品欧美一区二区三区| 亚洲一区二区三区在线视频| 亚洲天堂免费在线观看视频| 欧美在线免费观看| 欧美精品久久久久久久免费观看 | 亚洲国产日韩欧美在线99 | 亚洲网站在线播放| 久久精品国亚洲| 亚洲私人影院| 国产亚洲欧美一区二区三区| 亚洲精选大片| 久久国产主播| 欧美日韩国产bt| 午夜亚洲伦理| 亚洲高清久久| 久久精精品视频| 夜夜嗨av一区二区三区中文字幕| 亚洲欧美自拍偷拍| 99国产一区| 怡红院精品视频在线观看极品| 久久久精品一区二区三区| 欧美精品一区二区在线观看 | 久热精品在线视频| 亚洲精品久久视频| 国产精品亚洲综合天堂夜夜| 亚洲少妇自拍| 亚洲毛片在线看| 国产精品护士白丝一区av| 亚洲黄色成人久久久| 亚洲少妇一区| 91久久精品国产91久久性色| 国产精品高潮呻吟视频| 欧美在线亚洲在线| 久久超碰97人人做人人爱| 亚洲精品一级| 亚洲国产美国国产综合一区二区| 亚洲黄色影片| 亚洲综合色激情五月| 久久久精品国产一区二区三区| 久久国产高清| 99在线热播精品免费| 亚洲精品国产精品乱码不99| 国产精品久久久久久av下载红粉| 中文在线不卡| 亚洲电影免费观看高清完整版| 国产精品日韩一区二区三区| 亚洲一区日韩| 一区二区日韩伦理片| 性高湖久久久久久久久| 亚洲桃色在线一区|