Android:内容观察者的“onChange()”方法被多次调用
Posted
技术标签:
【中文标题】Android:内容观察者的“onChange()”方法被多次调用【英文标题】:Android: content observer's "onChange()" method is called multiple times 【发布时间】:2011-08-26 23:03:18 【问题描述】:我正在为content://sms
使用内容观察器。我正在将所有消息写入 SD 卡中的文本文件。但是内容观察器中的onChange()
方法被多次调用,并且同一条消息被多次写入文本文件。如何避免这种情况?另外我想知道内容观察器是否会减慢手机速度。
【问题讨论】:
那么问题解决了吗?我有同样的问题。你能帮我解决一下吗?我在服务中注册了我的观察员。它会重复两次、三次、几次。 【参考方案1】:您需要重写 DeliverSelfNotifications() 以返回 true。
class ObserverSms extends ContentObserver
private Context mContext;
public ObserverSms(Context context, Handler handler)
super(handler);
mContext = context;
@Override
public boolean deliverSelfNotifications()
return true;
@Override
public void onChange(boolean selfChange)
super.onChange(selfChange);
MyLog.logDebugInConsole(TAG, "Sms Database Changed");
【讨论】:
它没有改变结果,我为历史观察者编码并测试了它,问题仍然存在(按照你所说的)【参考方案2】:Vivek,请确保在您离开活动时取消注册您的内容观察者,例如在onDestroy()
方法中,调用unregisterContentObserver()
。希望这有帮助! (在我的情况下它有效)
【讨论】:
我没有任何活动。只是内容观察者在后台运行并将所有短信记录在一个文本文件中。因此,如果我在写入文本文件后取消注册内容观察者,那么在没有内容观察者的情况下如何知道收到或发送的下一条短信? 如果这个“后台”是一个服务,尝试在它的 onDestroy() 函数中注销。 你也可以尝试在注册ContentObserver而不是Activity的时候传递Application的Context。我还没有测试过,但这应该可以。 我试过了,但它对我不起作用。为什么它不是正确答案?【参考方案3】:尝试这样做以避免多次执行 onChange()
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();
【讨论】:
你能解释一下代码吗?你诱导了什么样的行为来改变?【参考方案4】:onChange() 事件为每列更改(即“正文”或“地址”)调用一次 - 因此是多次调用。只处理一次数据的最简单方法是存储“_id”列的值,查看它是否在下一次迭代中发生了变化,然后忽略。
cur.getColumnIndex("_id")
请参见此处的示例: Sms ContentObserver onChange() fires multiple times
【讨论】:
以上是关于Android:内容观察者的“onChange()”方法被多次调用的主要内容,如果未能解决你的问题,请参考以下文章