有一些代码:
struct BitmapDataAccessor
{
private readonly byte[] data;
private readonly int[] rowStarts;
public readonly int Height;
public readonly int Width;
public readonly int Padding;
public BitmapDataAccessor(byte[] data, int width, int height, int padding)
{
this.data = data;
this.Height = height;
this.Width = width + (4-padding)%4;
this.Padding = padding;
rowStarts = new int[Height];
for (int y = 0; y < Height; y++)
rowStarts[y] = y * Width;
}
public byte this[int x, int y, int color] // Maybe use an enum with Red = 0, Green = 1, and Blue = 2 members?
{
get { return data[(rowStarts[y] + x) * 3 + color]; }
set { data[(rowStarts[y] + x) * 3 + color] = value; }
}
public byte[] Data
{
get { return data; }
}
}
public static byte[, ,] Bitmap2Byte(Bitmap obraz)
{
int h = obraz.Height;
int w = obraz.Width;
byte[, ,] wynik = new byte[w, h, 3];
BitmapData bd = obraz.LockBits(new Rectangle(0, 0, w, h), ImageLockMode.ReadOnly, PixelFormat.Format24bppRgb);
int bytes = Math.Abs(bd.Stride) * h;
byte[] rgbValues = new byte[bytes];
IntPtr ptr = bd.Scan0;
System.Runtime.InteropServices.Marshal.Copy(ptr, rgbValues, 0, bytes);
int padding = Math.Abs(bd.Stride) - (((w * 24) + 7) / 8);
BitmapDataAccessor bda = new BitmapDataAccessor(rgbValues, w, h, padding);
for (int i = 0; i < h; i++)
{
for (int j = 0; j < w; j++)
{
wynik[j, i, 0] = bda[j, i, 2];
wynik[j, i, 1] = bda[j, i, 1];
wynik[j, i, 2] = bda[j, i, 0];
}
}
obraz.UnlockBits(bd);
return wynik;
}
这是我跑的测试:
private void btnLoad_Click(object sender, EventArgs e)
{
string s = "";
for (int i = 400; i < 409; i++)
{
Bitmap bmp = new Bitmap(i, i);
bool ok = true;
try
{
byte[,,] tab = Grafika.Bitmap2Byte(bmp);
}
catch
{
ok = false;
}
s += i + ": ";
if (ok) s += "OK";
else s += "NOT OK";
s += "\n";
}
StreamWriter w = StreamWriter(@"C:\lalala.txt");
w.Write(s);
w.Close();
return;
}
输出:
400: OK
401: NOT OK
402: NOT OK
403: OK
404: OK
405: NOT OK
406: NOT OK
407: OK
408: OK
为什么我会获得w % 4 ==1
或w % 4 == 2
的例外情况?
修改
我知道异常是什么。这是“索引越界”,它发生在行中
get { return data[(rowStarts[y] + x) * 3 + color]; }
。
不幸的是,我不知道如何避免这种情况,为什么它只发生在w % 4 ==1
或w % 4 == 2
。这就是我发布问题的原因。我不需要帮助识别异常或处理异常。我只需要修复BitmapDataAccessor
和/或Bitmap2Byte
即可使其正常工作。
编辑2:
感谢max的回答Bitmap2Byte
不会抛出异常。但是这使我的另一个函数Byte2Bitmap
抛出它们:
public static Bitmap Byte2Bitmap(byte[, ,] tablica)
{
if (tablica.GetLength(2) != 3)
{
throw new NieprawidlowyWymiarTablicyException();
}
int w = tablica.GetLength(0);
int h = tablica.GetLength(1);
int padding = w % 4;
int ww=w;
if (padding != 0)
ww += 4 - padding;
int bytes = 3 * ww * h;
byte[] rgbValues = new byte[bytes];
int counter = -3;
for (int j = 0; j < h; j++)
{
for (int i = 0; i < w; i++)
{
counter += 3;
rgbValues[counter] = tablica[i, j, 2];
rgbValues[counter + 1] = tablica[i, j, 1];
rgbValues[counter + 2] = tablica[i, j, 0];
}
if(padding!=0)
counter+= padding;
}
Bitmap obraz = new Bitmap(w, h, PixelFormat.Format24bppRgb);
BitmapData bd = obraz.LockBits(new Rectangle(0, 0, w, h), ImageLockMode.WriteOnly, PixelFormat.Format24bppRgb);
IntPtr ptr = bd.Scan0;
System.Runtime.InteropServices.Marshal.Copy(rgbValues, 0, ptr, bytes);
obraz.UnlockBits(bd);
return obraz;
}
我的输出显示哪个宽度有例外:
380: OK
381: NOT OK: Nastąpiła próba odczytu lub zapisu pamięci chronionej. Często wskazuje to, że inna pamięć jest uszkodzona.
382: OK
383: OK
384: OK
385: NOT OK: Nastąpiła próba odczytu lub zapisu pamięci chronionej. Często wskazuje to, że inna pamięć jest uszkodzona.
386: OK
387: OK
388: OK
389: NOT OK: Nastąpiła próba odczytu lub zapisu pamięci chronionej. Często wskazuje to, że inna pamięć jest uszkodzona.
390: OK
391: OK
392: OK
393: OK
394: NOT OK: Nastąpiła próba odczytu lub zapisu pamięci chronionej. Często wskazuje to, że inna pamięć jest uszkodzona.
395: OK
396: OK
397: NOT OK: Nastąpiła próba odczytu lub zapisu pamięci chronionej. Często wskazuje to, że inna pamięć jest uszkodzona.
398: OK
399: OK
400: OK
401: NOT OK: Nastąpiła próba odczytu lub zapisu pamięci chronionej. Często wskazuje to, że inna pamięć jest uszkodzona.
402: OK
403: OK
404: OK
405: NOT OK: Nastąpiła próba odczytu lub zapisu pamięci chronionej. Często wskazuje to, że inna pamięć jest uszkodzona.
406: NOT OK: Nastąpiła próba odczytu lub zapisu pamięci chronionej. Często wskazuje to, że inna pamięć jest uszkodzona.
407: OK
408: OK
409: NOT OK: Nastąpiła próba odczytu lub zapisu pamięci chronionej. Często wskazuje to, że inna pamięć jest uszkodzona.
410: OK
411: OK
412: OK
413: OK
414: NOT OK: Nastąpiła próba odczytu lub zapisu pamięci chronionej. Często wskazuje to, że inna pamięć jest uszkodzona.
415: OK
416: OK
417: NOT OK: Nastąpiła próba odczytu lub zapisu pamięci chronionej. Często wskazuje to, że inna pamięć jest uszkodzona.
418: OK
419: OK
420: OK
421: NOT OK: Nastąpiła próba odczytu lub zapisu pamięci chronionej. Często wskazuje to, że inna pamięć jest uszkodzona.
422: NOT OK: Nastąpiła próba odczytu lub zapisu pamięci chronionej. Często wskazuje to, że inna pamięć jest uszkodzona.
423: OK
424: OK
425: NOT OK: Nastąpiła próba odczytu lub zapisu pamięci chronionej. Często wskazuje to, że inna pamięć jest uszkodzona.
426: OK
427: OK
428: OK
行中发生异常:
System.Runtime.InteropServices.Marshal.Copy(rgbValues, 0, ptr, bytes);
及其文字表示:There was try to read or write to/from protected memory...
我认为这是因为bytes
未正确计算。我怎么能这样做?
我不知道如何用英语获取Exception消息。对不起。
答案 0 :(得分:3)
更有用的不是简单地收集'ok'或'not ok',而是实际检查抛出的异常,这将为您提供更多关于实际上不正常的信息。
尝试添加
Console.WriteLine(ex.ToString());
到catch
区块。
答案 1 :(得分:3)
尝试将填充计算为
int padding = Math.Abs(bd.Stride) - (w * 3);
(至少它更具可读性)
并修复BitmapDataAccessor
:
public BitmapDataAccessor(byte[] data, int width, int height, int padding)
{
this.data = data;
this.Height = height;
// width in bytes, not pixels
this.Width = width * 3 + padding;
this.Padding = padding;
rowStarts = new int[Height];
for(int y = 0; y < Height; y++)
rowStarts[y] = y * Width;
}
public byte this[int x, int y, int color]
{
// row offsets already in bytes, so don't myltiply them by 3
get { return data[rowStarts[y] + x * 3 + color]; }
set { data[rowStarts[y] + x * 3 + color] = value; }
}
关于第二个问题的说明:
此处类似的问题 - 您计算像素的填充,而不是字节。
int ww = w * 3;
int padding = (4 - (ww % 4)) % 4;
int bytes = (ww + padding) * h;
byte[] rgbValues = new byte[bytes];
int rowOffset = 0;
for (int j = 0; j < h; j++)
{
int pixelOffset = rowOffset;
for (int i = 0; i < w; i++)
{
rgbValues[pixelOffset + 0] = tablica[i, j, 2];
rgbValues[pixelOffset + 1] = tablica[i, j, 1];
rgbValues[pixelOffset + 2] = tablica[i, j, 0];
pixelOffset += 3;
}
rowOffset += ww + padding;
}
答案 2 :(得分:1)
填充不是数字或像素。这是字节数 但是,我仍然不知道如何修复你的代码。也许其他人可以做到这一点?