根本原因是 Go 的 image 包未自动注册 JPEG/GIF 解码器,需显式导入 "image/jpeg" 或 "image/gif";PNG 已内置注册。
image.Decode 读取图片时为什么总返回 “unknown format”
根本原因不是文件损坏,而是 Go 没有自动注册解码器。标准库的 image 包默认只注册了 png,jpeg 和 gif 需要显式导入对应子包,否则 image.Decode 碰到 .jpg 或 .gif 就会失败。
jpeg:必须 import _ "image/jpeg"
gif:必须 import _ "image/gif"
png:可省略(image/png 已内置注册)import _ "xxx",下划线表示只执行包初始化,不引入符号resize 不是 image 标准库的功能,得靠第三方包Go 标准库没有缩放、裁剪、旋转等高级操作——它只提供底层像素访问和基础格式编解码。想 resize 图片,必须引入外部包,最常用的是 github.com/nfnt/resize(轻量、纯 Go)或 golang.org/x/image/draw(官方维护,但需手动实现缩放逻辑)。
nfnt/resize 支持双线性插值,调用简单:resize.Resize(width, height, src, resize.Lanczos3)
golang.org/x/image/draw 更底层,需自己创建目标 *image.NRGBA,再调用 draw.Bilinear 等函数image.SubImage 做裁剪后保存——它返回的是共享底层数组的视图,若原图后续被修改或释放,结果图可能出错;应 copy 到新图像jpeg.Encode 的 *os.File 是否已关闭常见错误是打开文件写入后没等 jpeg.Encode 完成就 Close() 了,或者忘记在 Encode 前设置 os.O_CREATE | os.O_WRONLY | os.O_TRUNC 标志,导致写入失败但无报错。
os.OpenFile(path, os.O_CREATE|os.O_WRONLY|os.O_TRUNC, 0644)
jpeg.Encode 第二个参数必须是实现了 io.Writer 的对象(如 *os.File),不能传 nil 或未初始化的变量*image.NRGBA(含 alpha),需先转成 *image.RGBA 并丢弃 alpha,否则编码会静默失败或颜色偏移dst := image.NewRGBA(image.Rect(0, 0, w, h)) draw.Draw(dst, dst.Bounds(), src, image.Point{}, draw.Src) // 再 encode,不能直接 encode src 如果它是 NRGBA
image.Decode 后记得 src = nil
image.Decode 返回的图像对象(如 *image.RGBA)底层持有大块像素内存,GC 不会立即回收,尤其在循环中反复解码高分辨率图时,容易触发频繁 GC 或 OOM。
img = nil
resize.Resize 的流式方式(传入 io.Reader),避免一次性全量解码进内存image.Decode 本身不并发安全,多个协程共用同一输入流会出错image 对象生命周期里」。这些点不踩一遍坑,光看文档很难意识到。