Image.Save 保存 JPEG 时需用 EncoderParameters 显式设置 Quality 参数(0–100L),配合 ImageCodecInfo 获取 JPEG 编码器;SetResolution 仅改 DPI 元数据,不减体积;缩放需用 Graphics.DrawImage 并设 HighQualityBicubic;WebP 在 .NET 中质量不可控且跨平台支持差;务必 using 管理 Image/Bitmap 生命周期防内存泄漏。

用 Image.Save 时怎么控制 JPEG 质量
默认保存的 Jpeg 图片质量是 80%,但这个值藏在编码器参数里,不显式设置就根本改不了。直接调 image.Save("x.jpg", ImageFormat.Jpeg) 是无效的——它压根不读你的质量意图。
必须用 EncoderParameters 配合 ImageCodecInfo 才能生效。常见错误是只查到 Encoder.Quality,却没找到对应编码器实例,结果参数传不进去,还报 Parameter is not valid。
- 先用
ImageCodecInfo.GetImageEncoders()找到image/jpeg对应的编码器 -
EncoderParameters必须且只能包含一个EncoderParameter,EncoderParameter.Encoder值要严格匹配Encoder.Quality - 质量值是
long类型,范围 0–100,不是 float;传 75 就写75L,别传75.0或75(隐式 int 在某些 .NET 版本会失败)
var encoder = ImageCodecInfo.GetImageEncoders() .First(x => x.MimeType == "image/jpeg"); var parameters = new EncoderParameters(1); parameters.Param[0] = new EncoderParameter(Encoder.Quality, 60L); image.Save("out.jpg", encoder, parameters);
为什么 Bitmap.SetResolution 不影响文件体积
很多人以为调 bitmap.SetResolution(72, 72) 能“减小图片尺寸”,其实这只是改了 DPI 元数据,像素总数完全不变,保存后体积几乎无变化。真正影响体积的是像素数量和压缩质量。
如果你拿到的是高分辨率截图(比如 3840×2160),又想发微信或网页用,光设 DPI 没用,必须缩放画布。
- 用
Graphics.DrawImage缩放到目标尺寸,再保存,才是有效降体积手段 - 缩放时注意插值模式:
graphics.InterpolationMode = InterpolationMode.HighQualityBicubic,否则拉伸模糊或锯齿明显 - 别用
Bitmap.Clone(Rectangle, PixelFormat)截图式缩放——它不重采样,只是硬裁,视觉效果差
WebP 支持在 .NET 中有多坑
.NET 6+ 原生支持 ImageFormat.WebP,但仅限 Windows 10 1903+ 和 macOS 12.0+;Linux 下即使装了 libwebp,Save 仍会抛 System.PlatformNotSupportedException。
更隐蔽的问题是:WebP 默认用有损压缩,但没暴露质量参数入口。你写 image.Save("x.webp", ImageFormat.WebP),实际质量由系统决定(通常≈75),无法手动调。
- 真要控质量,得用第三方库如
Microsoft.Toolkit.WinUI.UI.Controls(Windows UWP)或ImageSharp -
ImageSharp是目前最稳的选择:encoder.Quality = 65直接生效,跨平台,不依赖系统编解码器 - 别在生产环境试 WebP +
Image.Save组合——本地跑通不等于服务器可用
内存泄漏比体积问题更值得警惕
反复用 Image.FromFile 加载再 Save,不调 Dispose,几轮下来就 OOM。尤其在 ASP.NET Core 后端批量处理图片时,Image 对象背后绑着 GDI+ 句柄,GC 不自动释放。
容易被忽略的一点:Bitmap 构造函数如果传入流(如 new Bitmap(stream)),该流 ** 必须保持打开状态直到 Bitmap 被释放 **。提前 stream.Close() 会导致后续 Save 报 Object is disposed。
- 务必用
using包裹所有Image、Bitmap、Graphics实例 - 从文件加载 → 处理 → 保存,整条链路都别用静态缓存
Image对象 - 若需复用,转成
byte[]或MemoryStream再重建,别留着原始Image引用
质量参数好设,但资源生命周期没管住,再小的图也能把服务拖垮。