C#怎么压缩图片体积_C#如何修改Image质量并保存【技巧】

1次阅读

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

C#怎么压缩图片体积_C# 如何修改 Image 质量并保存【技巧】

Image.Save 时怎么控制 JPEG 质量

默认保存的 Jpeg 图片质量是 80%,但这个值藏在编码器参数里,不显式设置就根本改不了。直接调 image.Save("x.jpg", ImageFormat.Jpeg) 是无效的——它压根不读你的质量意图。

必须用 EncoderParameters 配合 ImageCodecInfo 才能生效。常见错误是只查到 Encoder.Quality,却没找到对应编码器实例,结果参数传不进去,还报 Parameter is not valid

  • 先用 ImageCodecInfo.GetImageEncoders() 找到 image/jpeg 对应的编码器
  • EncoderParameters 必须且只能包含一个 EncoderParameterEncoderParameter.Encoder 值要严格匹配 Encoder.Quality
  • 质量值是 long 类型,范围 0–100,不是 float;传 75 就写 75L,别传 75.075(隐式 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() 会导致后续 SaveObject is disposed

  • 务必用 using 包裹所有 ImageBitmapGraphics 实例
  • 从文件加载 → 处理 → 保存,整条链路都别用静态缓存 Image 对象
  • 若需复用,转成 byte[]MemoryStream 再重建,别留着原始 Image 引用

质量参数好设,但资源生命周期没管住,再小的图也能把服务拖垮。

星耀云
版权声明:本站原创文章,由 星耀云 2026-03-23发表,共计1975字。
转载说明:转载本网站任何内容,请按照转载方式正确书写本站原文地址。本站提供的一切软件、教程和内容信息仅限用于学习和研究目的;不得将上述内容用于商业或者非法用途,否则,一切后果请用户自负。本站信息来自网络,版权争议与本站无关。
text=ZqhQzanResources