一篇文章帶你了解論C語言中算法的重要性
一、問題一(打印階乘)
問題描述:
打印出數(shù)字一到數(shù)字20的階乘
一開始,我總會多打印出一個1,這令我十分苦惱,并且從n等于13開始,數(shù)據(jù)就開始溢出

問題分析:
讓我們分析一下問題,這里面存在著兩個問題:
1.多打印出一個1
2.數(shù)據(jù)溢出
解決方案:
1.讓我們檢查一下結(jié)果,發(fā)現(xiàn)問題很有可能是循環(huán)的時候沒有循環(huán)本身
for (i = 1; i < num; i++)//這句話明顯錯了
改成
for (i = 1; i <= num; i++) {//i的值要乘以它本身!
n = n * i;
}
2.這里要引入C++中STL庫的一個知識點
常規(guī)的32位整數(shù)只能夠處理40億以下的數(shù)。
如果遇到比40億要大的數(shù),就要用到C++的64位擴展。不同的編譯器對64位整數(shù)的擴展有所不同。這個我也是聽別人科普的,大家可以站內(nèi)搜索一下。
優(yōu)化后的代碼如下:
#include <stdio.h>
void main() {
__int64 fac(int num);
int n = 1;
int num;
for (num = 0; num <= 20; ++num) {
printf("%3d! = %I64d\n", num, fac(num));
}
}
__int64 fac(int num) {
register __int64 n = 1, i; //寄存器變量
for (i = 1; i <= num; i++) {//i的值要乘以它本身!
n = n * i;
}
return n;
}

二、問題二(比較多項式計算時間)
問題描述:

這里先科普幾個測試代碼中的知識點:
這個表示本程序開始計時:
start = clock();
本程序結(jié)束計時:
stop = clock();
clock tick :時鐘打點
CLK_TCK:機器時鐘每秒所走的時鐘打點數(shù)
問題分析:
首先這個問題有兩種算法:
直接算
p1 += (pow(x, i)/i);
把x當成公因式提出計算(秦九韶法)
p2 = 1.0/(a[i - 1])+ (x*p2);
然后我發(fā)現(xiàn)了三個問題:
1.測量不出時間
2.程序重復性高
3.第一種結(jié)果和第二種結(jié)果不一致
解決方案:
1.讓被測函數(shù)重復運行多次,使得測出的總時鐘打點間隔充分長,最后計算被測函數(shù)除以運行次數(shù)即可得出平均每次的運行時間
duration = ((double)(stop - start)) / CLK_TCK / MAXK;
2.可以通過多設(shè)置幾個函數(shù),并調(diào)用函數(shù)解決問題
3.這是算法的問題
這個地方真的特別容易出錯,我改了不知道多少遍。。。。。。
double f2(int n, double a[], double x) {
int i;
double p2 = 1.0/a[n];
for (i = n; i > 0; i--) {
p2 = 1.0/(a[i - 1])+ (x*p2);//算法思路出毛病了(數(shù)學)
}
return p2;
}
總體的代碼:
#include <stdio.h>
#include <math.h>
#include <time.h>
clock_t start, stop;
double duration;
#define MAXN 101//數(shù)組里元素個數(shù)(多項式的系數(shù)),如果看n值需要減一,因為有a0
#define MAXK 1000//重復調(diào)用的次數(shù)
double f1(int n, double a[], double x)
{
double p1 = a[0];//a[0]都已經(jīng)算出來了,循環(huán)從1開始
for (int i = 1; i <= n; i++) {
p1 += (pow(x, i)/i);
}
return p1;
}
double f2(int n, double a[], double x) {
int i;
double p2 = 1.0/a[n];
for (i = n; i > 0; i--) {
p2 = 1.0/(a[i - 1])+ (x*p2);//算法思路出毛病了(數(shù)學)
}
return p2;
}
double ceshijian()
{
stop = clock();//停止計時
duration = ((double)(stop - start)) / CLK_TCK / MAXK;//計算單次運行時間
printf("ticks=%f\n", (double)(stop - start));
printf("duration=%6.2e\n", duration);
return 0;
}
int main()
{
int i;
double a[MAXN];
for (i = 0; i < MAXN; i++) {
a[i] = (double)i;
}//輸入的早就是i值了
a[0] = 1;
//不在測試范圍內(nèi)的準備工作寫在clock()調(diào)用之前
start = clock();//開始計時
for (int i = 0; i < MAXK; i++)//重復調(diào)用
f1(MAXN - 1, a, 1.1);//被測函數(shù),這里如果寫數(shù)組的話就越界了,而且要調(diào)用某個值是不確定的,只能寫a,因為要定義的就是a值
printf("第一種結(jié)果為%f\n", f1(MAXN - 1, a, 1.1));
ceshijian();
start = clock();//開始計時
for (i = 0; i < MAXK; i++)
f2(MAXN - 1, a, 1.1);//被測函數(shù),這里如果寫數(shù)組的話就越界了,而且要調(diào)用某個值是不確定的
printf("第二種結(jié)果為%f\n", f2(MAXN - 1, a, 1.1));
ceshijian();
return 0;
}
結(jié)果如下

總結(jié)
本篇文章就到這里了,希望能給你帶來幫助,也希望您能夠多多關(guān)注腳本之家的更多內(nèi)容!
相關(guān)文章
簡單分析C語言中指針數(shù)組與數(shù)組指針的區(qū)別
這篇文章主要介紹了C語言中指針數(shù)組與數(shù)組指針的區(qū)別,是C語言入門學習中的基礎(chǔ)知識,需要的朋友可以參考下2015-11-11
vs2019永久配置opencv開發(fā)環(huán)境的方法步驟
這篇文章主要介紹了vs2019永久配置opencv開發(fā)環(huán)境的方法步驟,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2020-03-03

