-
Go中接口变量底层是两个字宽的iface结构体,仅当tab和data均为nil时才是真nil;若tab非nil(如(*T)(nil)赋值),接口不为nil但data为空指针。
-
reflect.TypeOf返回变量的实际类型,Name()仅对命名类型非空,Kind()才可靠判断基础类别;结构体私有字段不可见,需用Elem()、Tag等配合;反射性能低且易panic,应缓存Type并先判空。
-
要让函数修改值类型变量需使用指针传递,因Go语言参数传递本质为值传递,函数接收变量副本,修改不影响原始变量;通过传递变量地址,函数可解引用指针修改原始值,如changeValue函数无法改变外部num,而changePointer通过指针成功修改anotherNum;Go坚持值传递以提升代码清晰度和可预测性,减少隐式副作用,利于并发编程,但对大型结构体可能带来性能开销,此时指针传递可优化性能;典型使用场景包括需修改外部状态、处理大型结构体、方法修改接收者及处理可选参数;使用指针需防范空指针解引用、避免不必
-
Go标准库container包里的List、Heap、Ring不是“通用替代品”,而是为特定场景设计的轻量工具——用错地方反而比手写slice或map更慢、更难维护。什么时候该用container/list.List?list.List是双向链表,唯一不可被slice替代的场景是:需要在任意位置频繁InsertBefore/InsertAfter/MoveToBack,且不能接受O(n)查找开销(比如LRU缓存淘汰、事件调度器中的动态优先级调整)。常见错误现象:
-
本文详解通过GitURL重写机制,让goget支持基于SSH地址(如git@1.2.3.4:group/repo.git)的私有GitLab仓库,解决Go原生不识别SSH格式导入路径的问题。
-
可用runtime.NumCPU()获取逻辑CPU数、runtime.NumGoroutine()获取瞬时Goroutine数,但需注意其局限性:前者含超线程且容器中常返回宿主机值,后者为快照且含系统goroutine,仅宜趋势观察。
-
用reflect.Value遍历时怎么避免栈溢出Go的反射本身不阻止循环引用,reflect.Value递归调用Interface()或Elem()时,一旦结构体字段指向自身或形成环,就会直接panic:「runtime:goroutinestackexceeds1000000000-bytelimit」。这不是反射的bug,而是你没设访问边界。实操上必须自己维护已访问对象的标识。不能只比对指针地址(unsafe.Pointer),因为相同地址可能来自不同reflect
-
用os.Stat检查文件是否存在是最稳妥的方式,需用os.IsNotExist(err)判断错误类型,而非err==os.ErrNotExist或os.Open;os.Stat不打开文件、轻量安全,且默认解引用符号链接,检查链接本身存在性应使用os.Lstat。
-
Go语言通过value,ok:=interface{}.(Type)安全检视接口值的具体类型,非强制转换;支持typeswitch批量判断,仅适用于接口类型,断言开销小但需避免重复,nil接口断言依目标类型而定。
-
UDP组播必须绑定0.0.0.0或局域网IP而非127.0.0.1,且需调用SetMulticastInterface指定发送网卡、JoinGroup加入组播组,否则无法收发。
-
replace只在本地生效,因它仅修改go.mod中的指令并影响当前模块构建,不上传远程、不被其他模块继承,且需手动清理才能安全上线。
-
goroutine泄漏比CPU跑满更常见,需用semaphore控并发数、tokenbucket控速率;错误复用或未释放会导致OOM或阻塞,应分层混合使用并监控。
-
使用别名可简化长包名引用并提升可读性,如jsoniter"github.com/json-iterator/go";2.同名包导入时需用别名避免冲突,如myutils"projectB/utils";3.第三方库与标准库同名时应为第三方库设别名以明确职责,如httphelper"myproject/pkg/http";4.匿名导入用于触发init副作用,如\_"github.com/go-sql-driver/mysql"注册驱动;5.团队协作中应统一别名风格,优先使用语义清晰的短别名。
-
结构体字段顺序影响内存占用是因为Go不自动重排字段,需手动按从大到小排列(如int64→int32→int16→bool)以减少对齐填充;验证需用unsafe.Sizeof/Offsetof实测,但大数组、CGO或语义分组场景下重排可能无效或有害。
-
使用无缓冲channel实现事件通知,主线程等待子任务完成。2.worker函数执行完毕后通过done<-true发送信号,main函数接收信号后继续执行,实现goroutine间同步。