代码示例:

package main

import (
  "fmt"
  "math"
  "reflect"
  "strconv"
)

func main() {
  numF := 0.2253

  // 保留两位小数, 通用
  value, _ := strconv.ParseFloat(fmt.Sprintf("%.2f", numF), 64)
  fmt.Println(reflect.TypeOf(value), value)

  num, _ := FormatFloat(numF, 2)
  fmt.Println(reflect.TypeOf(num), num)

  // 舍弃的尾数不为0,强制进位
  num, _ = FormatFloatCeil(0.2205, 2)
  fmt.Println(reflect.TypeOf(num), num)

  // 强制舍弃尾数
  num, _ = FormatFloatFloor(0.2295, 2)
  fmt.Println(reflect.TypeOf(num), num)

  // 四舍六入五考虑,五后非零就进一,五后为零看奇偶,五前为偶应舍去,五前为奇要进一
  fmt.Printf("9.8249	=>	%0.2f(四舍)\n", 9.8249)
  fmt.Printf("9.82671	=>	%0.2f(六入)\n", 9.82671)
  fmt.Printf("9.8351	=>	%0.2f(五后非零就进一)\n", 9.8351)
  fmt.Printf("9.82501	=>	%0.2f(五后非零就进一)\n", 9.82501)
  fmt.Printf("9.8250	=>	%0.2f(五后为零看奇偶,五前为偶应舍去)\n", 9.8250)
  fmt.Printf("9.8350	=>	%0.2f(五后为零看奇偶,五前为奇要进一)\n", 9.8350)
}

// 保留两位小数,舍弃尾数,无进位运算
// 主要逻辑就是先乘,trunc之后再除回去,就达到了保留N位小数的效果
func FormatFloat(num float64, decimal int) (float64, error) {
  // 默认乘1
  d := float64(1)
  if decimal > 0 {
    // 10的N次方
    d = math.Pow10(decimal)
  }
  // math.trunc作用就是返回浮点数的整数部分
  // 再除回去,小数点后无效的0也就不存在了
  res := strconv.FormatFloat(math.Trunc(num*d)/d, 'f', -1, 64)
  return strconv.ParseFloat(res, 64)
}

// 舍弃的尾数不为0,强制进位
func FormatFloatCeil(num float64, decimal int) (float64, error) {
  // 默认乘1
  d := float64(1)
  if decimal > 0 {
    // 10的N次方
    d = math.Pow10(decimal)
  }
  // math.trunc作用就是返回浮点数的整数部分
  // 再除回去,小数点后无效的0也就不存在了
  res := strconv.FormatFloat(math.Ceil(num*d)/d, 'f', -1, 64)
  return strconv.ParseFloat(res, 64)
}

// 强制舍弃尾数
func FormatFloatFloor(num float64, decimal int) (float64, error) {
  // 默认乘1
  d := float64(1)
  if decimal > 0 {
    // 10的N次方
    d = math.Pow10(decimal)
  }
  // math.trunc作用就是返回浮点数的整数部分
  // 再除回去,小数点后无效的0也就不存在了
  res := strconv.FormatFloat(math.Floor(num*d)/d, 'f', -1, 64)
  return strconv.ParseFloat(res, 64)
}

运行结果:

float64 0.23
float64 0.22
float64 0.23
float64 0.22
9.8249  =>      9.82(四舍)
9.82671 =>      9.83(六入)
9.8351  =>      9.84(五后非零就进一)
9.82501 =>      9.83(五后非零就进一)
9.8250  =>      9.82(五后为零看奇偶,五前为偶应舍去)
9.8350  =>      9.84(五后为零看奇偶,五前为奇要进一)

相关说明:

四舍六入五成双是一种比较精确比较科学的计数保留法,是一种数字修约规则,又名银行家舍入法。它比通常用的四舍五入法更加精确。

具体规则:

  • 被修约的数字小于5时,该数字舍去;
  • 被修约的数字大于5时,则进位;
  • 被修约的数字等于5时,要看5前面的数字,若是奇数则进位,若是偶数则将5舍掉,即修约后末尾数字都成为偶数;若5的后面还有不为“0”的任何数,则此时无论5的前面是奇数还是偶数,均应进位。

助记口诀:

四舍六入五考虑,五后非零就进一,五后为零看奇偶,五前为偶应舍去,五前为奇要进一

Golang中浮点型默认使用银行家舍入法。