
本文深入探讨了go语言中因未初始化map或其内部结构体指针而导致的`invalid memory address or nil pointer dereference`运行时错误。通过分析一个在并发环境中向嵌套切片追加数据的常见场景,文章详细解释了如何正确初始化map、处理map中不存在的键,并安全地初始化嵌套的结构体和切片,从而避免运行时恐慌,确保程序的健壮性。
在Go语言中,panic: runtime error: invalid memory address or nil pointer dereference 是一个常见的运行时错误,通常意味着程序尝试访问一个未初始化的指针(即nil指针)所指向的内存地址。当涉及到复杂的嵌套数据结构,特别是包含map和结构体指针时,这类问题更容易出现。
考虑以下代码示例,它尝试在一个并发的goroutine中向一个嵌套在map中的结构体切片追加数据:
package main
import (
"fmt"
"sync"
"time"
)
var PairNames = []string{ "kalle", "kustaa", "daavid", "pekka" }
type Data struct {
a int
b int
}
type Tickers struct {
Tickers []Data
}
type Pairs struct {
Pair map[string]*Tickers
Mutex sync.Mutex
}
func (pairs Pairs) CollectTickers() {
PairCount := len(PairNames)
for x := 0; x <= 1000; x++ {
for i := 0; i < PairCount-1; i++ {
var data Data
data.a = i * x
data.b = i + x
pairs.Mutex.Lock()
// 错误发生在这里
pairs.Pair[PairNames[i]].Tickers = append(pairs.Pair[PairNames[i]].Tickers, data)
pairs.Mutex.Unlock()
fmt.Printf("a = %v, b = %v\r\n", data.a, data.b)
}
}
}
func main() {
var pairs Pairs // pairs.Pair 此时为 nil
go pairs.CollectTickers()
time.Sleep(100 * time.Second)
}运行上述代码会导致 panic: runtime error: invalid memory address or nil pointer dereference 错误,错误发生在 pairs.Pair[PairNames[i]].Tickers = append(...) 这一行。
这个错误主要由以下两个原因造成:
立即学习“go语言免费学习笔记(深入)”;
简而言之,问题在于在尝试访问 Tickers 字段之前,pairs.Pair 或 pairs.Pair[PairNames[i]] 中的至少一个为 nil。
要解决这个问题,我们需要确保在使用map和其内部的指针字段之前,它们都已经被正确地初始化。
在 main 函数中,必须先使用 make 函数初始化 Pairs 结构体中的 Pair map:
func main() {
var pairs = Pairs{
Pair: make(map[string]*Tickers), // 初始化map
}
go pairs.CollectTickers()
time.Sleep(1 * time.Second)
}在 CollectTickers 方法中,当从 pairs.Pair 中获取 *Tickers 类型的实例时,需要检查该键是否存在。如果不存在,则需要创建一个新的 Tickers 实例并将其添加到map中。
Go语言提供了一种惯用的方式来检查map中是否存在某个键,即“逗号-ok”语法:value, ok := myMap[key]。
func (pairs Pairs) CollectTickers() {
PairCount := len(PairNames)
for x := 0; x <= 1000; x++ {
for i := 0; i < PairCount-1; i++ {
var data Data
data.a = i * x
data.b = i + x
pairs.Mutex.Lock()
name := PairNames[i]
if t, ok := pairs.Pair[name]; ok {
// 键已存在,直接追加数据到其Tickers切片
t.Tickers = append(t.Tickers, data)
} else {
// 键不存在,创建新的Tickers实例并初始化其Tickers切片
pairs.Pair[name] = &Tickers{
Tickers: []Data{data}, // 初始化时直接包含第一个数据
}
}
pairs.Mutex.Unlock()
fmt.Printf("a = %v, b = %v\r\n", data.a, data.b)
}
}
}将上述两步修正合并到一起,得到一个可以正确运行且避免nil指针恐慌的代码:
package main
import (
"fmt"
"sync"
"time"
)
var PairNames = []string{"kalle", "kustaa", "daavid", "pekka"}
type Data struct {
a int
b int
}
type Tickers struct {
Tickers []Data
}
type Pairs struct {
Pair map[string]*Tickers
Mutex sync.Mutex
}
func (pairs Pairs) CollectTickers() {
PairCount := len(PairNames)
for x := 0; x <= 1000; x++ {
for i := 0; i < PairCount-1; i++ {
var data Data
data.a = i * x
data.b = i + x
pairs.Mutex.Lock() // 在访问map前加锁
name := PairNames[i]
if t, ok := pairs.Pair[name]; ok {
// 如果键已存在,直接向其Tickers切片追加数据
t.Tickers = append(t.Tickers, data)
} else {
// 如果键不存在,则创建新的Tickers实例并初始化其Tickers切片
pairs.Pair[name] = &Tickers{
Tickers: []Data{data}, // 初始化时包含第一个数据
}
}
pairs.Mutex.Unlock() // 访问map后解锁
fmt.Printf("a = %v, b = %v\r\n", data.a, data.b)
}
}
}
func main() {
// 初始化Pairs结构体,特别是其内部的map字段
var pairs = Pairs{
Pair: make(map[string]*Tickers),
}
go pairs.CollectTickers()
// 为了确保goroutine有足够时间运行,将sleep时间调整为1秒
time.Sleep(1 * time.Second)
}invalid memory address or nil pointer dereference 错误在Go语言中通常是由于未初始化或不当使用指针导致的。在处理包含map和嵌套结构体(特别是当这些结构体字段也是指针或切片时)的复杂数据结构时,务必牢记以下几点:
通过遵循这些最佳实践,可以显著提高Go语言程序的健壮性和可靠性,有效避免常见的运行时恐慌。
以上就是Go语言中处理嵌套结构体与并发Map的Nil指针恐慌及解决方案的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号