zoukankan      html  css  js  c++  java
  • 信息列表中的ContentObserver、CONTENT_URI等

    1. 注册ContentObserver时Sms.Inbox.CONTENT_URI应改成Sms.CONTENT_URI.

    getContentResolver().registerContentObserver(Sms.Inbox.CONTENT_URI,
            true, mInboxObserver);

    此时Uri是Sms.Inbox.CONTENT_URI,它并没有表示一张表。我将它理解为Sms.CONTENT_URI的一个视图。

    getContentResolver().registerContentObserver(Sms.CONTENT_URI,
            true, mInboxObserver);

    2. 在使用getContentResolver().update()方法时必须设置selection参数。

    如果selection为null,表示Uri表示的所有数据都在update的范围之中。

    如果Uri表示某一张表,比如Sms.CONTENT_URI表示Sms表,如果没有设定selection,那么将修改Sms表中所有数据,造成低级但严重错误。

    private ContentObserver mInboxObserver = new ContentObserver(null) {
        public void onChange(boolean selfChange) {
            Cursor cursor = getContentResolver().query(Sms.Inbox.CONTENT_URI,
                    new String[]{Sms._ID, Sms.THREAD_ID, Sms.TYPE, Sms.BODY, Sms.DATE, Sms.READ},
                    "type = " + Sms.MESSAGE_TYPE_INBOX + " and read = 0 and thread_id = " + getOrCreateThreadId(),
                    null, null);
            TxrjMessage msg = null;
            if(cursor != null) {
                if(cursor.moveToFirst()) {
                    msg = new TxrjMessage();
                    msg.setMessageId(cursor.getInt(cursor.getColumnIndex(Sms._ID)));
                    msg.setThreadId(cursor.getInt(cursor.getColumnIndex(Sms.THREAD_ID)));
                    msg.setType(cursor.getInt(cursor.getColumnIndex(Sms.TYPE)));
                    msg.setBody(cursor.getString(cursor.getColumnIndex(Sms.BODY)));
                    msg.setTime(cursor.getLong(cursor.getColumnIndex(Sms.DATE)));
                    mMessages.add(msg);
                    mHandler.sendMessage(mHandler.obtainMessage()); // 在优化过程中会改掉。
                }
                cursor.close();
            }
            if(msg != null) {
                ContentValues values = new ContentValues();
               
    values.put(Sms._ID, msg.getMessageId()); // Bug语句。
                values.put(Sms.READ, 1);
                getContentResolver().update(Sms.Inbox.CONTENT_URI, values, null, null); // selection为null是个bug。
            }
        }
    };

    3. 在onChange()方法中使用Handler,使程序在主线程中更新UI,

    onChange()方法是否在主线程中执行,有待确定。

    private Handler mHandler = new Handler(){
        @Override
        public void handleMessage(Message msg) {
            super.handleMessage(msg);
            mListAdapter.notifyDataSetChanged();
            mListView.setSelection(mListView.getBottom());
        }
    };

    4. 在会话列表界面中,点击某个会话打开信息列表界面之前将此会话中的未读信息改为已读。

    在ConversationListActivity中点击某个item时首先执行

    if(thread.getUnReadCount()>0) {
        ContentValues values = new ContentValues();
        values.put(Sms.READ, 1);
        getContentResolver().update(Sms.Inbox.CONTENT_URI, values,
                "read=0 and thread_id=" + thread.getThreadId(), null);

    }

    5. 将收到的未读信息在ListView中显示出来,并且修改它的read字段。

    在执行getContentResolver.update()之前关闭cursor,否则:

    当update()方法执行之后修改了数据,然后再从之前query返回的cursor中获得数据将出现不同步结果。

    private ContentObserver mInboxObserver = new ContentObserver(null) {
        public void onChange(boolean selfChange) {
            Log.i("txrjsms", "receive a message.");           
            Cursor cursor = getContentResolver().query(Sms.CONTENT_URI,
                    new String[]{Sms._ID, Sms.THREAD_ID, Sms.TYPE, Sms.BODY, Sms.DATE, Sms.READ},
                    "type = " + Sms.MESSAGE_TYPE_INBOX + " and read = 0 and thread_id = " + mThreadId,
                    null, null);
            TxrjMessage msg = null;
            if(cursor != null) {
                if (cursor.moveToFirst()) {
                    msg = new TxrjMessage();
                    msg.setMessageId(cursor.getInt(cursor.getColumnIndex(Sms._ID)));
                    msg.setThreadId(cursor.getInt(cursor.getColumnIndex(Sms.THREAD_ID)));
                    msg.setType(cursor.getInt(cursor.getColumnIndex(Sms.TYPE)));
                    msg.setBody(cursor.getString(cursor.getColumnIndex(Sms.BODY)));
                    msg.setTime(cursor.getLong(cursor.getColumnIndex(Sms.DATE)));
                    cursor.close();
                    ContentValues values = new ContentValues();
                    values.put(Sms.READ, 1);
                    getContentResolver().update(Sms.CONTENT_URI, values, "_id=" + msg.getMessageId(), null);
                    mMessages.add(msg);
                    mHandler.sendMessage(mHandler.obtainMessage());
                }
            }
        }
    };

    6. 在uri中指定了具体哪条数据,因此可以将selection置为空。

    private void updateMsgType(Uri uri, int type) {
        ContentValues values = new ContentValues();
        values.put(Sms.TYPE, type);
        getContentResolver().update(uri, values, null, null);
    }

  • 相关阅读:
    转载ORACLE批量绑定FORALL与BULK COLLECT
    Oracle Locking Survival Guide
    转载:TOAD中查看执行计划
    Oracle 9i/10g编程艺术笔记第七章 并发与多版本
    C#调用Oracle存储过程返回多结果集
    转载oracle 字符集查看与修改
    转载:Oracle的优化器(Optimizer)
    Oracle 随笔
    转载:Oracle中SQL语句执行效率的查找与解决
    当查询和设置需要输入Pn时界面的处理方法
  • 原文地址:https://www.cnblogs.com/fengzhblog/p/3190992.html
Copyright © 2011-2022 走看看