-
答案:Go1.13引入errors.Is和errors.As用于处理包装错误。1.errors.Is判断错误值是否匹配目标,可穿透多层包装,适用于识别预定义错误如os.ErrNotExist。2.errors.As判断错误链中是否存在指定类型并赋值,适用于提取自定义错误类型的字段或方法。两者均能正确处理错误包装,避免传统==或类型断言无法穿透包装的问题,提升错误处理的准确性和健壮性。
-
Go跨包复用核心是导入路径与文件系统、go.mod、Git仓库三者严格一致:需用可导出标识符、合法importpath、模块初始化(go.mod),避免GOPATH干扰,私有包可通过replace或GitURL共享。
-
Go中通过指针操作数组可避免值拷贝,提升性能。数组是值类型,传递时会复制整个数组,而使用指向数组的指针(*[N]T)可直接修改原数据,适用于大数组或性能敏感场景。声明数组指针后,用&取地址赋值,可通过ptr[i]语法直接访问元素,无需显式解引用。函数传参时,数组指针仅传递8字节指针,效率远高于值传递的复制开销。相比切片,数组指针适用于固定长度、强调性能和内存布局可控的场景;而切片更灵活,适合动态长度操作。合理选择传递方式,能优化内存使用与程序效率。
-
不能,Go反射不支持自动类型推导转换;必须显式指定目标类型,否则Convert()或调用方法会panic;安全做法是类型断言处理已知类型,反射仅用于运行时类型不确定的场景如ORM、反序列化。
-
Go文件上传必须先调ParseMultipartForm,否则r.FormFile和r.MultipartForm.File均失效;常见错误是未调用或调用过晚(如在FormValue后),导致返回http.ErrNotMultipart。
-
interface{}与nil直接比较常失效,因其是含类型和数据的两字宽结构体;即使底层值为nil,只要类型信息非空,接口值就不等于nil。
-
chan适用于单机低频可丢任务场景;需缓冲和消费者goroutine防阻塞;asynq/machinery用于持久化跨进程等需求,配置错误易致任务不消费或注册失败。
-
Go中使用指针可避免数组拷贝并直接修改原数据,因数组是值类型,传参时需传递指针(如[3]int)以实现外部修改,通过&取地址、解引用操作元素,函数中常见此用法;而切片为引用类型,无需显式取指针即可修改底层数组。
-
跨服WebSocket单发与群发需依赖RedisPub/Sub实现消息中继,因gorilla/websocket仅支持本地连接操作,各节点须独立维护clientsmap并订阅统一频道,通过user_id定位目标实例后定向推送或按房间分组广播。
-
json.Marshal返回空字符串或nil的常见原因是结构体字段未导出(首字母小写),encoding/json只能序列化导出字段;解决方式是将字段首字母大写或用json:"name"显式标记,且字段必须导出。
-
直接用cmp.Diff比较Protobuf消息会误报不等,因其默认反射比对未导出字段(如XXX_sizecache、XXX_unrecognized)、time.Time纳秒差异及map顺序;须用protocmp.Transform()跳过私有字段并做语义比较。
-
sync.Pool仅对极少数场景有效,误用会导致RSS升高、GC变慢、bug难查;不可池化的对象包括*sql.DB等资源型、小对象、含sync.Mutex的结构体及大结构体;New是兜底工厂而非构造函数,需返回全新对象;Get后必须手动Reset;Put表示放弃所有权,需严守作用域避免竞争。
-
Go1.14+的for{}不再完全卡死调度器,因引入基于信号的异步抢占机制,可在系统调用、GC或定时器唤醒时强制调度,但对纯计算无函数调用、CGO、runtime临界区等场景仍无效。
-
Go语言处理长连接需协同TCP层心跳、应用层心跳、连接池管理与优雅关闭:启用SetKeepAlive并设周期,加PING/PONG应用心跳,用健康连接池自动重建,读写绑定context并响应信号优雅关闭。
-
defer不能捕获panic但保证清理执行,其核心价值在于panic时仍运行以确保资源释放;它不处理错误,需显式检查;命名返回参数可配合defer统一赋值错误,但要注意作用域与覆盖逻辑。