关于"gdi+ 发生一般性错误"的处理.

做一个将图片存入数据库的程序,写了一个将图片转为字节数组的函数,如下:
        public static byte[] GetImageArray(System.Drawing.Image image)
        
{
            System.IO.MemoryStream ms 
= new MemoryStream();
            image.Save(ms, Image.RawFormat);
            
byte[] b=ms.ToArray();
            ms.Close();
            
return b;
        }
在调用的时候,image参数来源于一个PictureBox中引用的Image对象.开始的时候以为这么简单的事情应该没什么,但是发现如果图片是从一个数据库中加载的,然后显示到PictureBox的时候,将这个PictureBox中引用的Image传送到这个方法中,将会发生"gdi+ 发生一般性错误",在网上搜索了一下找到一篇文章:
我遇到的情况:
      在编写Chem.NET的时候要保存图片,我先用openFileDialog打开图片文件,然后用saveFileDialog保存文件时就出现了 “GDI
+中发生一般性错误”,我当时就想到是打开的文件还没有释放出来,于是用openFileDialog1.Dispose()来释放,可是没有成功。同样从一个MemorySream 实例打开一个Image后,立即关闭了这个流,结果在Image.Save时也会发生这种错误。我“摆渡”了很久都是遇到和我一样问题的人,CSDN上面的同志也没有给出一个实用的答案。最后终于还是在微软的网站上找到了答案:(以下是官方解决办法)

症状
Bitmap 对象或一个 图像 对象从一个文件, 构造时该文件仍保留锁定对于对象的生存期。 因此, 无法更改图像并将其保存回它产生相同的文件。

替代方法
•    创建非索引映像。
•    创建索引映像。
这两种情况下, 原始 位图 上调用 Bitmap.Dispose() 方法删除该文件上锁或删除要求, 流或内存保持活动。

创建非索引图像
即使原始映像被索引格式中该方法要求新图像位于每像素 (超过 
8 位 --, 非索引像素格式。 此变通方法使用 Graphics.DrawImage() 方法来将映像复制到新 位图 对象:
1.    构造从流、 从内存, 或从文件原始 位图 。
2.    创建新 位图 的相同大小, 带有是超过 8 位 - - 像素 (BPP) 每像素格式。
3.    使用 Graphics.FromImage() 方法以获取有关二 位图 Graphics 对象。
4.    用于 Graphics.DrawImage() 绘制首 位图 到二 位图 。
5.    用于 Graphics.Dispose() 处置是 图形 。
6.    用于 Bitmap.Dispose() 是首 位图 处置。

创建索引映像
此解决办法在索引格式创建一个 Bitmap 对象:
1.    构造从流、 从内存, 或从文件原始 位图 。
2.    创建新 位图 具有相同的大小和像素格式作为首 位图 。
3.    使用 Bitmap.LockBits() 方法来锁定整个图像对于两 Bitmap 对象以其本机像素格式。
4.    使用 Marshal.Copy 函数或其他内存复制函数来从首 位图 复制到二 位图 图像位。
5.    使用 Bitmap.UnlockBits() 方法可以解锁两 Bitmap 对象。
6.    用于 Bitmap.Dispose() 是首 位图 处置。
由于外国人的思维和我们不一样,我重新用实例解释一下,我这里使用的是创建非索引图像。

private void ToolStripMenuItem_Click(object sender, EventArgs e)
        
{
            
if (openFileDialog1.ShowDialog() == DialogResult.OK)
            
{
                
//创建一个bitmap类型的bmp变量来读取文件。
                Bitmap bmp = new Bitmap(openFileDialog1 .FileName );
                
//新建第二个bitmap类型的bmp2变量,我这里是根据我的程序需要设置的。
                Bitmap bmp2 = new Bitmap(1024768, PixelFormat.Format16bppRgb555);
                
//将第一个bmp拷贝到bmp2中
                Graphics draw = Graphics.FromImage(bmp2);
                draw.DrawImage(bmp,
0,0);
                pictureBox1.Image 
= (Image)bmp2 ;//读取bmp2到picturebox
                FILE = openFileDialog1.FileName;
                openFileDialog1.Dispose();
                draw.Dispose();
                bmp.Dispose();
//释放bmp文件资源
            }


使用它的方法并没有解决我的问题,因为这里他使用的是创建非索引图像,按文章介绍还有另外一种做法,于是我将显示图片的地方做了一些修改,如下:
private void ShowImage(PictureBox pic,Image image)
        
{
            
if (image.Width <= pic.Width && image.Height <= pic.Height)
            
{
                pic.SizeMode 
= PictureBoxSizeMode.CenterImage;
            }

            
else
            
{
                pic.SizeMode 
= PictureBoxSizeMode.StretchImage;
            }

            Bitmap bmp 
= (Bitmap)image;
            Bitmap bmp2 
= new Bitmap(bmp.Width, bmp.Height, image.PixelFormat);
            Rectangle rect 
= new Rectangle(00, bmp.Width, bmp.Height);
            System.Drawing.Imaging.BitmapData bmpData 
=
                bmp.LockBits(rect, System.Drawing.Imaging.ImageLockMode.ReadWrite,
                bmp.PixelFormat);
            IntPtr ptr 
= bmpData.Scan0;
            
int bytes = bmpData.Stride * bmp.Height;
            
byte[] rgbValues = new byte[bytes];
            System.Runtime.InteropServices.Marshal.Copy(ptr, rgbValues, 
0, bytes);
            bmp.UnlockBits(bmpData);
            bmpData 
=
                bmp2.LockBits(rect, System.Drawing.Imaging.ImageLockMode.ReadWrite,
                bmp2.PixelFormat);
            ptr 
= bmpData.Scan0;
            System.Runtime.InteropServices.Marshal.Copy(rgbValues, 
0, ptr, bytes);
            bmp2.UnlockBits(bmpData);
            pic.Image 
= bmp2;
        }

将返回字节数组的函数做一些修改:
        public static byte[] GetImageArray(System.Drawing.Image image)
        
{
            System.IO.MemoryStream ms 
= new MemoryStream();
                foreach (ImageCodecInfo info in ImageCodecInfo.GetImageEncoders())
                {
                    if (info.FormatID.Equals(image.RawFormat.Guid))
                    {
                        System.Diagnostics.Debug.WriteLine(info.MimeType);
                    }
                }
                //这里不能使用image.RawFormat了,因为上面的循环根本就找不到使用上面代码生成的Image的解码器,我理解是因为是拷背生成的位图,压根就不是文件的格式
                image.Save(ms, ImageFormat.Jpeg);

            
byte[] b=ms.ToArray();
            ms.Close();
            
return b;
        }
经过以上的代码,是可以正常的执行了,但是还有一个问题就是这里存储的格式硬性的变成了ImageFormat.Jpeg了.暂时还没有找到解决办法.
posted @ 2008-03-28 09:10  吴东雷  阅读(1253)  评论(1编辑  收藏  举报