C#怎么给图片进行灰度处理_C#如何实现图像滤镜效果【解析】

3次阅读

应直接操作像素内存而非 GetPixel/SetPixel:先 LockBits 获取 Scan0 指针和 Stride,按加权公式 0.299R+0.587G+0.114B 计算灰度,优先用 Format32bppArgb 格式,处理完调 UnlockBits;ColorMatrix 适合 UI 预览但不灵活;保存 JPEG 需设 Quality≥95,跨 DPI 需指定分辨率防插值污染。

BitmapGetPixel/SetPixel 做灰度?别这么干

性能差到没法用,尤其图片稍大(比如 1000×1000),每像素调一次 getpixel 会触发 gdi+ 锁、颜色空间转换、边界检查——实际速度可能比纯 c++ 慢 50 倍以上。

真正该走的路是直接操作像素内存。核心就两步:锁定位图数据 → 按字节改 RGB 值。

  • 必须用 Bitmap.LockBits 获取 BitmapData,拿到 Scan0 指针和 Stride
  • 灰度公式推荐用加权平均:0.299 * R + 0.587 * G + 0.114 * B(人眼对绿色最敏感)
  • Stride 不等于宽度 × 字节,它按 4 字节对齐,比如 3 像素宽的 24 位图,Stride 是 12 而不是 9
  • 处理完务必调 UnlockBits,否则资源泄漏,多次调用后程序可能卡死或抛 OutOfMemoryException

为什么 PixelFormat.Format32bppArgbFormat24bppRgb 更好处理

32 位格式每像素固定 4 字节(RGBA),地址计算简单:第 (y * Stride + x * 4) 字节是 B,+1 是 G,+2 是 R,+3 是 A。而 24 位格式每行字节数不整除 4,x 偏移得手动算对齐,容易越界。

  • 创建新图时显式指定 PixelFormat.Format32bppArgb,避免隐式转换引入额外开销
  • 即使原图是 24 位,也先用 new Bitmap(src, ……) 转成 32 位再处理,比硬啃 24 位内存快且稳
  • Alpha 通道保留原值(不设为 0 或 255),否则透明区域会变黑或白

ColorMatrix 做灰度?适合批量但不够灵活

ColorMatrix 是 GDI+ 提供的矩阵变换方案,适合 UI 层快速预览或动画滤镜,但控制粒度粗、不能条件处理(比如只灰度非透明区域)。

  • 矩阵必须设成:0.299, 0.587, 0.114, 0, 0 这一行重复四次,最后一行保持 0,0,0,1,0(保证 Alpha 不变)
  • 要用 ImageAttributes + Graphics.DrawImage 才生效,不能直接改原图内存
  • 每次绘制都走完整渲染管线,CPU 占用高,不适合后台批量处理千张图
  • 如果图像带 ICC 配置文件,ColorMatrix 可能被忽略,结果偏色

灰度后保存为 JPEG 为什么发灰或对比度低?

不是算法问题,是 JPEG 编码器默认做了亮度 / 对比度补偿。原始灰度值 0–255 被压缩时,高频细节丢失更明显,尤其在暗部。

  • EncoderParameters 显式设置 Encoder.Quality 为 95 以上,避免过度压缩
  • 不要用 bitmap.Save("x.jpg") 默认参数,它等价于 Quality=80,暗部容易糊成一片
  • 如果需要精确控制,转成 Format8bppIndexed 并自定义调色板(只含 256 级灰阶),再保存为 PNG
  • 注意:JPEG 不支持 Alpha,保存前确保背景已合成,否则透明区域会变成黑色

最麻烦的其实是跨 DPI 场景下 Bitmap 创建时没指定分辨率,导致缩放后灰度值被插值污染——这个点连很多老手都会漏。

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