1

내 활동에서 서브 클래 싱으로 선언 된 ContentObserver onChange()가 있습니다. 하지만 항상 false를 반환합니다. 아무도 그 이유를 말할 수 있습니까?Android onChange() 메서드는 false 만 반환합니다.

(업데이트) 이 코드는 CallLog 콘텐츠 공급자가 변경되면 fillList를 호출해야합니다. 즉, 새로운 호출을하면 콜의 데이터가 컨텐트 프로 바이더에 삽입되므로 뭔가 변경되었다는 것을 관찰자에게 알려야하므로 fillList()가 호출됩니다.하지만 항상 반환합니다. 거짓, 심지어 에뮬레이터에서 새 전화를 걸면.

다음은 코드입니다. 이 아무것도을 반환하지 않기 때문에

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

...이 특정 상황에서는 함수가 호출되지 않지만 호출되는 시점을 알 수는 없으며 사실 호출되는 것입니다. 뭐? –

관련 문제