Code Kata:大整數比較大小&大整數四則運算---加減法 javascript實現

来源:http://www.cnblogs.com/jinyuGu/archive/2017/12/08/8007224.html
-Advertisement-
Play Games

大整數的四則運算已經是老生常談的問題了。很多的庫也已經包含了各種各樣的解決方案。 作為練習,我們從最簡單的加減法開始。 加減法的核心思路是用倒序數組來模擬一個大數,然後將兩個大數的利用豎式進行運算。 加法函數: 異符號相加時調用減法函數(減法函數後面給出) 同符號相加先確定符號 因為輸入輸出的為字元 ...


大整數的四則運算已經是老生常談的問題了。很多的庫也已經包含了各種各樣的解決方案。

作為練習,我們從最簡單的加減法開始。

加減法的核心思路是用倒序數組來模擬一個大數,然後將兩個大數的利用豎式進行運算。

加法函數

  • 異符號相加時調用減法函數(減法函數後面給出)
  • 同符號相加先確定符號
  • 因為輸入輸出的為字元串,需要去除字元串開頭的0
 1 function add(a, b) { /*輸入兩個字元串類型大數字*/
 2 
 3     if(a.indexOf('-') >= 0 && b.indexOf('-') < 0){
 4 
 5         return minus(b,a);
 6     }
 7     else if(a.indexOf('-') < 0 && b.indexOf('-') >= 0){
 8 
 9         return minus(a,b);
10     }
11 
12     var sign = "";
13 
14     if(a.indexOf('-') >= 0 && b.indexOf('-') >= 0){ /*兩個負數相加,指定符號*/
15 
16         sign = "-";
17 
18         a = a.substr(1);
19 
20         b = b.substr(1);
21     }
22 
23     var aArr = a.replace(/^0+/,'').split('').reverse();
24 
25     var bArr = b.replace(/^0+/,'').split('').reverse(); /*利用倒序數組存儲*/
26 
27     var carry = 0; /*進位值*/
28 
29     var sumArr = [];
30 
31     var len = Math.max(aArr.length, bArr.length); /*取得位數較大的一個數的位數*/
32 
33     for(var i=0;i<=len-1;i++){
34 
35         var digA = parseInt(aArr[i]) ? parseInt(aArr[i]) : 0;
36 
37         var digB = parseInt(bArr[i]) ? parseInt(bArr[i]) : 0;
38 
39         var digTotal = digA + digB + carry;
40 
41         if(i == len-1){/*排除'012' + '012'這樣的情況*/
42 
43             if(digTotal > 0){
44 
45                 sumArr.unshift(digTotal);
46             }
47 
48             break;
49         }
50 
51         carry = Number(digTotal >= 10);
52 
53         digTotal = digTotal % 10;
54 
55         sumArr.unshift(digTotal);
56 
57     }
58 
59     return sign + sumArr.join('');
60 }

 

在寫減法時,發現需要先比較大小,因此需要一個大數字比較大小的函數

比較小大函數:

  • 異符號比較大小,正數大於負數
  • 正數比較大小,先比較長度,長度大的數值大
  • 正數長度一致,從最高位開始逐位比較,只到出現較大的一方,則數值更大
  • 負數比較大小,方法同正數,結果取反即可
  • 因為輸入輸出的為字元串,需要去除字元串開頭的0
 1 function compare(a,b){
 2 
 3     var sign = 1;
 4 
 5     if(a.indexOf('-') >= 0 && b.indexOf('-') < 0){ /*異符號比較*/
 6 
 7         return -1;
 8     }
 9     else if(a.indexOf('-') < 0 && b.indexOf('-') >= 0){ /*異符號比較*/
10 
11         return 1;
12     }
13     else if(a.indexOf('-') >= 0 && b.indexOf('-') >= 0){ /*同為負數,指定取反,同時改為正數比較方式*/
14 
15         sign = -1;
16 
17         a = a.substr(1);
18 
19         b = b.substr(1);
20     }
21 
22     a = a.replace(/^0+/,'');
23 
24     b = b.replace(/^0+/,'');
25 
26     var flag;
27 
28     if(a.length < b.length){    /*比較長度*/
29 
30         flag = -1;
31     }
32     else if(a.length > b.length){
33 
34         flag = 1;
35     }
36     else{
37 
38         flag = 0;
39     }
40 
41     if(flag == 0){ /*相同長度逐位比較*/
42 
43         var aArr = a.split('');
44 
45         var bArr = b.split('');
46 
47         for(var i=0;i<=aArr.length;i++){
48 
49             if(aArr[i] > bArr[i]){
50 
51                 flag = 1;
52 
53                 break;
54             }
55             else if(aArr[i] > bArr[i]){
56 
57                 flag = -1;
58 
59                 break;
60             }
61         }
62     }
63 
64     return sign * flag;
65 }

 

減法函數:

  • 異符號相減時調用加法函數
  • 同符號相減需要先確定大小
  • 因為輸入輸出的為字元串,需要去除字元串開頭的0
 1 function minus(a, b) {
 2 
 3     if(a.indexOf('-') >= 0 && b.indexOf('-') < 0){
 4 
 5         return add(a,"-" + b);
 6     }
 7     else if(a.indexOf('-') < 0 && b.indexOf('-') >= 0){
 8 
 9         a = a.substr(1);
10 
11         return add(a,b);
12     }
13 
14     var sign = "";
15 
16     if(compare(a,b) < 0){
17 
18         var temp = b;
19 
20         b = a;
21 
22         a = temp;
23 
24         sign = "-";
25     }
26 
27     var aArr = a.replace(/^0+/,'').split('').reverse();
28 
29     var bArr = b.replace(/^0+/,'').split('').reverse(); /*利用倒序數組存儲*/
30 
31     var borrow = 0; /*借位值*/
32 
33     var minusArr = [];
34 
35     var len = Math.max(aArr.length, bArr.length); /*取得位數較大的一個數的位數*/
36 
37     for(var i=0;i<=len-1;i++){
38 
39         var digA = parseInt(aArr[i]) ? parseInt(aArr[i]) : 0;
40 
41         var digB = parseInt(bArr[i]) ? parseInt(bArr[i]) : 0;
42 
43         var digMinus;
44 
45         if(i == len-1){
46 
47             if(digA - borrow <= digB){ /*最高位不夠減直接跳出迴圈*/
48 
49                 break;
50             }
51         }
52 
53         if(digA - digB - borrow >= 0){
54 
55             digMinus = digA - digB - borrow;
56 
57         }else{
58 
59             digMinus = digA + 10 - digB - borrow;
60 
61             borrow = 1;
62         }
63 
64         minusArr.unshift(digMinus);
65 
66     }
67 
68     return sign + minusArr.join('');
69 }

以上給出的是帶符號大整數加減法基礎實現,但效率並不是特別高。

網上也有通過10000進位優化的豎式演算法,以及通過位運算實現四則運算的方法,大家也可以搜索看看,今天的練習就到這裡了,下周會給出乘除法的基本實現。

 


 

如果喜歡我的文章,可以掃描二維碼關註我的微信公眾號

爭取每天都分享一點我自己的開發和練習體驗~
qrcode_for_gh_a80ae0bf035f_344


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 前言 網路早期最大的問題之一是如何管理狀態。簡而言之,伺服器無法知道兩個請求是否來自同一個瀏覽器。當時最簡單的方法是在請求時,在頁面中插入一些參數,併在下一個請求中傳回參數。這需要使用包含參數的隱藏的表單,或者作為URL參數的一部分傳遞。這兩個解決方案都手動操作,容易出錯。 網景公司當時一名員工Lo ...
  • 註:本文轉載自大神同學的博客,http://www.cnblogs.com/st-leslie/p/5617130.html ,僅供學習不用於其他用途,大家想要更多的乾貨,請移步到該大神博客園 一、什麼是localStorage、sessionStorage 在HTML5中,新加入了一個localS ...
  • 一、介紹 本次博客主要介紹函數表達式的內容,主要是閉包。 二、函數表達式 定義函數的兩種方式:一個是函數聲明,另一個就是函數表達式。 區別: 1.函數聲明是用function後面有函數名,函數表達式是賦值形式給一個變數。 2.函數聲明可以提升函數,而函數表達式不會提升 函數提升就是函數會被自動提升到 ...
  • //多個值同時變化function getStyle(obj, name)//函數幫助獲取不在行間樣式,不受非行間border,padding等得影響{ //style只獲取行間樣式。offset受非行間border,padding等得影響 if(obj.currentStyle) { return ...
  • 1.JS事件的基本知識 JS事件的學習和JS方法的學習揉雜在一起,JS相對於Java等語言,方法定義和使用上比較隨意和簡單,但是還是有一些區別,需要理清楚. 2.jQuery方式綁定事件 這裡多多贅述一點,由於jQuery可以理解為是對JS的一種高級封裝,這種封裝是單向的,所以我們可以在JS中加入j ...
  • 地圖與地理定位 定位在大部分項目中都需要實現,如何實現主要有如下的幾種方法 1. H5定位 在HTML5中navigator有很強大的功能,其中就有定位的方法 這個服務其實是谷歌提供的,在我們國內使用的可能性較低 2. 後端定位 前端調用一個後端提供的介面,後端進行定位操作,返回給前端 在工作中公司 ...
  • 基礎數據結構與演算法 現在有兩個不同的JSON,比較複雜,可以參考這裡的DEMO中返回的JSON。要比較它們的差異,除了用現成的工具如beyond compare以外,如果我們的機器上沒有安裝這個工具,能如何較快解決?作為一個程式員,一個個對比是不可行的,對比完也不會有什麼收穫。我會把之放進Excel ...
  • 跨域問題的產生: 因為瀏覽器有同源策略,只有在同功能變數名稱,同埠,同協議的情況下才可以進行數據交互;有的時候,例如:在公司開發項目的時候,前端開發的伺服器可能和後端伺服器不是同一個,因為可能是通過gulp、webpack搭建的開發伺服器,就需要解決跨域問題,再例如,在大公司數據伺服器不只有一個,所以跨域 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...