自定义 JSON 解组字符串编码数字

Custom JSON Unmarshalling for string-encoded number

我有一个 struct,其中包含各种货币值,以美分 (1/100 USD) 为单位:

type CurrencyValues struct {
   v1 int `json:"v1,string"`
   v2 int `json:"v2,string"`
} 

我想为带有千位分隔符的货币值创建自定义 json 解组器。这些值被编码为字符串,带有一个或多个千位分隔符 (,),可能还有一个小数点 (.)。

为此 JSON {"v1": "10", "v2": "1,503.21"},我想 JSON 解编一个 CurrencyValues{v1: 1000, v2: 150321}

按照此处的类似回答:Golang: How to unmarshall both 0 and false as bool from JSON,我继续为我的货币字段创建自定义类型,其中包括自定义解组函数:

type ConvertibleCentValue int

func (cents *ConvertibleCentValue) UnmarshalJSON(data []byte) error {
    asString := string(data)

    // Remove thousands separators
    asString = strings.Replace(asString, ",", "", -1)

    // Parse to float, then convert dollars to cents
    if floatVal, err := strconv.ParseFloat(asString, 32); err == nil {
        *cents = ConvertibleCentValue(int(floatVal * 100.0))
        return nil
    } else {
        return err
    }
}

但是,在编写单元测试时:

func Test_ConvertibleCentValue_Unmarshal(t *testing.T) {
    var c ConvertibleCentValue
    assert.Nil(t, json.Unmarshal([]byte("1,500"), &c))
    assert.Equal(t, 150000, int(c))
}

我遇到这个错误:

Error:      Expected nil, but got: &json.SyntaxError{msg:"invalid character ',' after top-level value", Offset:2}

我在这里错过了什么?

您正在尝试解组字符串 1,500,该字符串在 JSON 中无效。我想你的意思是解组 JSON 字符串 "1,500":

assert.Nil(t, json.Unmarshal([]byte(`"1,500"`), &c))

注意反引号。这是一个简化的例子:

b := []byte(`1,500`)
var s string
err := json.Unmarshal(b, &s)
fmt.Println(s, err) // Prints error.

b = []byte(`"1,500"`)
err = json.Unmarshal(b, &s)
fmt.Println(s, err) // Works fine.

游乐场:http://play.golang.org/p/uwayOSgmTv.