代码示例:

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中浮点型默认使用银行家舍入法。