Android用ContentObserver實(shí)現(xiàn)監(jiān)聽(tīng)電話(huà)狀態(tài)

公司項(xiàng)目有個(gè)需求锨苏,因?yàn)槭莾?nèi)部人員使用的APP公罕,需要對(duì)撥出的電話(huà)做記錄翠胰,然后實(shí)時(shí)跟進(jìn)每一撥電話(huà)是否接通或未接容贝。首先想到的是PhoneStateListener這個(gè)類(lèi),但是實(shí)現(xiàn)后之景,發(fā)現(xiàn)“接通”這個(gè)判斷并不靠譜斤富,然后想到一些APP在注冊(cè)或登錄的時(shí)候,能夠直接讀取短信并填充實(shí)現(xiàn)不用手動(dòng)輸入驗(yàn)證碼即可直接登錄的操作锻狗,最終發(fā)現(xiàn)了ContentObserver的妙用满力,并且非常簡(jiǎn)單焕参。

思路就是讀取手機(jī)通話(huà)記錄,判斷當(dāng)前撥出的號(hào)碼與通話(huà)記錄中的第一條已撥電話(huà)記錄做比較油额,如果一致并且時(shí)長(zhǎng)大于0既為撥出已接通叠纷。

但是實(shí)現(xiàn)之后有個(gè)bug,就是onChange方法會(huì)調(diào)用多次悔耘,調(diào)用多次跟進(jìn)上傳讲岁,最終在stackoverflow找到相關(guān)問(wèn)題,并結(jié)合線程解決了這個(gè)問(wèn)題衬以。

直接上最后的代碼:

public class CallContentObserver extends ContentObserver {

public Logger gLogger = Logger.getLogger(this.getClass());
private static volatile int initialPos;
private static final Uri outSMSUri = CallLog.Calls.CONTENT_URI;
private Context context;
private String address;
private Handler mHandler;   //更新UI線程


ExecutorService singleThreadExecuto



public CallContentObserver(Handler handler, Context context, String number) {
    super(handler);
    this.context = context;
    this.address = number;
    this.mHandler = handler;
    singleThreadExecuto=  Executors.newSingleThreadExecutor();
}

public void onChange(boolean selfChange) {
    super.onChange(selfChange);
    gLogger.debug("on change called");
    queryLastCall();

}

//標(biāo)記位置防止多次調(diào)用onchange
public int getLastCallId() {
    try {
        if (ActivityCompat.checkSelfPermission(context, Manifest.permission.READ_CALL_LOG) != PackageManager.PERMISSION_GRANTED) {
            return -1;
        }
        Cursor cur = context.getContentResolver().query(outSMSUri, null, null, null, CallLog.Calls.DATE + " desc");
        cur.moveToFirst();
        int lastMsgId = cur.getInt(cur.getColumnIndex("_id"));
        return lastMsgId;
    } catch (Exception e) {
        e.printStackTrace();
    }
    return -1;
}


protected  void queryLastCall() {

    singleThreadExecuto.execute(new Runnable() {
        @Override
        public void run() {
            try {
                if (ActivityCompat.checkSelfPermission(context, Manifest.permission.READ_CALL_LOG) != PackageManager.PERMISSION_GRANTED) {
                    return;
                }
                Cursor cur =
                        context.getContentResolver().query(outSMSUri, null, null, null,  CallLog.Calls.DATE + " desc");
                if (cur.moveToNext()) {
                    if (initialPos != getLastCallId()) {
                        if(!TextUtils.isEmpty(address)){
                            if (cur.getString(cur.getColumnIndex("number")).contains(address)) {
                                int _id = cur.getInt(cur.getColumnIndex("_id"));
                                int type = cur.getInt(cur.getColumnIndex("type"));//通話(huà)類(lèi)型缓艳,1 來(lái)電 .INCOMING_TYPE;2 已撥 .OUTGOING_看峻;3 未接 .MISSED_
                                String number = cur.getString(cur.getColumnIndex("number"));// 電話(huà)號(hào)碼
                                int duration = cur.getInt(cur.getColumnIndex("duration"));//通話(huà)時(shí)長(zhǎng)阶淘,單位:秒
                                String msgObj = "\nID:" + _id + "\n類(lèi)型:" + type + "\n號(hào)碼:" + number + "\n時(shí)長(zhǎng):" + duration;
                                gLogger.debug(msgObj);

                                if(type==2){
                                    if(duration>0){
                                        mHandler.obtainMessage(answered, isSuccessSend).sendToTarget();
                                    }else {
                                        mHandler.obtainMessage(noAnswer, isSuccessSend).sendToTarget();


                                    }
                                }

                            }
                        }
                        initialPos = getLastCallId();
                    }

                }
                cur.close();
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    });

}

}

以上就是修復(fù)后的代碼,目前沒(méi)有發(fā)現(xiàn)什么問(wèn)題互妓。
同理溪窒,獲取來(lái)電,或者短信記錄都可以用類(lèi)似的方案解決

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末冯勉,一起剝皮案震驚了整個(gè)濱河市澈蚌,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌灼狰,老刑警劉巖宛瞄,帶你破解...
    沈念sama閱讀 221,695評(píng)論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異交胚,居然都是意外死亡份汗,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,569評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門(mén)蝴簇,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)杯活,“玉大人,你說(shuō)我怎么就攤上這事熬词∨跃” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 168,130評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵荡澎,是天一觀的道長(zhǎng)均践。 經(jīng)常有香客問(wèn)我,道長(zhǎng)摩幔,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 59,648評(píng)論 1 297
  • 正文 為了忘掉前任鞭铆,我火速辦了婚禮或衡,結(jié)果婚禮上焦影,老公的妹妹穿的比我還像新娘。我一直安慰自己封断,他們只是感情好斯辰,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,655評(píng)論 6 397
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著坡疼,像睡著了一般彬呻。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上柄瑰,一...
    開(kāi)封第一講書(shū)人閱讀 52,268評(píng)論 1 309
  • 那天闸氮,我揣著相機(jī)與錄音,去河邊找鬼教沾。 笑死蒲跨,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的授翻。 我是一名探鬼主播或悲,決...
    沈念sama閱讀 40,835評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼堪唐!你這毒婦竟也來(lái)了巡语?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 39,740評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤淮菠,失蹤者是張志新(化名)和其女友劉穎男公,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體兜材,經(jīng)...
    沈念sama閱讀 46,286評(píng)論 1 318
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡理澎,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,375評(píng)論 3 340
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了曙寡。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片糠爬。...
    茶點(diǎn)故事閱讀 40,505評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖举庶,靈堂內(nèi)的尸體忽然破棺而出执隧,到底是詐尸還是另有隱情,我是刑警寧澤户侥,帶...
    沈念sama閱讀 36,185評(píng)論 5 350
  • 正文 年R本政府宣布镀琉,位于F島的核電站,受9級(jí)特大地震影響蕊唐,放射性物質(zhì)發(fā)生泄漏屋摔。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,873評(píng)論 3 333
  • 文/蒙蒙 一替梨、第九天 我趴在偏房一處隱蔽的房頂上張望钓试。 院中可真熱鬧装黑,春花似錦、人聲如沸弓熏。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 32,357評(píng)論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)挽鞠。三九已至疚颊,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間信认,已是汗流浹背材义。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,466評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留狮杨,地道東北人母截。 一個(gè)月前我還...
    沈念sama閱讀 48,921評(píng)論 3 376
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像橄教,于是被迫代替她去往敵國(guó)和親清寇。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,515評(píng)論 2 359

推薦閱讀更多精彩內(nèi)容