C語(yǔ)言求質(zhì)數(shù)的幾種簡(jiǎn)單易懂方式
質(zhì)數(shù)就是除了1和它本身外沒有其他因數(shù)
一. 暴力枚舉
假設(shè)現(xiàn)在有一個(gè)數(shù)num,要求我們判斷是否是質(zhì)數(shù),由定義知我們可以遍歷從2到 num-1的所有數(shù),假
設(shè)都不能被整除,則num是質(zhì)數(shù),否則不是,C語(yǔ)言代碼實(shí)現(xiàn)如下。
其中track用來檢測(cè)是否遍歷完從2到num-1的所有數(shù)
int main() { int n = 0; int track = 0; printf("請(qǐng)輸入要判斷的數(shù): "); scanf("%d", &n); for (int i = 2; i < n; i++) { if (n % i == 0) { track = 1; break; } } if (track == 1) printf("這個(gè)數(shù)不是質(zhì)數(shù)\n"); else printf("這個(gè)數(shù)是質(zhì)數(shù)\n"); return 0; }
二. 暴力求解的優(yōu)化版本
實(shí)際上我們只需要遍歷從2到√num的數(shù)就可以了。
因?yàn)閷?duì)于非質(zhì)數(shù)num來說有 a * b = num;其中a和b不可能同時(shí)大于√num,也就是說num是非質(zhì)數(shù)的充要條件是在 [2,num-1]的區(qū)間上有因數(shù),根據(jù)這一點(diǎn)可以對(duì)代碼進(jìn)行優(yōu)化。
int main() { int n = 0; int track = 0; printf("請(qǐng)輸入要判斷的數(shù): "); scanf("%d", &n); for (int i = 2; i <= sqrt(n); i++) { if (n % i == 0) { track = 1; break; } } if (track == 1) printf("這個(gè)數(shù)不是質(zhì)數(shù)\n"); else printf("這個(gè)數(shù)是質(zhì)數(shù)\n"); return 0; }
三.埃拉托斯特尼篩法
如果要求我們判斷的數(shù)字很多,那么上面兩種方法的效率就極其低下,因?yàn)槊颗袛嘁粋€(gè)數(shù)都要從2開始遍歷,計(jì)算機(jī)會(huì)做很多重復(fù)操作。
換一種思路,我們可以選一批質(zhì)數(shù),質(zhì)數(shù)的倍數(shù)是合數(shù)(非質(zhì)數(shù)),那么就可以把這些合數(shù)篩掉,經(jīng)過多輪篩選后剩下的就全部是質(zhì)數(shù)了。
看了前面的敘述可能有的朋友會(huì)有點(diǎn)懵,我解釋一下。
細(xì)節(jié)部分
1. 怎樣選一批素?cái)?shù)能將區(qū)間內(nèi)所有合數(shù)都篩完?
從2開始到√n的所有素?cái)?shù)。首先1肯定沒有篩選功能(1的任意倍數(shù)都是其本身)。
對(duì)于√n之后的素?cái)?shù),比如說用√n + 1進(jìn)行篩選 ,得到的可篩選的數(shù)是 [(√n + 1) * 2, (√n +1) * √n] 中的整數(shù),但是這些整數(shù)都有一個(gè)小于等于√n的約數(shù),因此在我們遍歷前面的數(shù)時(shí)已經(jīng)將他們刪除掉了,所以沒必要重復(fù),只需要 [2,√n)的所有素?cái)?shù)即可。
2.篩選過程具體是怎樣的?
不清楚篩選過程的朋友可以看看這張圖,這張圖搬運(yùn) 自公眾號(hào) “coder梁”,做的特別清楚。
3.具體代碼
C語(yǔ)言實(shí)現(xiàn)。
int main() { //埃式篩法 int n = 0; printf("請(qǐng)輸入要判斷的數(shù) "); scanf("%d", &n); int* prime = (int*)malloc(n * sizeof(int));//定義一個(gè)可以存放n個(gè)數(shù)的數(shù)組 if (!prime) { printf("創(chuàng)建數(shù)組失敗\n"); exit(-1); } //將prime數(shù)組全部初始化成1,表示全部是素?cái)?shù) for (int i = 0; i < n; i++) { prime[i] = 1; } //從2開始篩選 for (int i = 2; i <= sqrt(n); i++) { if (prime[i - 1]) //如果是質(zhì)數(shù) { for (int j = i * i; j <= n; j += i) //則將其剔除 prime[j - 1] = 0; } } //打印 for (int i = 0; i < n; i++) { if (prime[i] != 0) printf("%d ", i + 1); } printf("\n"); return 0; }
總結(jié)
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
Qt數(shù)據(jù)庫(kù)應(yīng)用之?dāng)?shù)據(jù)打印到pdf
因?yàn)閤ls打開以后用戶可以修改數(shù)據(jù)造假之類的,而pdf默認(rèn)是不可編輯的,除非借助專業(yè)的工具,所以如果想要限定用戶導(dǎo)出數(shù)據(jù)不能被更改,那導(dǎo)出pdf是最佳選擇。所以本文將為代價(jià)介紹Qt實(shí)現(xiàn)數(shù)據(jù)打印到pdf的方法,需要的可以參考一下2022-01-01C++詳解默認(rèn)參數(shù)的構(gòu)造函數(shù)及簡(jiǎn)單實(shí)例代碼
這篇文章主要介紹了 C++詳解默認(rèn)參數(shù)的構(gòu)造函數(shù)及簡(jiǎn)單實(shí)例代碼的相關(guān)資料,需要的朋友可以參考下2017-02-02C語(yǔ)言之函數(shù)遞歸的實(shí)現(xiàn)
本文主要介紹了C語(yǔ)言之函數(shù)遞歸的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-07-07淺談C++內(nèi)存分配及變長(zhǎng)數(shù)組的動(dòng)態(tài)分配
下面小編就為大家?guī)硪黄獪\談C++內(nèi)存分配及變長(zhǎng)數(shù)組的動(dòng)態(tài)分配。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2016-09-09C語(yǔ)言實(shí)現(xiàn)撲克牌計(jì)算24點(diǎn)
這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言如何實(shí)現(xiàn)撲克牌計(jì)算24點(diǎn),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-10-10c++之time_t和struct tm及時(shí)間戳的正確使用方式
C++中處理時(shí)間的常用數(shù)據(jù)類型有time_t和struct tm,time_t通常用來表示時(shí)間戳,即從1970年1月1日至今的秒數(shù),struct tm是一個(gè)結(jié)構(gòu)體,用來存儲(chǔ)年、月、日、時(shí)、分、秒等信息,時(shí)間戳可以通過gmtime()轉(zhuǎn)換為struct tm類型,反之亦然2024-10-10