簡體   English   中英

android getContentResolver()。notifyChange()不重啟我的加載器

[英]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/*

編輯1

在審核了您的示例項目后,我找到了您的問題所在的另一個區域。 使用游標加載器時,游標由加載器擁有 這意味着除了迭代數據之外,你不應該改變它。 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.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM