关于"gdi+ 发生一般性错误"的处理.
做一个将图片存入数据库的程序,写了一个将图片转为字节数组的函数,如下:
使用它的方法并没有解决我的问题,因为这里他使用的是创建非索引图像,按文章介绍还有另外一种做法,于是我将显示图片的地方做了一些修改,如下:
将返回字节数组的函数做一些修改:
经过以上的代码,是可以正常的执行了,但是还有一个问题就是这里存储的格式硬性的变成了ImageFormat.Jpeg了.暂时还没有找到解决办法.
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+ 发生一般性错误",在网上搜索了一下找到一篇文章:{
System.IO.MemoryStream ms = new MemoryStream();
image.Save(ms, Image.RawFormat);
byte[] b=ms.ToArray();
ms.Close();
return b;
}
我遇到的情况:
在编写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(1024, 768, 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文件资源
}
在编写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(1024, 768, 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(0, 0, 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;
}
{
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(0, 0, 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;
}
{
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;
}
一点说明:为什么在标题中要嵌入英文?原因是为了能够让国外的网友能查询到这篇文章。平常在Google上查资料的时候,经常参考国外网友的博客,帮助我解决了很多问题,所以我也想让他们能够参考我写的内容。当然文中我不可能全部译为英文,所以我尽量把代码粘全,靠代码说话吧。