Image.Save 保存到 MemoryStream 时出现异常“GDI+ 中发生一般错误”

本文关键字:GDI+ 错误 异常 保存 Save MemoryStream Image | 更新日期: 2023-09-27 18:34:13

我有一个服务器客户端应用程序,我想从服务器获取屏幕截图,但是在bitmap.Save(ms, System.Drawing.Imaging.ImageFormat.Png);
我得到这个异常的行
A generic error occurred in GDI+.

    private Socket ScreenSocket;
    private MemoryStream ms;
    public void ConnectScreenShot(IPEndPoint ep)
    {
        if (ScreenSocket != null)
        {
            ScreenSocket.Dispose();
            ScreenSocket = null;
        }
        if (ms != null)
        {
            ms.Dispose();
            ms = null;
        }
        ScreenSocket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
        ScreenSocket.Connect(ep);
        ms = new MemoryStream();
        Rectangle bounds = Screen.GetBounds(Point.Empty);
        using (Bitmap bitmap = new Bitmap(bounds.Width, bounds.Height))
        {
            using (Graphics g = Graphics.FromImage(bitmap))
            {
                g.CopyFromScreen(Point.Empty, Point.Empty, bitmap.Size);
            }
            bitmap.Save(ms, System.Drawing.Imaging.ImageFormat.Png);
        }
    }

为什么会发生这种情况,我将如何解决它?

更新:当我使用 ImageFormat.Jpeg 而不是 ImageFormat.Png 时它可以工作,但我仍然需要 PNG 格式。

Image.Save 保存到 MemoryStream 时出现异常“GDI+ 中发生一般错误”

你说:

我想从服务器获取屏幕截图

与 GDI+ 相关的错误表明运行此代码的上下文可能是服务上下文(如 IIS),而不是桌面。这是对的吗?如果是这样,您希望在屏幕截图中返回什么?

作为记录,以下(最小)代码工作正常:

    var ms = new MemoryStream();
    Rectangle bounds = Screen.GetBounds(Point.Empty);
    using (Bitmap bitmap = new Bitmap(bounds.Width, bounds.Height))
    {
        using (Graphics g = Graphics.FromImage(bitmap))
        {
            g.CopyFromScreen(Point.Empty, Point.Empty, bitmap.Size);
        }
        bitmap.Save(ms, System.Drawing.Imaging.ImageFormat.Png);
    }

bitmapms都包含预期的数据。因此,我认为您需要提供有关此代码运行的位置和内容的信息。

我知道

这是一个旧帖子,但我也想添加这些信息,以防像我这样的人将来遇到这个问题。

我无法保存.PNG图像,这是因为 IIS 应用程序池帐户无法访问此密钥。

HKEY_CLASSES_ROOT''CLSID{FAE3D380-FEA4-4623-8C75-C6B61110B681}

当您使用 png 图形时,这是必需的。

对我来说

,我正在使用Image.Save(Stream, ImageCodecInfo, EncoderParameters),显然这导致了臭名昭著的A generic error occurred in GDI+错误。

我试图使用EncoderParameter以 100% 的质量保存 jpeg。这在"我的机器"(doh!)而不是生产上完美运行。

当我改用Image.Save(Stream, ImageFormat)时,错误消失了!所以像个白痴一样,我继续使用后者,尽管它以默认质量保存它们,我认为只有 50%。

希望这些信息对某人有所帮助。

它以这种方式与我一起工作:

  • 当客户端想要接收屏幕截图时,您需要在传输开始之前知道图像大小。
    调用 GetScreenShotSize() 以获取图像的大小。

  • 获得大小后,调用GetScreenShot()以接收图像数据。

我用过using (MemoryStream ms = new MemoryStream())所以现在 PNG 格式可以工作了。

    private Image img = null;
    public long GetScreenShotSize()
    {
        Rectangle bounds = Screen.GetBounds(Point.Empty);
        using (Bitmap bmp = new Bitmap(bounds.Width, bounds.Height))
        {
            using (Graphics g = Graphics.FromImage(bmp))
            {
                g.CopyFromScreen(Point.Empty, Point.Empty, bounds.Size);
            }
            using (MemoryStream ms = new MemoryStream())
            {
                bmp.Save(ms, System.Drawing.Imaging.ImageFormat.Png);
                img = Image.FromStream(ms);
                return ms.Length;
            }
        }
    }
    public void GetScreenShot(IPEndPoint ep)
    {
        using (Socket socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp))
        {
            socket.Connect(ep);
            using (MemoryStream ms = new MemoryStream())
            {
                img.Save(ms, System.Drawing.Imaging.ImageFormat.Png);
                socket.Send(ms.ToArray(), SocketFlags.None);
            }
            img.Dispose();
            img = null;
        }
    }
abrirConexion();
cmd = new SqlCommand(" INSERT INTO [SICRE_BI].[Credito_CCSS].[bi_img](Descripcion,img) VALUES (@nom_img,@img)",cn);                                
cmd.Parameters.Add("@Descripcion", SqlDbType.NChar);
cmd.Parameters.Add("@img", SqlDbType.Image);
cmd.Parameters["@Descripcion"].Value = descripcion;
System.IO.MemoryStream ms = new System.IO.MemoryStream();
pbImagen.Image.Save(ms, System.Drawing.Imaging.ImageFormat.Jpeg);//Aqui se me cae.Aqui se me cae.Aqui .
cmd.Parameters["@img"].Value = ms.GetBuffer();    //ms.GetBuffer();
cmd.ExecuteNonQuery();
return true;