为什么我一直得到例外?

时间:2011-01-13 13:32:20

标签: c# winforms image-processing exception-handling

有一些代码:

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 ==1w % 4 == 2的例外情况?

修改
我知道异常是什么。这是“索引越界”,它发生在行中 get { return data[(rowStarts[y] + x) * 3 + color]; }
不幸的是,我不知道如何避免这种情况,为什么它只发生在w % 4 ==1w % 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消息。对不起。

3 个答案:

答案 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)

填充不是数字或像素。这是字节数 但是,我仍然不知道如何修复你的代码。也许其他人可以做到这一点?