我无法解释这一点(来自Eclipse调试的截图):
(0,0)处的像素没有为其设置的值!
所有其他像素都很好,它们具有相同的值。
修改 我做了一些挖掘,setPixel的代码调用了一个本机函数:
1391 public void setPixel(int x, int y, int color) {
1392 checkRecycled("Can't call setPixel() on a recycled bitmap");
1393 if (!isMutable()) {
1394 throw new IllegalStateException();
1395 }
1396 checkPixelAccess(x, y);
1397 nativeSetPixel(mNativeBitmap, x, y, color, mIsPremultiplied);
1398 }
所以我用不同的值运行setPixel(0,0),从setPixel(0,1)的OK值开始。
首先,我只改变了其中一个论点。结论是阿尔法值是罪魁祸首,但不确定如何。我尝试了很多其他的alpha值,看起来大约在0xB0左右,结果又回来了
我在其他像素上尝试了相同的值,问题不依赖于像素坐标,对其他像素也是如此。这个问题似乎与数据有关。
source.setPixel(0, 0, Color.argb(0x40, 0x1A, 0x11, 0x12));
int sp00 = source.getPixel(0, 0); // sp00 1075580948 [0x401c1014] BAD
source.setPixel(0, 0, Color.argb(0xFE, 0xFF, 0xEE, 0xDE));
sp00 = source.getPixel(0, 0); // sp00 -16781602 [0xfeffeede] OK!
source.setPixel(0, 0, Color.argb(0x40, 0xFF, 0xEE, 0xDE));
sp00 = source.getPixel(0, 0); // sp00 1090514911 [0x40ffefdf] BAD
source.setPixel(0, 0, Color.argb(0xFE, 0x1A, 0xEE, 0xDE));
sp00 = source.getPixel(0, 0); // sp00 -31789346 [0xfe1aeede] OK!
source.setPixel(0, 0, Color.argb(0xFE, 0xFF, 0x11, 0xDE));
sp00 = source.getPixel(0, 0); // sp00 -16838178 [0xfeff11de] OK!
source.setPixel(0, 0, Color.argb(0xFE, 0xFF, 0xEE, 0x12));
sp00 = source.getPixel(0, 0); // sp00 -16781806 [0xfeffee12] OK!
source.setPixel(0, 0, Color.argb(0x00, 0xFF, 0xEE, 0xDE));
sp00 = source.getPixel(0, 0); // sp00 0 [0x0] Strange, why Color has to police the alpha value?
source.setPixel(0, 0, Color.argb(0x10, 0xFF, 0xEE, 0xDE));
sp00 = source.getPixel(0, 0); // sp00 285208543 [0x10ffefdf] BAD
source.setPixel(0, 0, Color.argb(0x20, 0xFF, 0xEE, 0xDE));
sp00 = source.getPixel(0, 0); // sp00 553643999 [0x20ffefdf] BAD
source.setPixel(0, 0, Color.argb(0x30, 0xFF, 0xEE, 0xDE));
sp00 = source.getPixel(0, 0); // sp00 822079455 [0x30ffefdf] BAD
source.setPixel(0, 0, Color.argb(0x50, 0xFF, 0xEE, 0xDE));
sp00 = source.getPixel(0, 0); // sp00 1358950367 [0x50ffefdf] BAD
source.setPixel(0, 0, Color.argb(0x60, 0xFF, 0xEE, 0xDE));
sp00 = source.getPixel(0, 0); // sp00 1627385823 [0x60ffefdf] BAD
source.setPixel(0, 0, Color.argb(0x70, 0xFF, 0xEE, 0xDE));
sp00 = source.getPixel(0, 0); // sp00 1895821279 [0x70ffefdf] BAD
source.setPixel(0, 0, Color.argb(0x80, 0xFF, 0xEE, 0xDE));
sp00 = source.getPixel(0, 0); // sp00 -2130711075 [0x80ffeddd] BAD but change of pattern
source.setPixel(0, 0, Color.argb(0x90, 0xFF, 0xEE, 0xDE));
sp00 = source.getPixel(0, 0); // sp00 -1862275619 [0x90ffeddd] BAD
source.setPixel(0, 0, Color.argb(0xA0, 0xFF, 0xEE, 0xDE));
sp00 = source.getPixel(0, 0); // sp00 -1593840162 [0xa0ffedde] BAD but change of pattern again
source.setPixel(0, 0, Color.argb(0xB0, 0xFF, 0xEE, 0xDE));
sp00 = source.getPixel(0, 0); // sp00 -1325404450 [0xb0ffeede] OK!
source.setPixel(0, 0, Color.argb(0xC0, 0xFF, 0xEE, 0xDE));
sp00 = source.getPixel(0, 0); // sp00 -1056968994 [0xc0ffeede] OK!
source.setPixel(0, 0, Color.argb(0xD0, 0xFF, 0xEE, 0xDE));
sp00 = source.getPixel(0, 0); // sp00 -788533538 [0xd0ffeede] OK!
source.setPixel(0, 0, Color.argb(0xE0, 0xFF, 0xEE, 0xDE));
sp00 = source.getPixel(0, 0); // sp00 -520098082 [0xe0ffeede] OK!
source.setPixel(0, 0, Color.argb(0xF0, 0xFF, 0xEE, 0xDE));
sp00 = source.getPixel(0, 0); // sp00 -251662626 [0xf0ffeede] OK!
source.setPixel(0, 0, Color.argb(0xA7, 0xFF, 0xEE, 0xDE));
sp00 = source.getPixel(0, 0); // sp00 -1476399395 [0xa7ffeedd] BAD
source.setPixel(0, 0, Color.argb(0xA3, 0xFF, 0xEE, 0xDE));
sp00 = source.getPixel(0, 0); // sp00 -1543508258 [0xa3ffeede] OK!
source.setPixel(0, 0, Color.argb(0xA1, 0xFF, 0xEE, 0xDE));
sp00 = source.getPixel(0, 0); // sp00 -1577062690 [0xa1ffeede] OK!
source.setPixel(0, 0, Color.argb(0xAB, 0xFF, 0xEE, 0xDE));
sp00 = source.getPixel(0, 0); // sp00 -1409290274 [0xabffefde] BAD
source.setPixel(0, 0, Color.argb(0xA9, 0xFF, 0xEE, 0xDE));
sp00 = source.getPixel(0, 0); // sp00 -1442844962 [0xa9ffeede] OK!
答案 0 :(得分:2)
嗯,或多或少我发现了正在发生的事情
在api19没有提供任何控制此功能的方法之前,它被称为“预乘Alpha”和位图,默认情况下它们是预乘的,无法更改。
在api19上,Bitmap
有两种新方法:isPremultiplied()和setPremultiplied(boolean)
根据新的文件:
当像素预乘时,RGB分量已经过 乘以alpha分量。例如,如果是原始颜色 是50%半透明红色(128,255,0,0),预乘形式是 (128,128,0,0)。
this post和this other post也提供了更多解释。据此,还有一些测试显示:
Bitmap source = Bitmap.createBitmap(2, 2, Config.ARGB_8888);
source.setPixel(0, 0, Color.argb(0x02, 0x10, 0x20, 0x30));
source.setPixel(0, 1, Color.argb(0x03, 0x10, 0x20, 0x30));
source.setPixel(1, 0, Color.argb(0x05, 0x78, 0x96, 0x64));
source.setPixel(1, 1, Color.argb(128, 255, 200, 150));
int sp00 = source.getPixel(0, 0); // sp00 33554432 [0x2000000]
int sp01 = source.getPixel(0, 1); // sp01 50331733 [0x3000055]
int sp10 = source.getPixel(1, 0); // sp10 90610022 [0x5669966]
int sp11 = source.getPixel(1, 1); // sp11 -2130720875 [0x80ffc795]
对于较低的颜色值,舍入会导致信息丢失(请参阅上面的sp00) 此外,对于较低的值,alpha本身的值不会被检索到原始值 除此之外,给定的公式不解释我看到的值。
最后要使用未修改的像素,我现在使用此代码设置位图像素:
Bitmap source = Bitmap.createBitmap(2, 2, Config.ARGB_8888);
IntBuffer data = IntBuffer.wrap(new int[] {
Color.argb(0x06, 0xff, 0xa0, 0x8d),
Color.argb(0x2a, 0xab, 0xce, 0x9f),
Color.argb(0x8f, 0xfe, 0x05, 0x18),
Color.argb(0xff, 0xc8, 0xcf, 0xd4)
});
source.copyPixelsFromBuffer(data);
并检索我使用的像素:
IntBuffer sourceData = IntBuffer.allocate(4);
source.copyPixelsToBuffer(sourceData);
使用这些方法不会预先计算颜色。