• <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>
            posts - 297,  comments - 15,  trackbacks - 0
            在計算機中,長整型(long int)變量的范圍是 -2147483648 至 2147483647,因此若用長整型變量做乘法運算,乘積最多不能超過 10位數。即便用雙精度型(double)變量,也僅能保證 16 位有效數字的精度。在某些需要更高精度的乘法運算的場合,需要用別的辦法來實現乘法運算。 比較容易想到的是做多位數乘法時列豎式進行計算的方法,只要寫出模擬這一過程的程序,就能實現任意大整數的乘法運算。經過查閱資料,找到一種更易于編程的方法,即“列表法”。 下面先介紹“列表法”: 例如當計算8765 x 234時,把乘數與被乘數照如下列出,見表1: 8 7 6 5 8765 2 16 14 12 10 2 3 24 21 18 15 3 4 32 28 24 20 4 表 1 表 2 把每個空格所在的行列的數字的乘積填入空格中,得表2。把表2中的數按圖示斜線分組(橫縱坐標和相等的數分為一組),把每組數的累加起來所得的和記在表格下方,見表 3: 16 14 12 10 24 21 18 15 32 28 24 20 16 38 65 56 39 20 表 3 最后把表格下方一行數作如下處理,見表4: 從最低位的 20 開始,保留個位數字“0”,把個位以外的數“2”進到前一位;把次低位的 39 加上低位進上來的 2 得 41,保留個位數字“1”,把“4”進到前一位;以此類推,直至最高位的 16,16 加上低位進上來的4得 20,保留“0”,把2進到最高位,得乘積答數 2051010。 16 38 65 56 39 20 2 16+4=20 38+7=45 65+6=71 56+4=60 39+2=41 留0進2 留5進4 留1進7 留0進6 留1進4 留0進2 2 0 5 1 0 1 0 表 4 根據以上思路就可以編寫C 程序了,再經分析可得: 1、一個m 位的整數與一個 n 位的整數相乘,乘積為m+n-1 位或m+n 位。 2、程序中,用三個字符數組分別存儲乘數、被乘數與乘積。由第 1 點分析知,存放乘積的字符數組 的長度應不小于存放乘數與被乘數的兩個數組的長度之和。 3、可以把第二步“計算填表”與第三四步“累加進位”放在一起完成,可以節省存儲表格 2所需的空間。 4、程序關鍵部分是兩層循環,內層循環累計一組數的和,外層循環處理保留的數字與進位。 編寫的程序如下: 程序 1 清單: /* multiply.c */ /* 11/20/2008 */ #define MAXLENGTH 1000 #include #include void compute(char *a, char *b, char *c); void main(void) { char a[MAXLENGTH], b[MAXLENGTH], c[MAXLENGTH * 2]; puts("Input multiplier :"); gets(a); puts("Input multiplicand :"); gets(b); compute(a, b, c); puts("Answer :"); puts(c); getchar(); } void compute(char *a, char *b, char *c) { int i, j, m, n; long sum, carry; m = strlen(a) - 1; n = strlen(b) - 1; for (i = m; i >= 0; i--) a[i] -= '0'; for (i = n; i >= 0; i--) b[i] -= '0'; c[m + n + 2] = '\0'; carry = 0; for (i = m + n; i >= 0; i--) /* i 為坐標和 */ { sum = carry; if ((j = i - m) < 0) j = 0; for ( ; j<=i && j<=n; j++) /* j 為縱坐標 */ sum += a[i-j] * b[j]; /* 累計一組數的和 */ c[i + 1] = sum % 10 + '0'; /* 算出保留的數字 */ carry = sum / 10; /* 算出進位 */ } if ((c[0] = carry+'0') == '0') /* if no carry, */ c[0] = '\040'; /* c[0] equals to space */ } 效率分析:用以上算法計算 m位整數乘以n 位整數,需要先進行 m x n次乘法運算,再進行約 m + n次加法運算和 m + n次取模運算(實為整數除法)。把這個程序稍加修改,讓它自己產生乘數與被乘 數,然后計算隨機的 7200位整數互乘,在Cyrix 6x86 pr166機器的純DOS方式下耗時 7秒(用Borland C 3.1編譯)。 經過改進,此算法效率可以提高約9 倍。 注意到以下事實:8216547 x 96785 將兩數從個位起,每 3位分為節,列出乘法表,將斜線間的 數字相加; 8 216 547 96 785 768 20736 52512 6280 169560 429395 768 27016 222072 429395 將表中最后一行進行如下處理:從個位數開始,每一個方格里只保留三位數字,超出 1000 的部 分進位到前一個方格里; 768 27016 222072 429395 768+27 27016+222 222072+429 =795 =27238 =222501 395 795 238 501 395 所以8216547 x 96785 = 795238501395 也就是說我們在計算生成這個二維表時,不必一位一位地乘,而可以三位三位地乘;在累加時也是滿1000進位。這樣,我們在計算 m位整數乘以 n位整數,只需要進行 m x n / 9次乘法運算,再進行約(m + n) / 3次加法運算和(m + n) /3 次取模運算。總體看來,效率約是前一種算法的 9倍。 有人可能會想:既然能夠三位三位地乘,為什么不4位 4位甚至5位5位地乘呢?那不是可以提高 16 乃至 25 倍效率嗎?聽我解來:本算法在累加表中斜線間的數字時,如果用無符號長整數(范圍 0至~4294967295)作為累加變量,在最不利的情況下(兩個乘數的所有數字均是 9),能夠累加約4294967295/(999*999)=4300 次,也就是能夠準確計算任意兩個均不超過 12900(每次累加的結果"值"三位,故 4300*3=12900)位的整數相乘。如果 4 位 4 位地乘,在最不利的情況下,能夠累加約4294967295/(9999*9999)=43 次,僅能夠確保任意兩個不超過 172 位的整數相乘,沒有什么實用價值,更不要說5位了。 請看改進后的算法的實例程序: 該程序隨機產生兩個72xx位的整數,把乘數與積保存在 result.txt中。 在Borland C++ 3.1 中用 BCC -3 -O2 -G -mh -Z -f287 -pr -T- dashu.cpp 編譯生成的exe文件在Cyrix 6x86 pr166的機器上運行耗時0.82 秒。 程序 2 清單: #include #include #include #include #include #define N 7200 //作 72xx 位的整數乘法 int max(int,int,int); int initarray(int a[]); void write(int a[],int l); FILE *fp; void main() { int a[5000]={0},b[5000]={0},k[10001]={0}; //聲明存放乘數、被乘數與積的數組 clock_t start, end; //聲明用于計時的變量 unsigned long c,d,e; //聲明作累加用的無符號長整數變量 int i,j,la,lb,ma,mi,p,q,t; //聲明其它變量 randomize(); //初始化隨機數 la=initarray(a); //產生被乘數,并返回其長度 lb=initarray(b); //產生乘數,并返回其長度 if(lala)?lb:la; for (q=0;q=0;i--) //累加斜線間的數,i 為橫縱坐標之和 { c=d; //將前一位的進位標志存入累加變量 c ma=max(0,i-la+1,i-lb+1); //求累加的下限 mi=(i>la-1)?(la-1):i; //求累加的上限 for(j=ma;j<=mi;j++) //計算出橫縱坐標之和為 i 的單元內的數,并累加到 C 中 c+=(long)a[j]*b[i-j]; d=c/1000; //求進位標志 if(c>999) c%=1000; //取 c 的末三位 k[i]=c; //保存至表示乘積的數組 k[] } e=k[0]+1000*d; //求出乘積的最高位 end = clock();//停止計時 fp = fopen("result.txt", "w+"); //保存結果到 result.txt printf("\nThe elapsed time was: %3.4f\n", (end - start) / CLK_TCK); //打印消耗的時間 fprintf(fp,"%d",a[0]); //打印被乘數最高位 write(a,la); //打印被乘數其他位 fprintf(fp,"%d",b[0]); //打印乘數最高位 write(b,lb); //打印乘數其他位 fprintf(fp,"%ld",e); //打印乘積最高位 write(k,la+lb-1); //打印乘積其他位 fclose(fp); } max(int a,int b,int c) { int d; d=(a>b)?a:b; return (d>c)?d:c; } int initarray(int a[]) { int q,p,i; q=N+random(100); if(q%3==0) p=q/3; else p=q/3+1; for(i=0;iposted on 2009-12-06 23:26 chatler 閱讀(1142) 評論(0)  編輯 收藏 引用 所屬分類: Algorithm
            <2009年5月>
            262728293012
            3456789
            10111213141516
            17181920212223
            24252627282930
            31123456

            常用鏈接

            留言簿(10)

            隨筆分類(307)

            隨筆檔案(297)

            algorithm

            Books_Free_Online

            C++

            database

            Linux

            Linux shell

            linux socket

            misce

            • cloudward
            • 感覺這個博客還是不錯,雖然做的東西和我不大相關,覺得看看還是有好處的

            network

            OSS

            • Google Android
            • Android is a software stack for mobile devices that includes an operating system, middleware and key applications. This early look at the Android SDK provides the tools and APIs necessary to begin developing applications on the Android platform using the Java programming language.
            • os161 file list

            overall

            搜索

            •  

            最新評論

            閱讀排行榜

            評論排行榜

            亚洲午夜久久久久久噜噜噜| 精品久久无码中文字幕| 国产精品综合久久第一页| 国产精品免费久久| 偷窥少妇久久久久久久久| 久久精品中文无码资源站| 国产精品久久久99| 777午夜精品久久av蜜臀| 久久国产精品成人免费| 久久午夜福利电影| 99久久精品费精品国产一区二区| 99久久www免费人成精品| 亚洲中文字幕无码一久久区| 久久美女人爽女人爽| 久久精品国产亚洲αv忘忧草| 成人午夜精品久久久久久久小说| 国产69精品久久久久APP下载| 久久久久久免费一区二区三区 | 伊人久久综在合线亚洲2019| 久久久久一本毛久久久| 九九精品99久久久香蕉| 久久99国产精品久久99小说| 精品久久久久久无码中文字幕 | 精品亚洲综合久久中文字幕| 久久精品国产亚洲AV蜜臀色欲| 国产精品无码久久四虎| 成人综合伊人五月婷久久| 日日躁夜夜躁狠狠久久AV| 久久综合狠狠综合久久97色| 中文字幕亚洲综合久久2| 国产精品福利一区二区久久| 色诱久久久久综合网ywww| 亚洲伊人久久成综合人影院| 久久久久国产一级毛片高清板| 996久久国产精品线观看| 亚洲va久久久噜噜噜久久| 国产成人精品综合久久久久| 日本久久中文字幕| 亚洲午夜久久久| 亚洲精品无码久久久久久| 亚洲精品无码成人片久久|