go 自定义mysql中gorm时间字段属性值为“0001-01-01 00:00:00“情况处理

在开发中,遇到MySQL时间字段转换为JSON时格式不符的问题,导致使用不便。通过创建自定义LocalTime类型并重写MarshalJSON和UnmarshalJSON方法,解决了时间字段为null时显示为0001-01-01 00:00:00而非null的状况。现在,当时间为空或零值时,将返回null,确保了数据交换的准确性。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

        我们在开发中mysql的时间字段属性一般用到的是time.Time但是做成json返回的时候时间就不是 “2022-04-13 01:26:46” 这种时间格式,导致我们使用的时候很不方便,跟前端或者其他系统接收数据都会造成麻烦。所以我们要自定义时间类型,从而能解决这个问题。

        晚上找到这种时间的写法后发现自己的项目用起来一个问题就是,如果时间字段值是null,它默认会把时间的写成 "0001-01-01 00:00:00" 并不是null,这时候我们就需要修改localTime的MarshalJSON方法:

func (t LocalTime) MarshalJSON() ([]byte, error) {
	tTime := time.Time(t)
	// 如果时间值是空或者0值 返回为null 如果写空字符串会报出异常时间
    // 下面是修复0001-01-01问题的
	if &t == nil || t.IsZero() {
		return []byte("null"), nil
	}
	return []byte(fmt.Sprintf("\"%s\"", tTime.Format("2006-01-02 15:04:05"))), nil
}

这时候我们的返回值就不会有问题了。对比一下前后区别:

 加入后:

 我这里再贴一下我写的LocalTime 跟网络上的不完全一样,做了下修改,不然很多人的项目跑起来会有各种各样的问题,我这个相对比较完善:

package utils

import (
	"database/sql/driver"
	"fmt"
	"strings"
	"time"
)

/**
 * @note: 返回值为 年月日时分秒格式,加强易读性
 * @auth: t
 * @date 2022年3月29日15:34:03
**/
type LocalTime time.Time

func (t LocalTime) Value() (driver.Value, error) {
	var zeroTime time.Time
	tlt := time.Time(t)
	if tlt.UnixNano() == zeroTime.UnixNano() {
		return nil, nil
	}
	return tlt, nil
}

func (t *LocalTime) Scan(v interface{}) error {
	if value, ok := v.(time.Time); ok {
		*t = LocalTime(value)
		return nil
	}
	return fmt.Errorf("can not convert %v to timestamp", v)
}

func (t *LocalTime) String() string {
	// 如果时间 null 那么我们需要把返回的值进行修改
	if t == nil || t.IsZero() {
		return ""
	}
	return fmt.Sprintf("%s", time.Time(*t).Format("2006-01-02 15:04:05"))
}

func (t *LocalTime) IsZero() bool {
	return time.Time(*t).IsZero()
}

func (t *LocalTime) UnmarshalJSON(data []byte) error {
	if string(data) == "null" {
		return nil
	}
	var err error
	//前端接收的时间字符串
	str := string(data)
	//去除接收的str收尾多余的"
	timeStr := strings.Trim(str, "\"")
	t1, err := time.ParseInLocation("2006-01-02 15:04:05", timeStr, time.Local)
	*t = LocalTime(t1)
	return err
}

func (t LocalTime) MarshalJSON() ([]byte, error) {
	tTime := time.Time(t)
	// 如果时间值是空或者0值 返回为null 如果写空字符串会报错
	if &t == nil || t.IsZero() {
		return []byte("null"), nil
	}
	return []byte(fmt.Sprintf("\"%s\"", tTime.Format("2006-01-02 15:04:05"))), nil
}
type SensitiveRuleInfoRsp struct {
	Id           int             `gorm:"primary_key" json:"id"`
	Name         string          `gorm:"name" json:"name"`
	RuleType     int             `gorm:"rule_type" json:"rule_type"`
	Status       int             `gorm:"status" json:"status"`
	Operator     string          `gorm:"operator" json:"operator"`
	CreatedAt    utils.LocalTime `gorm:"created_at" json:"created_at"`
	UpdatedAt    utils.LocalTime `gorm:"updated_at"json:"updated_at"`
	WordsCount   int             `json:"words_count"`    // 敏感词数量
	StatusStr    string          `json:"status_str"`     // 状态中文
	RuleTypeName string          `json:"rule_type_name"` // 规则类型中文名称
}

评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值