最近写个本地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配置其大小,所以 要么改变别人 要么改变自己。。乖乖压缩下图片存储。终于正常了。
网上随手搜了个图片压缩类,改了改。
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;
}
}
goods.goods_picture = ImageCompresser.compressImage(bitmap,180,90,200);
先进行像素尺寸压缩,在进行磁盘质量压缩!完美。
转载请注明:天狐博客 » Android开发之图片写入到Sqlite读取崩溃