最大余額法(計(jì)算百分比)

假設(shè)有數(shù)組 array = [7, 18, 23, 17]
要求計(jì)算每個(gè)數(shù)字占總和的百分比

普通計(jì)算方式

  1. 計(jì)算總和 7 + 18 + 23 + 17 = 65

  2. 每個(gè)數(shù)依次除以總和并乘以100
    7 / 65 * 100 = 10.76923076923077
    18 / 65 * 100 = 27.692307692307693
    23 / 65 * 100 = 35.38461538461539
    17 / 65 * 100 = 26.153846153846157

  3. 四舍五入保留兩位小數(shù)
    10.76923076923077 => 10.77
    27.692307692307693 => 27.69
    35.38461538461539 => 35.38
    26.153846153846157 => 26.15

最終結(jié)果 10.77 + 27.69 + 35.38 + 26.15 = 99.99 < 100

完整代碼
let array = [7, 18, 23, 17]
let sum = array.reduce((total, value) => total + value)
let result = array.map(val => ((val / sum) *100).toFixed(2)) 

為了解決上面的問(wèn)題搁胆,于是就有了最大余額法蔑赘,至于什么是最大余額法甸饱,百度上有续滋,這里只講過(guò)程

最大余額法

  1. 計(jì)算總和 7 + 18 + 23 + 17 = 65

  2. 計(jì)算總份額
    由于計(jì)算的是百分比阐斜,所以需要擴(kuò)大100倍
    百分比保留兩位小數(shù)名惩,所以再擴(kuò)大100倍(三位小數(shù)就是1000倍)
    所以總份額為 100 * 100 = 10000

  3. 按比例分配份額
    7 / 65 * 10000 = 1076.923076923077
    18 / 65 * 10000 = 2769.2307692307693
    23 / 65 * 10000 = 3538.461538461539
    17 / 65 * 10000 = 2615.3846153846157

  4. 取分配份額的整數(shù)部分 [1076, 2769, 3538, 2615]
    1076 + 2769 + 3538 + 2615 = 9998 < 10000
    10000 - 9998 = 2 因此還剩兩份需要分配

  5. 取分配份額的小數(shù)部分 [0.923076923077, 0.2307692307693, 0.461538461539, 0.3846153846157]
    找出小數(shù)部分最大的兩個(gè)數(shù)(剩幾份找?guī)讉€(gè))蛮拔,下標(biāo)分別為0 和 2

  6. 為整數(shù)部分下標(biāo)為0 和 2的數(shù)各加上1
    [1076 + 1, 2769, 3538 + 1, 2615] => [1077, 2769, 3539, 2615]

  7. 除以100得到最終的百分比
    因?yàn)楸A?位小數(shù)碟摆,之前乘以了100晃财,所以最后要除以100
    [1077, 2769, 3539, 2615] / 100 => [10.77, 27.69, 35.39, 26.15]

最終結(jié)果 10.77 + 27.69 + 35.39 + 26.15 = 100

完整代碼
const getPercentValue = (array, precision = 2) => {
  // 如果不是數(shù)字則賦值為0
  let arrayList = array.map(value => (isNaN(value) ? 0 : value))
  // 計(jì)算總和
  let sum = arrayList.reduce((total, value) => total + value)
  // 0不能做除數(shù),直接返回
  if (sum === 0) return
  // 小數(shù)位擴(kuò)大倍數(shù)
  let digits = Math.pow(10, precision)
  // 總份額
  let total = digits * 100
  // 分配份額
  let shareList = arrayList.map(val => (val / sum) * total)
  // 取整數(shù)部分
  let integerList = shareList.map(val => Math.trunc(val))
  // 計(jì)算整數(shù)部分的總和
  let shareSum = integerList.reduce((total, val) => total + val)
  // 取小數(shù)部分
  let decimalsList = shareList.map((value, index) => value - integerList[index])
  // 整數(shù)部分總和小于總份額時(shí)
  while (shareSum < total) {
    let max = decimalsList[0]
    let maxIndex = 0
    // 找出小數(shù)位最大的下標(biāo)
    for (let i = 1; i < decimalsList.length; i++) {
      if (decimalsList[i] > max) {
        max = decimalsList[i]
        maxIndex = i
      }
    }
    // 小數(shù)位最大的加1
    integerList[maxIndex] += 1
    // 加1后小數(shù)清零典蜕,不參與下次比較
    decimalsList[maxIndex] = 0
    // 總數(shù)也需要加1
    shareSum += 1
  }
  // 返回每項(xiàng)占比
  return integerList.map(value => (value / digits).toFixed(precision))
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末断盛,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子愉舔,更是在濱河造成了極大的恐慌钢猛,老刑警劉巖,帶你破解...
    沈念sama閱讀 217,509評(píng)論 6 504
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件轩缤,死亡現(xiàn)場(chǎng)離奇詭異命迈,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)火的,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,806評(píng)論 3 394
  • 文/潘曉璐 我一進(jìn)店門(mén)壶愤,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人馏鹤,你說(shuō)我怎么就攤上這事征椒。” “怎么了湃累?”我有些...
    開(kāi)封第一講書(shū)人閱讀 163,875評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵勃救,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我治力,道長(zhǎng)蒙秒,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,441評(píng)論 1 293
  • 正文 為了忘掉前任琴许,我火速辦了婚禮税肪,結(jié)果婚禮上溉躲,老公的妹妹穿的比我還像新娘榜田。我一直安慰自己,他們只是感情好锻梳,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,488評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布箭券。 她就那樣靜靜地躺著,像睡著了一般疑枯。 火紅的嫁衣襯著肌膚如雪辩块。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 51,365評(píng)論 1 302
  • 那天,我揣著相機(jī)與錄音废亭,去河邊找鬼国章。 笑死,一個(gè)胖子當(dāng)著我的面吹牛豆村,可吹牛的內(nèi)容都是我干的液兽。 我是一名探鬼主播,決...
    沈念sama閱讀 40,190評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼掌动,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼四啰!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起粗恢,我...
    開(kāi)封第一講書(shū)人閱讀 39,062評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤柑晒,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后眷射,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體匙赞,經(jīng)...
    沈念sama閱讀 45,500評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,706評(píng)論 3 335
  • 正文 我和宋清朗相戀三年妖碉,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了罚屋。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,834評(píng)論 1 347
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡嗅绸,死狀恐怖脾猛,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情鱼鸠,我是刑警寧澤猛拴,帶...
    沈念sama閱讀 35,559評(píng)論 5 345
  • 正文 年R本政府宣布,位于F島的核電站蚀狰,受9級(jí)特大地震影響愉昆,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜麻蹋,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,167評(píng)論 3 328
  • 文/蒙蒙 一跛溉、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧扮授,春花似錦芳室、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,779評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至荔仁,卻和暖如春伍宦,著一層夾襖步出監(jiān)牢的瞬間芽死,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 32,912評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工次洼, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留关贵,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 47,958評(píng)論 2 370
  • 正文 我出身青樓卖毁,卻偏偏與公主長(zhǎng)得像坪哄,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子势篡,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,779評(píng)論 2 354

推薦閱讀更多精彩內(nèi)容

  • 前言 Google Play應(yīng)用市場(chǎng)對(duì)于應(yīng)用的targetSdkVersion有了更為嚴(yán)格的要求翩肌。從 2018 年...
    申國(guó)駿閱讀 64,088評(píng)論 14 98
  • """1.個(gè)性化消息: 將用戶的姓名存到一個(gè)變量中,并向該用戶顯示一條消息禁悠。顯示的消息應(yīng)非常簡(jiǎn)單念祭,如“Hello ...
    她即我命閱讀 2,890評(píng)論 0 5
  • 我們都是軟弱的人,所以才會(huì)說(shuō)謊碍侦。我們都是膽小的人粱坤,所以才要武裝。我們都是一群笨蛋瓷产,所以才會(huì)互相傷害站玄。
    所羅門(mén)的偽證_dc0a閱讀 2,057評(píng)論 0 3
  • 為了讓我有一個(gè)更快速、更精彩濒旦、更輝煌的成長(zhǎng)株旷,我將開(kāi)始這段刻骨銘心的自我蛻變之旅!從今天開(kāi)始尔邓,我將每天堅(jiān)持閱...
    李薇帆閱讀 1,717評(píng)論 0 2
  • 似乎最近一直都在路上晾剖,每次出來(lái)走的時(shí)候感受都會(huì)很不一樣。 1梯嗽、感恩一直遇到好心人齿尽,很幸運(yùn)。在路上總是...
    時(shí)間里的花Lily閱讀 1,188評(píng)論 0 1