• Go invalid memory address or nil pointer dereference错误 空指针问题


    Go 指针声明后赋值,出现 panic: runtime error: invalid memory address or nil pointer dereference,这种是内存地址错误。

    首先我们要了解指针,指针地址
    在 Go 中 * 代表取指针地址中存的值,& 代表取一个值的地址
    对于指针,我们一定要明白指针储存的是一个值的地址,但本身这个指针也需要地址来储存

    错误示例

    1. package main
    2. import "fmt"
    3. func main() {
    4. var i *int
    5. fmt.Println(&i, i)
    6. *i = 1
    7. fmt.Println(&i, i, *i)
    8. }

    错误提示

    1. 0xc00009a008 <nil>
    2. panic: runtime error: invalid memory address or nil pointer dereference
    3. [signal SIGSEGV: segmentation violation code=0x1 addr=0x0 pc=0x1092f9c]

    初始化指针,指针变量的地址为0xc00009a008, 指针值为nil
    此时,i为nil,系统没有给*i分配地址,相当于给一个nil地址赋值,肯定会出错

    解决办法是,预先分配一个内存地址给到指针变量

    new初始化内存地址

    1. package main
    2. import "fmt"
    3. func main() {
    4. var i *int
    5. fmt.Println(&i, i) // 0xc000094010
    6. i = new(int)
    7. fmt.Println(&i, i, *i) //0xc000094010 0xc0000a2010 0
    8. }
    • i 为指针变量,它的内存地址为0xc000094010
    • i的值为0xc0000a2010
    • 内存地址0xc0000a2010值为0
    举个例子:

    当status 请求参数超出范围时报错 runtime error: invalid memory address or nil pointer dereference

    1. func UpdateRiskEventStatus(ctx *gin.Context, req *message.UpdateRiskEventStatusReq) (*message.UpdateRiskEventStatusResp, error) {
    2. var (
    3. err error
    4. now = time.Now()
    5. DealRecord = &models.DealRecord{}
    6. result = &message.UpdateRiskEventStatusResp{}
    7. )
    8. //1.修改事件状态
    9. if req.Status < 0 || req.Status > 4 {
    10. log.Info("处置状态超出范围【0-4】")
    11. //result.ReplyErrCode(message.ParamsParseErrCode, err.Error())
    12. return result, err
    13. }
    14. err = models.UpdateRiskEventStatus(req.EventID, map[string]interface{}{
    15. "deal_status": req.Status,
    16. })
    17. if err != nil {
    18. result.ReplyErrCode(message.DBErrCode, err.Error())
    19. log.Errorf("UpdateRiskEventStatus.err: %v", err.Error())
    20. return result, err
    21. }
    22. 2.增加人员操作记录
    23. DealRecord.CreatedAt = now.Unix()
    24. DealRecord.EventID = req.EventID
    25. DealRecord.Status = req.Status
    26. DealRecord.Remark = req.Remark
    27. DealRecord.OperateUser = req.OperateUser
    28. err = models.CreateDealRecord(DealRecord)
    29. if err != nil {
    30. result.ReplyErrCode(message.DBErrCode, err.Error())
    31. log.Errorf("CreateDealRecord.err %v", err.Error())
    32. return result, err
    33. }
    34. return result, err
    35. }

    在你的代码中,当 req.Status 超出预期范围时,你返回 resulterr,但在此之后,你仍然继续执行代码,而没有检查 result 是否为空。这可能导致在处理之后 result 是空指针,导致 "nil pointer dereference" 错误。

    针对请求参数超出范围时,你已经返回了 resulterr,应该在这一点上结束函数的执行,而不是继续执行余下的代码。

    建议在参数超出范围时直接返回,避免继续执行后续代码,可以像下面这样修改你的代码:

     err = errors.New("Status out of range [0-4]") // 注意添加测行,即可
    1. func UpdateRiskEventStatus(ctx *gin.Context, req *message.UpdateRiskEventStatusReq) (*message.UpdateRiskEventStatusResp, error) {
    2. var (
    3. err error
    4. now = time.Now()
    5. DealRecord = &models.DealRecord{}
    6. result = &message.UpdateRiskEventStatusResp{}
    7. )
    8. // 1.修改事件状态
    9. if req.Status < 0 || req.Status > 4 {
    10. log.Info("处置状态超出范围【0-4】")
    11. result.ReplyErrCode(message.ParamsParseErrCode, "Status out of range [0-4]")
    12. err = errors.New("Status out of range [0-4]") // 注意添加测行,即可
    13. return result, err
    14. }
    15. err = models.UpdateRiskEventStatus(req.EventID, map[string]interface{}{
    16. "deal_status": req.Status,
    17. })
    18. if err != nil {
    19. result.ReplyErrCode(message.DBErrCode, err.Error())
    20. log.Errorf("UpdateRiskEventStatus.err: %v", err.Error())
    21. return result, err
    22. }
    23. // 2.增加人员操作记录
    24. DealRecord.CreatedAt = now.Unix()
    25. DealRecord.EventID = req.EventID
    26. DealRecord.Status = req.Status
    27. DealRecord.Remark = req.Remark
    28. DealRecord.OperateUser = req.OperateUser
    29. err = models.CreateDealRecord(DealRecord)
    30. if err != nil {
    31. result.ReplyErrCode(message.DBErrCode, err.Error())
    32. log.Errorf("CreateDealRecord.err: %v", err.Error())
    33. return result, err
    34. }
    35. return result, err
    36. }

    这样修改后,当 req.Status 超出范围时,函数将立即返回,避免继续执行可能导致空指针错误的代码。

  • 相关阅读:
    JSTL标签库和EL表达式
    自媒体视频剪辑中的视频素材是从哪里找的?
    2023.11.12使用flask对图片进行黑白处理(base64编码方式传输)
    【MATLAB第74期】#源码分享 | 基于MATLAB的ARX-ARMAX线性自回归移动平均外生模型(结合最小二乘思路)
    安装使用RocketMQ一套保姆全教程-最快完成SpringBoot使用消息队列demo
    【Linux】环境基础开发工具使用 - 软件包管理yum _vim _gcc/g++ _gdb
    ActiveReports.NET 16.2RPX 部分报告的完全支持
    Nginx R31 doc-17-debugging 调试
    如何认识python
    Java 8的菱形继承冲突解决之道
  • 原文地址:https://blog.csdn.net/lxw1844912514/article/details/134209673