2011-09-20 99 views
12

我有以下ContentObserver實現接收和寫短信,但它被稱爲多次。爲什麼多次調用ContentObserver?

代碼:

public class SMSObserverActivity extends Activity { 
    protected MyContentObserver observer = null; 

    /** Called when the activity is first created. */ 
    @Override 
    public void onCreate(Bundle savedInstanceState){ 
     super.onCreate(savedInstanceState); 
     setContentView(R.layout.main); 
     String url = "content://mms-sms/"; 
     Uri uri = Uri.parse(url); 
     observer = new MyContentObserver(new Handler()); 
     getContentResolver().registerContentObserver(uri, true, observer); 
    } 

    @Override 
    protected void onDestroy(){ 
     super.onDestroy(); 

     getContentResolver().unregisterContentObserver(observer); 
    } 

    class MyContentObserver extends ContentObserver { 
     ContentValues values = new ContentValues(); 
     Handler handler; 

     public MyContentObserver(Handler handler){ 
      super(handler); 
      this.handler = handler; 
     } 

     @Override 
     public boolean deliverSelfNotifications(){ 
      return false; 
     } 


     @Override 
     public void onChange(boolean arg0){ 
      super.onChange(arg0); 

      Log.v("SMS", "Notification on SMS observer"); 
      values.put("status", 5); 
      Message msg = new Message(); 
      msg.obj = "xxxxxxxxxx"; 
      int threadId = 0; 
      handler.sendMessage(msg); 

      Uri uriSMSURI = Uri.parse("content://sms/"); 
      Cursor cur = 
        getContentResolver().query(uriSMSURI, null, null, null, 
          null); 
      cur.moveToNext(); 
      Log.e("sms", cur.getString(4)+" "+cur.getString(11)); 
     } 
    } 
} 

清單:

<?xml version="1.0" encoding="utf-8"?> 
<manifest xmlns:android="http://schemas.android.com/apk/res/android" 
    package="com.test" 
    android:versionCode="1" 
    android:versionName="1.0"> 
    <uses-sdk android:minSdkVersion="8" /> 
    <uses-permission android:name="android.permission.READ_SMS"></uses-permission> 
    <uses-permission android:name="android.permission.WRITE_SMS"></uses-permission> 

    <application android:icon="@drawable/icon" android:label="@string/app_name"> 
     <activity android:name=".SMSObserverActivity" 
        android:label="@string/app_name"> 
      <intent-filter> 
       <action android:name="android.intent.action.MAIN" /> 
       <category android:name="android.intent.category.LAUNCHER" /> 
      </intent-filter> 
     </activity> 

    </application> 
</manifest> 

爲什麼叫多次?

編輯:
有一個想法,這個問題是由缺少unregisterContentObserver引起的,但它沒有區別。

+0

你有沒有想出解決辦法? –

+0

還沒有,我還沒有時間,但閱讀答案的評論,並嘗試與服務。也許你會得到它的工作。 ;) – CSchulz

+0

如果在服務中運行,問題是一樣的恐怕 –

回答

11

發生這種情況是因爲您正在爲整個SMS數據庫註冊您的內容觀察員。因此,每當數據庫中的表條目得到更新時,您的內容觀察者都會收到通知。

在這種情況下,當一個消息發送例如大約7個表格條目得到更新,以便您的內容觀察者得到7次通知。

因爲我只是在發送郵件時感興趣,所以我已經更改爲只觀察排隊的郵件,這意味着我的觀察者總是得到三次通知,所以我已經實現了防止這種情況的代碼。

可能還有一些其他問題,如多收件人或多部分郵件,但基本工作到目前爲止。

+0

好吧,我認爲觀察者只被稱爲每次對象的所有相關更新的一次。我認爲有一個唯一的ID可以看到,哪些通知屬於一次更新,不是嗎? – CSchulz

0

如果你想啓用的觀察者只有當活動處於活躍狀態,我勸你還是分別移動registerContentObserver()unregisterContentObserver()的方法onResume()onPause()。如果您的應用程序退出,則可能不會調用onDestroy(),但onPause()保證爲。

+0

當* Activity *正在運行時,觀察者應始終處於活動狀態。這只是一個觀測內容測試。 – CSchulz

+0

活動在'onResume()'和'onPause()'之間運行,另一次是在後臺或不可見狀態。它甚至可能在沒有調用'onDestroy()'的情況下被殺死,並且當你在'onCreate()'中重新創建應用程序時,你將會第二次註冊一個觀察者。 – Malcolm

+0

爲什麼要註冊兩次?我有一個註銷聲明。 – CSchulz

2

爲了避免內容觀察者發送多個短信試試這個

public class SmsObserver extends ContentObserver { 
    SharedPreferences trackMeData; 
    private Context context; 
    private static int initialPos; 
    private static final String TAG = "SMSContentObserver"; 
    private static final Uri uriSMS = Uri.parse("content://sms/sent"); 

    public SmsObserver(Handler handler, Context ctx) { 
     super(handler); 
     context = ctx; 
     trackMeData = context.getSharedPreferences("LockedSIM", 0); 
     initialPos = getLastMsgId(); 

    } 

    @Override 
    public void onChange(boolean selfChange) { 
     super.onChange(selfChange); 
     queryLastSentSMS(); 
    } 

    public int getLastMsgId() { 

     Cursor cur = context.getContentResolver().query(uriSMS, null, null, null, null); 
     cur.moveToFirst(); 
     int lastMsgId = cur.getInt(cur.getColumnIndex("_id")); 
     Log.i(TAG, "Last sent message id: " + String.valueOf(lastMsgId)); 
     return lastMsgId; 
    } 

    protected void queryLastSentSMS() { 

     new Thread(new Runnable() { 

      @Override 
      public void run() { 
       Cursor cur = 
        context.getContentResolver().query(uriSMS, null, null, null, null); 

       if (cur.moveToNext()) { 



        try { 

         String body = cur.getString(cur.getColumnIndex("body")); 

         if (initialPos != getLastMsgId()) { 

          String receiver = cur.getString(cur.getColumnIndex("address")); 
          Log.i("account", myDeviceId); 
          Log.i("date", day + "-" + month + "-" + year + " " 
           + hour + ":" + minute + ":" + seconde); 
          Log.i("sender", myTelephoneNumber); 
          Log.i("receiver", receiver); 


          // Then, set initialPos to the current position. 
          initialPos = getLastMsgId(); 

          sendsmstoph(receiver, body); 
         } 
        } catch (Exception e) { 
         // Treat exception here 
        } 
       } 
       cur.close(); 
      } 
     }).start(); 

    } 
+0

調用initialPos = getLastMsgId();從構造函數拋出空指針異常 –

+0

我注意到,這是好的,但並不總是傻瓜證明。所以我認爲使用SharedPreferences可以解決這個問題。以下是我如何完成它: if(!sharedPreferences.getString(「lastMessageTime」,「0」) .equals(String.valueOf(timeInMillis)) ||!sharedPreferences.getString(「lastMessageNumber」,「0」) .equals(數字)) – Usman

相關問題