[英]android getContentResolver().notifyChange() does not restart my loader
代碼:
首先是我的Uris
public static final String PACKAGE = "my.url.contentprovider";
public static final String TABLE_NAME = "NetworkTransaction";
public static final String AUTHORITY = PACKAGE + ".NetTransContentProvider";
public static final Uri BASE_URI = Uri.parse("content://"+AUTHORITY);
public static final Uri CONTENT_URI_ANY_OBSERVER = Uri.withAppendedPath(BASE_URI,TABLE_NAME+"/*");
public static final Uri CONTENT_URI_FIND_BY_ID = Uri.withAppendedPath(BASE_URI,TABLE_NAME+"/FIND/ID");
public static final Uri CONTENT_URI_INSERT_OR_REPLACE_BY_ID = Uri.withAppendedPath(BASE_URI,TABLE_NAME+"/INSERT/REPLACE/ID");
public static final Uri CONTENT_URI_INSERT_BY_ID = Uri.withAppendedPath(BASE_URI,TABLE_NAME+"/INSERT/ID");
和我的活動代碼加載器:
@Override
protected void onResume() {
super.onResume();
getSupportLoaderManager().restartLoader(NET_TRANS_LOADER_ID,mBundle,this).forceLoad();
}
@Override
public Loader<Cursor> onCreateLoader(int id, Bundle bundle) {
Uri uri = null;
CursorLoader cl=null;
switch (id) {
case NET_TRANS_LOADER_ID:
uri = NetTransContentProvider.CONTENT_URI_FIND_BY_ID;
cl = new CursorLoader(ChoosingUserNameActivity.this, uri, NetTransDbUtils.allColumns,
NetTransDbUtils.COLUMN_ID + " = ? ",
new String[]{String.valueOf(bundle.getLong(EXTRA_TRANSACTION_ID,-1))}, null);
break;
default:
break;
}
return cl;
}
@Override
public void onLoadFinished(Loader<Cursor> loader, Cursor cursor) {
final int loaderId = loader.getId();
switch (loaderId) {
case NET_TRANS_LOADER_ID:
if(mTransactionId != null){
NetTrans netTrans = NetTransDbUtils.cursorToNetTrans(cursor);
if(netTrans != null && netTrans.getStatus() != null
&& !netTrans.getStatus().equals(NetTrans.STATUS_PENDING)){
EventBus.getDefault().post(new NetTransMsg(true, mTransactionId, netTrans.getMessage()));
}
}
break;
default:
break;
}
}
並且在我調用的已啟動服務中的ExecutorService
上運行的runnable中
mContext.getContentResolver().insert(NetTransContentProvider.CONTENT_URI_INSERT_OR_REPLACE_BY_ID, cv );
插入的值但加載器不會調用:
@Override
public Uri insert(Uri uri, ContentValues values) {
int uriType = sUriMatcher.match(uri);
switch (uriType) {
case INSERT_OR_REPLACE_BY_ID:
mDatabase.insertWithOnConflict(TABLE_NAME, null, values, SQLiteDatabase.CONFLICT_REPLACE);
break;
case INSERT_BY_ID:
mDatabase.insert(TABLE_NAME, null, values);
break;
default:
break;
}
getContext().getContentResolver().notifyChange(CONTENT_URI_ANY_OBSERVER, null);
return null;
}
@Override
public Cursor query(Uri uri, String[] projection, String selection, String[] selectionArgs,String sortOrder) {
Cursor cursor = mDatabase.query(TABLE_NAME,projection,selection, selectionArgs, null, null, null);
cursor.setNotificationUri(getContext().getContentResolver(), CONTENT_URI_ANY_OBSERVER);
return cursor;
}
我的問題是getContext().getContentResolver().notifyChange(CONTENT_URI_ANY_OBSERVER, null);
在insert方法中無法使我的loader重啟。
UPDATE
我創建了一個示例項目 ,按下按鈕,創建一個新的NetTrans
對象並寫入數據庫,然后該線程休眠5000毫秒並覆蓋該值(以模擬網絡操作)。 但是那個裝載機沒有重新啟動之后。 我的蟲子在哪里?
如果您想對注冊的觀察者CONTENT_URI_ANY_OBSERVER
當變化發生在被通知CONTENT_URI_FIND_BY_ID
,你需要確保的兩件事情。
首先, CONTENT_URI_ANY_OBSERVER
需要是CONTENT_URI_FIND_BY_ID
的父級。 如果您將其視為文件系統上的文件夾,則“CONTENT_URI_ANY_OBSERVER”應在其子文件夾中包含“CONTENT_URI_FIND_BY_ID”。
其次,在注冊內容觀察者時,必須為notifyDescendants
參數傳遞true。
Android嘗試查找匹配的內容觀察者時沒有外卡考慮因素(注釋中提供的鏈接僅適用於UriMatcher)。 因此,要解決您的問題,您應該從CONTENT_URI_ANY_OBSERVER
刪除/ *,它應該開始匹配。 你可以看到my.url.contentprovider/NetworkTransaction
現在是my.url.contentprovider/NetworkTransaction/INSERT/REPLACE/ID
的父“文件夾”, my.url.contentprovider/NetworkTransaction/INSERT/REPLACE/ID
你之前有my.url.contentprovider/NetworkTransaction/*
。
在審核了您的示例項目后,我找到了您的問題所在的另一個區域。 使用游標加載器時,游標由加載器擁有 。 這意味着除了迭代數據之外,你不應該改變它。 在NetTransDbUtils.cursorToNetTrans(cursor)
方法中,您將關閉光標,這將阻止CursorLoader有效監視光標數據的更改。
簡單回答:不要在NetTransDbUtils.cursorToNetTrans(cursor);
調用close cursor.close()
NetTransDbUtils.cursorToNetTrans(cursor);
對於這個用例。
用簡單的解析替換你的Uris:
public static final Uri CONTENT_URI_ANY_OBSERVER = Uri
.parse(BASE_URI + "/" + TABLE_NAME);
public static final Uri CONTENT_URI_FIND_BY_ID = Uri
.parse(BASE_URI + "/" + TABLE_NAME + "/FIND/ID");
public static final Uri CONTENT_URI_INSERT_OR_REPLACE_BY_ID = Uri
.parse(BASE_URI + "/" + TABLE_NAME + "/INSERT/REPLACE/ID");
public static final Uri CONTENT_URI_INSERT_BY_ID = Uri
.parse(BASE_URI + "/" + TABLE_NAME + "/INSERT/ID");
在RestService類中,使用處理程序進行回調:
private Handler mHandler;
@Override
public void onCreate() {
...
mHandler = new Handler();
}
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
...
mHandler.post(task);
...
}
聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.