2

次の問題があります。私は、どんな種類の変換でも問題なく動作するビットマップ変換ルーチンから始めました。

  Bitmap transform(Bitmap src) {
    // ... any kind of transformation , for example GAMMA
    double gama = 0.8;
    int[] tR = new int[256];
    int[] gG = new int[256];
    int[] tB = new int[256];
    for(int i = 0; i < 256; ++i) {
      tR[i] = (int)Math.min(255, (int)((255.0 * Math.pow(i / 255.0, 1.0 / gama)) + 0.5));
      tG[i] = (int)Math.min(255, (int)((255.0 * Math.pow(i / 255.0, 1.0 / gama)) + 0.5));
      tB[i] = (int)Math.min(255, (int)((255.0 * Math.pow(i / 255.0, 1.0 / gama)) + 0.5));
    }
    // apply transformation to the old bitmap -> bmOut  
    int wid = src.getWidth(), hei = src.getHeight();
    Bitmap bmOut = Bitmap.createBitmap(wid, hei, src.getConfig());
    int A, R, G, B;
    for(int x = 0; x < wid; x++) {
      for(int y = 0; y < hei; y++) {
        int pixel = src.getPixel(x, y);
        A = Color.alpha(pixel);
        R = tR[Color.red(pixel)];
        G = tG[Color.green(pixel)];
        B = tB[Color.blue(pixel)];
        bmOut.setPixel(x, y, Color.argb(A, R, G, B));
      }
    }
    return bmOut;
  }

しかし、それは非常に遅いです - getPixel() / setPixel() 兄弟、姉妹が原因です。問題ありません、と私は言います。メモリ バッファを使用するだけです (昔の StretchBlt() の時代のように)。それで、私は大規模な書き直しを行い、次のソフトウェアエンジニアリングの逸品を作成しました:)

  Bitmap transform(Bitmap src) {
    // ... transformation array are built here

    // apply transformation  
    int wid = src.getWidth(), hei = src.getHeight();
    Bitmap bmOut = Bitmap.createBitmap(wid, hei, src.getConfig());

    int[] pixs = new int[wid*hei];                  // changed
    src.getPixels(pixs, 0, wid, 0, 0, wid, hei);    // changed

    int A, R, G, B;
    for(int x = 0; x < wid; x++) {
      for(int y = 0; y < hei; y++) {
        int off = ( x * y ) + y;                    // changed
        int pixel = pixs[off];                      // changed
        A = Color.alpha(pixel);
        R = tR[Color.red(pixel)];
        G = tG[Color.green(pixel)];
        B = tB[Color.blue(pixel)];
        pixs[off] = Color.argb(A, R, G, B);         // changed      
      }
    }
    bmOut.setPixels(pixs, 0, wid, 0, 0, wid, hei);  // changed
    return bmOut;
  }

高速に実行され、変換がない場合でも正しい結果が得られます。しかし、ピクセルをマッサージしようとすると(変換を適用すると)バラバラになります。したがって、getPixel() からの ARGB ピクセルと getPixel(...) からのピクセル値の配列を比較しましたが、それらは異なります (最初の 2 つは同じであり、そうでないものは無数に残ります)。

array     getPixel
a r g b   a r g b
------------------
ff65340b  ff65340b
ff64330a  ff64330a
ff66320b  ff63320a
ff65310a  ff613008
ff66300c  ff62300d
ff67310d  ff62300d
ff68300d  ff622d0d
ff69310e  ff5f2a0a
....

今回、私が何を間違っているか知っている人はいますか?mem-array ソリューションの速度をまだ諦めたくありません。ありがとう、ショーン

4

1 に答える 1

1

そのはず

int off = ( y * wid ) + x;

ちなみに、2 つのループは不要だと思います。次のように簡単に実行できます。

for (int off = pixs.length - 1; off >= 0; off--) 
于 2013-03-16T06:00:54.847 に答える