openContactPhotoInputStream导致"java.lang.IollegalStateEx



背景

我正在尝试检查是否存在特定联系人的联系人图像(稍后才能真正获得,要晚得多)。查询应该尽可能少,并避免不需要的内存使用。

我已经发现这可能是不可能使用正常的方式,所以现在我使用:

final Uri lookupUri = Uri.withAppendedPath(ContactsContract.Contacts.CONTENT_LOOKUP_URI, contactKey);
final Cursor contactCur = mContentResolver.query(lookupUri, new String[] { ContactsContract.Contacts._ID },
null, null, null);
if (contactCur == null)
return false;
contactCur.moveToFirst();
long contactId = 0;
if (contactCur.getCount() != 0)
contactId = contactCur.getLong(contactCur.getColumnIndex(ContactsContract.Contacts._ID));
contactCur.close();
if (contactId == 0)
return false;
final Uri contactUri = ContentUris.withAppendedId(Contacts.CONTENT_URI, contactId);
final InputStream inputStream;
if (VERSION.SDK_INT >= VERSION_CODES.ICE_CREAM_SANDWICH)
inputStream = ContactsContract.Contacts.openContactPhotoInputStream(mContext.getContentResolver(),
contactUri, bigPicture);
else
inputStream = ContactsContract.Contacts.openContactPhotoInputStream(mContext.getContentResolver(),
contactUri);
if (inputStream != null) {
IOUtils.closeQuietly(inputStream);
return true;
}
return false;

问题

似乎在一些罕见的情况下,设备会给我返回一个例外:

java.lang.IollegalStateException:从第0行第0列获取字段槽在android.database.CursorWindow.getBlob_native(native Method)上失败在android.database.CursorWindow.getBlob(CursorWindow.java:288)android.database.AbstractWindowedCursor.getBlob(AbstractWindowedCursor.java:35)在android.database.CursorWrapper.getBlob(CursorWrapper.java:143)android.provider.ContactsContract$Contacts.openContactPhotoInputStream(ContactsContrct.java:1174)

事实证明,有很多关于这个问题的报告,但没有关于"openContactPhotoInputStream"功能的报告。

根据我所读到的内容(例如此处),这是由于需要读取的blob太大而导致的,但在这种情况下,我在Android的代码中遇到了这个问题。。。

问题

克服这种情况的最佳方法是什么?

这可能是安卓系统上的一个bug吗?

我的猜测是,即使我使用try-catch,当我稍后尝试获取图像时,我也会得到同样的错误,对吧?

我该怎么解决这个问题?

检查照片是否存在(未经测试):

ContentResolver cr = mContext.getContentResolver();
if (bigPicture && VERSION.SDK_INT >= VERSION_CODES.ICE_CREAM_SANDWICH) {
Uri displayPhotoUri = Uri.withAppendedPath(contactUri,
ContactsContract.Contacts.Photo.DISPLAY_PHOTO);
AssetFileDescriptor fd = null;
try {
fd = cr.openAssetFileDescriptor(displayPhotoUri, "r");
} catch (FileNotFoundException e) {}
if (fd != null) {
try {
fd.close();
} catch (IOException e) {
e.printStackTrace();
}
return true;
}
}
Uri photoUri = Uri.withAppendedPath(contactUri,
ContactsContract.Contacts.Photo.CONTENT_DIRECTORY);
Cursor cursor = cr.query(photoUri, new String[] {BaseColumns._ID},
ContactsContract.Contacts.Photo.PHOTO + " IS NOT NULL", null, null);
if (cursor == null) {
return false;
}
try {
return cursor.moveToFirst();
} finally {
cursor.close();
}

这里是openContactPhotoInputStream()方法的一个修改版本,它应该能够在没有问题的情况下加载大小接近1MB的照片缩略图(再次未测试):

public static InputStream openContactPhotoInputStream(ContentResolver cr, Uri contactUri,
boolean preferHighres) {
if (preferHighres && VERSION.SDK_INT >= VERSION_CODES.ICE_CREAM_SANDWICH) {
Uri displayPhotoUri = Uri.withAppendedPath(contactUri,
ContactsContract.Contacts.Photo.DISPLAY_PHOTO);
try {
return cr.openAssetFileDescriptor(displayPhotoUri, "r").createInputStream();
} catch (IOException e) {}
}
Uri photoUri = Uri.withAppendedPath(contactUri,
ContactsContract.Contacts.Photo.CONTENT_DIRECTORY);
try {
return cr.openAssetFileDescriptor(displayPhotoUri, "r").createInputStream();
} catch (IOException e) {}
return null;
}

这将导致ContactsProvider将缩略图BLOB读取到共享内存,并发送指向它的AssetFileDescriptor,从中我们可以直接打开InputStream

相关内容

最新更新