确保导出的JPEG小于最大文件大小

时间:2011-05-11 20:53:36

标签: c# image encoding screenshot jpeg

我目前有一个应用程序,它会截取演示者桌面的屏幕截图,然后通过自定义协议将其广播给观众。为了使图像能够足够快地传输以获得每秒2-3个图像的帧速率,我需要确保图像大小始终小于~300 KB。

我正在使用C#作为演示者应用程序,它通过以下过程将屏幕截图编码为JPEG。我担心的是,使用静态压缩设置时图像质量会有很大差异。如果我有应用程序捕获我的屏幕,当我有Visual Studio全屏时,图像输出将是~200 KB,但如果我最小化我的屏幕并显示我的桌面背景,它将是~400 KB。

我可以将编码过程放入循环中,并不断减小图像大小,直到字节数组的大小小于300 KB,但这似乎是一项繁琐的操作。我还有其他方法吗?

提前致谢。

// get the screenshot
System.Drawing.Rectangle totalSize = System.Drawing.Rectangle.Empty;
//foreach (Screen s in Screen.AllScreens)
totalSize = System.Drawing.Rectangle.Union(totalSize, Screen.PrimaryScreen.Bounds);
Bitmap screenShotBitmap = new Bitmap(totalSize.Width, totalSize.Height, System.Drawing.Imaging.PixelFormat.Format32bppRgb);
screenShotBitmap.SetResolution(96, 96);
Graphics screenShotGraphics = Graphics.FromImage(screenShotBitmap);
screenShotGraphics.CopyFromScreen(totalSize.X, totalSize.Y,
                    0, 0, totalSize.Size, CopyPixelOperation.SourceCopy);
screenShotGraphics.Dispose();

// image codec information
ImageCodecInfo imageCodecInfo = GetEncoderInfo("image/jpeg");

// encoder settings
System.Drawing.Imaging.Encoder encoderQuality;
System.Drawing.Imaging.Encoder encoderColor;
encoderQuality = System.Drawing.Imaging.Encoder.Quality;
encoderColor = System.Drawing.Imaging.Encoder.ColorDepth;

// compression & quality for JPEG output
Int64 quality = 40L;

// storage for exported JPEG
byte[] screenShotByteArray;

// encoder parameters
EncoderParameter encoderQualityParameter = new EncoderParameter(encoderQuality, quality);
//EncoderParameter encoderColorParameter = new EncoderParameter(encoderColor, 8L);

// encoder parameters table
EncoderParameters encoderParameters = new EncoderParameters(1);
encoderParameters.Param[0] = encoderQualityParameter;
//encoderParameters.Param[1] = encoderColorParameter;

// get the code into a memory stream
MemoryStream screenShotMemoryStream = new MemoryStream();
screenShotBitmap.Save(screenShotMemoryStream, imageCodecInfo, encoderParameters);

// convert to a byte array
screenShotByteArray = screenShotMemoryStream.GetBuffer();

// close the memory stream
screenShotMemoryStream.Close();

1 个答案:

答案 0 :(得分:5)

如果你把东西放进一个循环中,小心使用类似于二分搜索的东西,而不是仅仅按照固定的数量增加/减少质量参数,直到达到所需的大小。

编辑:稍微解释二进制搜索。假设压缩到quality*10000字节的图片的假设情况,因此最佳质量设置为30.现在,天真的方法是尝试一些固定的质量设置(fe 80,这将给出800,000字节),然后减少一定量,直到达到300000字节。如果你f.e.在每个步骤中将图像质量降低5,您可以使用此方法尝试12种质量设置,直到找到所需的设置。 binary search会更快地提供结果,如下所示:

Quality    Size    Next step
80         800000  Too big, so quality := quality/2
40         400000  Too big, so quality := quality/2
20         200000  Too small, so quality := (40+20)/2
30         300000  Reached desired size

这仅在4次尝试后给出结果(或3次,取决于200000字节太小或对你来说没问题)。由于大小与质量没有线性关系,这个例子有点不切实际,但二元搜索仍然应该比天真的方法给你更好的结果。

您还可以使用一些典型的图像进行“训练”。使用不同的质量设置(f.e. 100,90,...,20,10)对它们进行编码,看看它们相对于原始尺寸有多大。在大多数情况下,这可能会给出一个很好的初步估计,尽管在遇到包含更多或更少细节的图像时仍需要进行调整。

或者,看看JPEG2000编码器,那些可以选择设置文件大小而不是质量。

编辑:我不知道用于C#的JPEG2000编码库,似乎只有解码器浮动,所以这可能比我想象的要复杂得多。您可以尝试CSJ2K,但说明听起来不像是可以使用的。