1

У меня есть ContentObserver onChange(), объявленный как подкласс в моей деятельности. Но он всегда возвращает false. Может ли кто-нибудь сказать мне, почему?Метод onChange() Android возвращает false

(Обновить) Этот код должен вызывать fillList, если поставщик содержимого CallLog изменяется. Я имею в виду, если я сделаю новый вызов, поэтому данные вызова будут вставлены в контент-провайдер, поэтому он должен вернуться к наблюдателю, чтобы что-то там изменилось, поэтому он будет называть fillList(), но он всегда возвращается false, даже если я сделаю новый вызов на эмуляторе.

Вот код.

public class RatedCalls extends ListActivity { 

private static final String LOG_TAG = "RatedCallsObserver"; 
private Handler handler = new Handler(); 
private RatedCallsContentObserver callsObserver = null; 
private SQLiteDatabase db; 
private CallDataHelper dh = null; 
StringBuilder sb = new StringBuilder(); 
OpenHelper openHelper = new OpenHelper(RatedCalls.this); 

class RatedCallsContentObserver extends ContentObserver { 
    public RatedCallsContentObserver(Handler h) { 
     super(h); 
    } 

    public void onChange(boolean selfChange) { 
     Log.d(LOG_TAG, "RatedCallsContentObserver.onChange(" + selfChange 
       + ")"); 

    } 
} 

@Override 
public void onCreate(Bundle savedInstanceState) { 
    super.onCreate(savedInstanceState); 
    registerContentObservers(); 
    fillList(); 
} 

@Override 
public void onStart() { 

    super.onStart(); 
    registerContentObservers(); 

} 

@Override 
public void onStop() { 

    super.onStop(); 
    unregisterContentObservers(); 

} 

private void fillList() { 

    Cursor cursor = getContentResolver().query(
      android.provider.CallLog.Calls.CONTENT_URI, null, null, null, 
      android.provider.CallLog.Calls.DATE + " DESC "); 

    cursor.setNotificationUri(getBaseContext().getContentResolver(), 
      android.provider.CallLog.Calls.CONTENT_URI); 

    dh = new CallDataHelper(this); 
    db = openHelper.getWritableDatabase(); 

    startManagingCursor(cursor); 
    int numberColumnId = cursor 
      .getColumnIndex(android.provider.CallLog.Calls.NUMBER); 
    int durationId = cursor 
      .getColumnIndex(android.provider.CallLog.Calls.DURATION); 
    int contactNameId = cursor 
      .getColumnIndex(android.provider.CallLog.Calls.CACHED_NAME); 
    int dateId = cursor.getColumnIndex(android.provider.CallLog.Calls.DATE); 
    int numTypeId = cursor 
      .getColumnIndex(android.provider.CallLog.Calls.CACHED_NUMBER_TYPE); 
    // int contactIdColumnId = 
    // cursor.getColumnIndex(android.provider.ContactsContract.RawContacts.CONTACT_ID); 

    Date dt = new Date(); 
    int hours = dt.getHours(); 
    int minutes = dt.getMinutes(); 
    int seconds = dt.getSeconds(); 
    String currTime = hours + ":" + minutes + ":" + seconds; 

    ArrayList<String> callList = new ArrayList<String>(); 
    if (cursor.moveToFirst()) { 

     do { 
      String contactNumber = cursor.getString(numberColumnId); 
      String contactName = cursor.getString(contactNameId); 
      String duration = cursor.getString(durationId); 
      String callDate = DateFormat.getDateInstance().format(dateId); 
      String numType = cursor.getString(numTypeId); 

      ContentValues values = new ContentValues(); 

      values.put("contact_id", 1); 
      values.put("contact_name", contactName); 
      values.put("number_type", numType); 
      values.put("contact_number", contactNumber); 
      values.put("duration", duration); 
      values.put("date", callDate); 
      values.put("current_time", currTime); 
      values.put("cont", 1); 

      getBaseContext().getContentResolver().notifyChange(
        android.provider.CallLog.Calls.CONTENT_URI, null); 

      callList.add("Contact Number: " + contactNumber 
        + "\nContact Name: " + contactName + "\nDuration: " 
        + duration + "\nDate: " + callDate); 
      this.db.insert(CallDataHelper.TABLE_NAME, null, values); 
      Toast.makeText(getBaseContext(), "Inserted!", Toast.LENGTH_LONG); 

     } while (cursor.moveToNext()); 
     setListAdapter(new ArrayAdapter<String>(this, R.layout.listitem, 
       callList)); 
     ListView lv = getListView(); 
     lv.setTextFilterEnabled(true); 

     lv.setOnItemClickListener(new android.widget.AdapterView.OnItemClickListener() { 

      @Override 
      public void onItemClick(AdapterView<?> parent, View view, 
        int position, long id) { 

       Toast.makeText(getApplicationContext(), 
         ((TextView) view).getText(), Toast.LENGTH_SHORT) 
         .show(); 

      } 
     }); 
    } 
} 

private void registerContentObservers() { 
    ContentResolver cr = getContentResolver(); 
    callsObserver = new RatedCallsContentObserver(handler); 
    cr.registerContentObserver(android.provider.CallLog.Calls.CONTENT_URI, 
      true, callsObserver); 
} 

private void unregisterContentObservers() { 

    ContentResolver cr = getContentResolver(); 
    if (callsObserver != null) { // just paranoia 
     cr.unregisterContentObserver(callsObserver); 
     callsObserver = null; 
    } 

} 
} 

ответ

2

функция не возвращение ложное, потому что он не возвращает ничего . Его тип возврата - void. Он принимает false как параметр.

Почему?

Ну, я напечатал «андроид OnChange» в Google и выбрал первый результат, и обнаружил следующее:

This method is called when a change occurs to the cursor that is being observed. 
Parameters 
selfChange true if the update was caused by a call to commit on the cursor 
       that is being observed. 

Так что все, что произошло, что был изменен курсор и не по телефону его .commit() способ. Вы выберете только «истинный» вход для этой функции, если вызывается .commit().

+0

Моя точка зрения была более «так вы применяете некоторые базовые навыки решения проблем, чтобы хотя бы лучше понять, что происходит». Но в любом случае ... даже если вы вызываете '.commit()' где-то, это просто приводит к другому вызову 'onChange (true)'; он не предотвращает или не отменяет вызов 'onChange (false)', который запускается чем-то другим. Итак, что мне интересно в этот момент, почему факт, что параметр «ложь» беспокоит вас? Вы знаете, почему функция вообще называется? Затем вы должны уже знать, что делать внутри функции, когда она вызывается. –

+0

И метод onChange вызывается, когда происходит изменение курсора, который наблюдается. Поэтому я анализирую URI поставщика контента. И изменение происходит с курсором, потому что я сделал новый вызов, а курсор - новые данные. , но он получает новые данные со всеми старыми данными, существующими там в поставщике контента. Поэтому, если я получу новые данные, это вызывает изменение. И почему onChange не называется? Внутри метода onChange я вызову метод fillList, то есть метод, который извлекает данные из поставщика контента и вставляет эти данные в базу данных. Понял? – rogcg

+0

... Значит, вы знаете, что функция не вызывается в этой конкретной ситуации, но вы не знаете, когда она вызывается, только то, что она фактически называется? Какие? –

Смежные вопросы