1

私は自分のアクティビティでサブクラスとして宣言されたContentObserver onChange()を持っています。しかし、常にfalseを返します。なぜ誰かが私に言うことができますか?Android onChange()メソッドはfalseを返すだけです

(更新) このコードは、CallLogコンテンツプロバイダが変更された場合は、fillListを呼び出す必要があります。つまり、新しいコールを行うと、コールのデータがコンテンツプロバイダに挿入されるため、オブザーバに何か変更があったため、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」をタイプし、最初の結果を選択し、見つかった以下:

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)'の呼び出しを防ぐか、または取り消しません。だから私はこの時点で疑問に思っているのは、なぜパラメータが 'false'であることが気になるのでしょうか?なぜその関数が全く呼び出されているのか知っていますか?次に、呼び出されたときに関数内で何をすべきかを既に知っているはずです。 –

+0

そして、onChangeメソッドは、観察されているカーソルに変更が生じたときに呼び出されます。そこで、私はコンテンツプロバイダのURIを解析します。そして、私は新しい呼び出しを行ったので、カーソルに変化が起こっています、そして、カーソルは、新しいデータを取得します。 が新しいデータを取得し、古いデータはすべてコンテンツプロバイダに存在します。新しいデータを取得した場合、その原因が変わります。なぜonChangeが呼ばれていないのですか? onChangeメソッドの中で、コンテンツプロバイダからデータを取得してこのデータをデータベースに挿入するメソッドであるfillListメソッドを呼び出します。とった? – rogcg

+0

...関数がこの特定の状況で呼び出されているわけではありませんが、呼び出されているのはわかりません。何? –