最近写个本地App blob,需要把图片放到android的sqlite中就行读写。测试发现,有的图可以正常保存显示,有的图保存后,select会导致app崩溃。崩溃原因为:
java.lang.IllegalStateException: Couldn't read row 0, col 0 from CursorWindow. Make sure the Cursor is initialized correctly before accessing data from it.
谷歌了很多,发现导致导致Cursor的原因就几个,列不存在,Cursor错误,CursorWindow size问题。
研究一番发现,Android SQLite CursorWindow指定了一个最大为2MB的config_cursorWindowSize值。
在Android源码中config_cursorWindowSize值定义如下:
sdk\platforms\android-23\data\res\values\configs.xml:
<!-- When a database query is executed, the results retuned are paginated in pages of size (in KB) indicated by this value -->
<integer name="config_cursorWindowSize">2048</integer>
显然这个值是写死的,不能通过app配置其大小,所以 要么改变别人 要么改变自己。。乖乖压缩下图片存储。终于正常了。
网上随手搜了个图片压缩类,改了改。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 |
import android.graphics.Bitmap; import android.graphics.BitmapFactory; import java.io.ByteArrayInputStream; import java.io.ByteArrayOutputStream; public class ImageCompresser { /** * 压缩图片到指定宽高,并进行质量压缩,最终大小保持在maxDiskSize K以下 * * @param sourceBm * @param targetWidth * @param targetHeight * @return */ public static Bitmap compressImage(Bitmap sourceBm, float targetWidth, float targetHeight,float maxDiskSize) { BitmapFactory.Options newOpts = new BitmapFactory.Options(); // 开始读入图片,此时把options.inJustDecodeBounds 设回true了 newOpts.inJustDecodeBounds = true; // 可删除 newOpts.inPurgeable = true; // 可共享 newOpts.inInputShareable = true; // 转成数组 ByteArrayOutputStream baos = new ByteArrayOutputStream(); sourceBm.compress(Bitmap.CompressFormat.JPEG, 100, baos); byte[] temp = baos.toByteArray(); // 此时返回bm为空 Bitmap bitmap = BitmapFactory.decodeByteArray(temp, 0, temp.length, newOpts); newOpts.inJustDecodeBounds = false; int w = newOpts.outWidth; int h = newOpts.outHeight; // 现在主流手机比较多是800*480分辨率,所以高和宽我们设置为 float hh = targetHeight; float ww = targetWidth; // 缩放比。由于是固定比例缩放,只用高或者宽其中一个数据进行计算即可 int be = 1;// be=1表示不缩放 // 如果宽度大的话根据宽度固定大小缩放 if (w > h && w > ww) { be = (int) (newOpts.outWidth / ww); } else if (w < h && h > hh) { // 如果高度高的话根据宽度固定大小缩放 be = (int) (newOpts.outHeight / hh); } if (be <= 0) { be = 1; } // 设置缩放比例 newOpts.inSampleSize = be; // 重新读入图片,注意此时已经把options.inJustDecodeBounds 设回false了 bitmap = BitmapFactory.decodeByteArray(temp, 0, temp.length, newOpts); // 压缩好比例大小后再进行质量压缩 return compressImageWitDisksize(bitmap,maxDiskSize); } /** * @Description 质量压缩方法 * @param image * @param maxDiskSize 最大kb尺寸 * @return */ public static Bitmap compressImageWitDisksize(Bitmap image,float maxDiskSize) { ByteArrayOutputStream baos = new ByteArrayOutputStream(); // 质量压缩方法,这里100表示不压缩,把压缩后的数据存放到baos中 image.compress(Bitmap.CompressFormat.JPEG, 100, baos); int options = 100; // 循环判断如果压缩后图片是否大于100kb,大于继续压缩 while (baos.toByteArray().length / 1024 > maxDiskSize) { // 重置baos即清空baos baos.reset(); // 这里压缩options%,把压缩后的数据存放到baos中 image.compress(Bitmap.CompressFormat.JPEG, options, baos); // 每次都减少10 options -= 10; } // 把压缩后的数据baos存放到ByteArrayInputStream中 ByteArrayInputStream isBm = new ByteArrayInputStream(baos.toByteArray()); // 把ByteArrayInputStream数据生成图片 Bitmap bitmap = BitmapFactory.decodeStream(isBm, null, null); return bitmap; } /** * 只进行分辨率压缩,不进行图片的质量压缩 * * @param sourceBm * @param targetWidth * @param targetHeight * @return */ public static Bitmap compressImageWithResolution(Bitmap sourceBm, float targetWidth, float targetHeight) { BitmapFactory.Options newOpts = new BitmapFactory.Options(); // 开始读入图片,此时把options.inJustDecodeBounds 设回true了 newOpts.inJustDecodeBounds = true; // 可删除 newOpts.inPurgeable = true; // 可共享 newOpts.inInputShareable = true; // 转成数组 ByteArrayOutputStream baos = new ByteArrayOutputStream(); sourceBm.compress(Bitmap.CompressFormat.JPEG, 100, baos); byte[] temp = baos.toByteArray(); // 此时返回bm为空 Bitmap bitmap = BitmapFactory.decodeByteArray(temp, 0, temp.length, newOpts); newOpts.inJustDecodeBounds = false; int w = newOpts.outWidth; int h = newOpts.outHeight; // 现在主流手机比较多是800*480分辨率,所以高和宽我们设置为 float hh = targetHeight; float ww = targetWidth; // 缩放比。由于是固定比例缩放,只用高或者宽其中一个数据进行计算即可 // be=1表示不缩放 int be = 1; if (w > h && w > ww) { // 如果宽度大的话根据宽度固定大小缩放 be = (int) (newOpts.outWidth / ww); } else if (w < h && h > hh) { // 如果高度高的话根据宽度固定大小缩放 be = (int) (newOpts.outHeight / hh); } if (be <= 0) { be = 1; } // 设置缩放比例 newOpts.inSampleSize = be; // 重新读入图片,注意此时已经把options.inJustDecodeBounds 设回false了 bitmap = BitmapFactory.decodeByteArray(temp, 0, temp.length, newOpts); // 压缩好比例大小后再进行质量压缩 return bitmap; } } |
1 |
goods.goods_picture = ImageCompresser.compressImage(bitmap,180,90,200); |
先进行像素尺寸压缩,在进行磁盘质量压缩!完美。
转载请注明:天狐博客 » Android开发之图片写入到Sqlite读取崩溃