Android簡訊監聽實現,及Android4.4之後簡訊機制變更

来源:https://www.cnblogs.com/yelanggu/archive/2023/02/16/17127562.html
-Advertisement-
Play Games

Spring配置 1. 別名 alias 設置別名 , 為bean設置別名 , 可以設置多個別名 <!--設置別名:在獲取Bean的時候可以使用別名獲取--> <alias name="userT" alias="userNew"/> 2. Bean的配置 <!--bean就是java對象,由Spr ...


Android簡訊監聽實現,及Android4.4之後簡訊機制變更

 

前陣子公司有一個項目,簡單的監聽簡訊應用,功能只有如下兩個:

1.監聽簡訊並獲取簡訊內容上傳伺服器; 2.從伺服器獲取簡訊內容,發送出去    按照傳統的思路,監聽簡訊我們有兩種方式;第一種是使用廣播方式監聽簡訊廣播到來,第二種則是通過觀察者,監聽資料庫數據變化。     其中,指的註意的是Android4.4之後版本,新引入了預設簡訊應用的概念,系統可以設置由某個應用來處理簡訊; 本文我們將帶人們分析以下幾個問題:    1.監聽資料庫變化方式監聽簡訊    2.通過廣播監聽簡訊內容    3.Android 4.4以上版本簡訊許可權問題    4.Android4.4版本以上設置預設簡訊應用    

1.監聽資料庫變化方式監聽簡訊內容

既然是監聽資料庫變化 那我們就應該清楚簡訊的資料庫表結構: sms主要結構:
  _id:          簡訊序號,如100
  thread_id:對話的序號,如100,與同一個手機號互發的簡訊,其序號是相同的
  address:  發件人地址,即手機號,如+86138138000
  person:   發件人,如果發件人在通訊錄中則為具體姓名,陌生人為null
  date:       日期,long型,如1346988516,可以對日期顯示格式進行設置
  protocol: 協議0SMS_RPOTO簡訊,1MMS_PROTO彩信
  read:      是否閱讀0未讀,1已讀
  status:    簡訊狀態-1接收,0complete,64pending,128failed
  type:       簡訊類型1是接收到的,2是已發出
  body:      簡訊具體內容
  service_center:簡訊服務中心號碼編號,如+8613800755500    既然需要操作資料庫,便少不了使用ContentResolver,所以我們應該還需要瞭解,簡訊的content uri : 全部簡訊:content://sms/ 收件箱:content://sms/inbox 發件箱:content://sms/sent 草稿箱:content://sms/draft   複製代碼
private Uri SMS_INBOX = Uri.parse("content://sms/inbox");  
    public void getSmsFromPhone() {  
        ContentResolver cr = getContentResolver();  
        String[] projection = new String[] { "body","address" };//"_id", "address", "person",, "date", "type  
        String where = " date >  "  
                + (System.currentTimeMillis() - 10 * 60 * 1000);  
        Cursor cur = cr.query(SMS_INBOX, projection, where, null, "date desc");  
        if (null == cur)  
            return;  
        if (cur.moveToFirst()) {  
            String number = cur.getString(cur.getColumnIndex("address"));//手機號  
            String body = cur.getString(cur.getColumnIndex("body"));  
            //TODO 這裡是具體處理邏輯 
            
        }  
    }
複製代碼

 

在這裡我們只是寫了一個方法查詢資料庫,但是還有一個問題就是我們應該在什麼時候去查資料庫,總不能起個線程去輪訓,這樣太耗費資源了,這裡我們可以是用觀察者模式;

 

複製代碼
private SmsObserver smsObserver;  
  
    protected void onCreate(Bundle savedInstanceState) {  
        super.onCreate(savedInstanceState);  
        setContentView(R.layout.app_login);  
        smsObserver = new SmsObserver(this, smsHandler);  
        getContentResolver().registerContentObserver(SMS_INBOX, true,  
                smsObserver);  
  
    }  
    public Handler smsHandler = new Handler() {  
        //這裡可以進行回調的操作  
        //TODO  
  
    };  
    class SmsObserver extends ContentObserver {  
  
        public SmsObserver(Context context, Handler handler) {  
            super(handler);  
        }  
  
        @Override  
        public void onChange(boolean selfChange) {  
            super.onChange(selfChange);  
            //每當有新簡訊到來時,使用我們獲取短消息的方法  
            getSmsFromPhone();  
        }  
    }
複製代碼

 

在這裡我們註冊了一個觀察者,監聽收件箱的變化,一旦收件箱變化,我們就查詢資料庫,去除最新的一條數據 相應的許可權這裡就不貼出來了 至此我們就實現了通過監聽資料庫的方式來監聽簡訊內容    

 2.通過廣播監聽簡訊內容

複製代碼
public class SmsReceiver extends BroadcastReceiver {
    public static final String SMS_RECEIVED = "android.provider.Telephony.SMS_RECEIVED";
    private static final String TAG = "yjj";
    public SmsReceiver() {
        Log.i("yjj", "new SmsReceiver");
    }
    @Override
    public void onReceive(final Context context, Intent intent) {
        Log.i(TAG, "jie shou dao");
        Cursor cursor = null;
        try {
            if (SMS_RECEIVED.equals(intent.getAction())) {
                Log.d(TAG, "sms received!");
                Bundle bundle = intent.getExtras();
                if (bundle != null) {
                    Object[] pdus = (Object[]) bundle.get("pdus");
                    final SmsMessage[] messages = new SmsMessage[pdus.length];
                    for (int i = 0; i < pdus.length; i++) {
                        messages[i] = SmsMessage.createFromPdu((byte[]) pdus[i]);
                    }
                    if (messages.length > 0) {
                        String content = messages[0].getMessageBody();
                        String sender = messages[0].getOriginatingAddress();
                        long msgDate = messages[0].getTimestampMillis();
                        String smsToast = "New SMS received from : "
                                + sender + "\n'"
                                + content + "'";
                        Toast.makeText(context, smsToast, Toast.LENGTH_LONG)
                                .show();
                        Log.d(TAG, "message from: " + sender + ", message body: " + content
                                + ", message date: " + msgDate);
                        //自己的邏輯
                    }
                }
                cursor = context.getContentResolver().query(Uri.parse("content://sms"), new String[] { "_id", "address", "read", "body", "date" }, "read = ? ", new String[] { "0" }, "date desc");
                if (null == cursor){
                    return;
                }
                Log.i(TAG,"m cursor count is "+cursor.getCount());
                Log.i(TAG,"m first is "+cursor.moveToFirst());
            }
        } catch (Exception e) {
            e.printStackTrace();
            Log.e(TAG, "Exception : " + e);
        } finally {
            if (cursor != null) {
                cursor.close();
                cursor = null;
            }
        }
    }
}
複製代碼

 

這個很簡單就是定義一個廣播接收者,並且在清單文件中註冊(註冊有兩種方式,這裡就不展開了)

複製代碼
<receiver android:name=".message.SmsReceiver" android:permission="android.permission.BROADCAST_SMS">
            <intent-filter android:priority="2147483647">
                <action android:name="android.provider.Telephony.SMS_DELIVER" />
                <action android:name="android.provider.Telephony.SMS_RECEIVED" />
            </intent-filter>
</receiver>
複製代碼

 

3.Android 4.4以上版本簡訊許可權問題

Android4.4版本之前,簡訊有著一個問題,任何應用只要想,就可以操作簡訊,著包括監聽簡訊、修改簡訊、刪除簡訊、攔截簡訊等,因而市面上有著成片的簡訊應用,著也使得Android系統簡訊的管理變得越發的混亂。 針對這個問題Google在Android4.4版本之後,引進了一個新的概念----預設簡訊應用。即用戶可以在系統中選擇由哪個應用預設對簡訊進行處理。 針對Android4.4版本的,Google提供了 SMS_DELIVER_ACTION(sms)和 WAP_PUSH_DELIVER_ACTION(MMS)這兩個intent給預設的簡訊使用,也就是說只有預設簡訊才可以收到這兩個廣播,也只有收到這兩個廣播的簡訊應用才可以對簡訊資料庫機型操作,其他的簡訊應用可以使用SMS_RECEIVED_ACTION對簡訊進行監聽,但僅僅只能讀取(理論上可以監聽,但是在一臺6.0系統的三星機器上並不能監聽到,具體什麼原因沒查出來,當然這是在我所寫的應用沒有成為手機預設簡訊應用的情況下,當設置為預設簡訊應用後監聽也是正常的) 另外,值得一提的是,在Android4.4版本之前SMS_RECEIVED_ACTION是一個有序廣播,這意味著在Android4.4版本之前,應用在接受廣播之後可以對廣播進行攔截;但是在Android4.4之後,這個攔截動作不會生效,這就意味著Android4.4之後,非預設簡訊應用對簡訊除了讀操作外,沒有更多的許可權了

 

 

4.Android4.4版本以上設置預設簡訊應用

我們已經分析了Android4.4版本之後簡訊的改變----預設簡訊應用,但是並不是每個應用都可以被設置為預設簡訊應用,接下來我們來實現一下怎麼讓我們的簡訊應用可以被設置為預設短息應用。 複製代碼
<!-- BroadcastReceiver that listens for incoming MMS messages -->
        <receiver android:name=".message.MmsReceiver"
            android:permission="android.permission.BROADCAST_WAP_PUSH">
            <intent-filter>
                <action android:name="android.provider.Telephony.WAP_PUSH_DELIVER" />
                <data android:mimeType="application/vnd.wap.mms-message" />
            </intent-filter>
        </receiver>
        <!-- Activity that allows the user to send new SMS/MMS messages -->
        <activity android:name=".message.ComposeSmsActivity" >
            <intent-filter>
                <action android:name="android.intent.action.SEND" />
                <action android:name="android.intent.action.SENDTO" />
                <category android:name="android.intent.category.DEFAULT" />
                <category android:name="android.intent.category.BROWSABLE" />
                <data android:scheme="sms" />
                <data android:scheme="smsto" />
                <data android:scheme="mms" />
                <data android:scheme="mmsto" />
            </intent-filter>
        </activity>
        <!-- Service that delivers messages from the phone "quick response" -->
        <service android:name=".message.HeadlessSmsSendService"
            android:permission="android.permission.SEND_RESPOND_VIA_MESSAGE"
            android:exported="true" >
            <intent-filter>
                <action android:name="android.intent.action.RESPOND_VIA_MESSAGE" />
                <category android:name="android.intent.category.DEFAULT" />
                <data android:scheme="sms" />
                <data android:scheme="smsto" />
                <data android:scheme="mms" />
                <data android:scheme="mmsto" />
            </intent-filter>
        </service>
複製代碼

 

首先我們子在清單文件中寫入如下信息,可以看到我們定義了一個MmsReceiver、一個ComposeSmsActivity、一個HeadlessSmsSendService 很顯然我們需要這三個對應的java文件,即一個receiver類、一個service類、一個Activity類;這三個了都可以不需要任何內容,具體代碼如下 MmsReceiver.java
public class MmsReceiver extends BroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {
    }
}

 

ComposeSmsActivity.java

複製代碼
public class ComposeSmsActivity extends Activity {
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        // TODO Auto-generated method stub
        super.onCreate(savedInstanceState);
    }
}
複製代碼

 

HeadlessSmsSendService.java

複製代碼
public class HeadlessSmsSendService extends Service {
    @Override
    public IBinder onBind(Intent intent) {
        return null;
    }
}
複製代碼

 

通過以上步驟,我們所寫的應用就可以被設置為預設簡訊應用了

最後別忘了添加相應的許可權:

複製代碼
<uses-permission android:name="android.permission.CALL_PHONE" />
    <uses-permission android:name="android.permission.INTERNET" />
    <uses-permission android:name="android.permission.READ_CONTACTS" />
    <uses-permission android:name="android.permission.WRITE_CONTACTS" />
    <uses-permission android:name="android.permission.READ_PHONE_STATE" />
    <uses-permission android:name="android.permission.READ_CALL_LOG" />
    <uses-permission android:name="android.permission.WRITE_CALL_LOG" />
    <uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS" />
    <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
    <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
    <uses-permission android:name="android.permission.READ_LOGS" />
    <uses-permission android:name="android.permission.PROCESS_OUTGOING_CALLS" />
    <uses-permission android:name="android.permission.READ_SMS" />
    <uses-permission android:name="android.permission.RECEIVE_SMS" />
    <uses-permission android:name="android.permission.SEND_SMS" />
    <uses-permission android:name="android.permission.WRITE_SMS" />
    <uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW" />
    <uses-permission android:name="android.permission.SYSTEM_OVERLAY_WINDOW" />
複製代碼

 

這裡貼出的是我整個項目的許可權,世紀應該只需要SMS相關的許可權,這裡就不做區分了

下麵貼出參考的相關博客,供大家全面瞭解Android簡訊機制: http://blog.csdn.net/maybe_windleave/article/details/17740345 http://blog.csdn.net/liuhe688/article/details/7020612 http://blog.csdn.net/subaohao/article/details/39368961 http://blog.csdn.net/mad1989/article/details/22426415/

 

 









您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • JOSN處理和HttpMessageConverter 1.JSON處理-@ResponseBody 說明:在實際開發中,我們往往需要伺服器返回的數據都是 JSON 格式。 SpringMVC 提供了 @ResponseBody 註解,用來標註 Controller 方法的返回的格式為 JSON,將 ...
  • 時間輪工作原理解析 一.時間輪介紹 1.時間輪的簡單介紹 時間輪(TimeWheel)作為一種高效率的計時器實現方案,在1987年發表的論文Hashed and Hierarchical Timing Wheels中被首次提出。 其被髮明的主要目的在於解決當時操作系統的計時器功能實現中,維護一個定時 ...
  • 前言 對中文標題使用餘弦相似度演算法和編輯距離相似度分析進行相似度分析。 準備數據集part1 本次使用的數據集來源於前幾年的碩士學位論文,可根據實際需要更換。結構如下所示: 學位論文題名 基於捲積神經網路的人臉識別研究 P2P流媒體視頻點播系統設計和研究 校園網安全體系的設計與實現 無線感測器網路中 ...
  • 介紹 vector(矢量; 向量),vector是C++標準模板庫(STL)中的部分內容,中文偶爾譯作“容器”,但並不准確。它是一個多功能的,能夠操作多種數據結構和演算法的模板類和函數庫。vector之所以被認為是一個容器,是因為它能夠像容器一樣存放各種類型的對象,簡單地說,vector是一個能夠存放 ...
  • 目錄: finally的使用 finally面試題 final、finally、finalize的區別 finally子句的使用: 1、在finally子句中的代碼是最後且一定會執行的,即使try語句塊中的代碼出現異常 2、finally子句必須和try一起出現,不能單獨編寫 3、finally子句 ...
  • 一.併發和並行 多任務:一定時間段內,充分利用cpu資源,同時去執行多個任務 併發: 快速交替的 去執行多任務 並行: 真正同時的 去執行多任務 ,就是同時進行 二.多進程 1.多進程入門 知識點: 進程含義: 運行中的程式 進程特點: cpu資源分配的 最小單位 多進程模塊: multiproce ...
  • 一、ElasticSearch概述 官網:https://www.elastic.co/cn/downloads/elasticsearch Elaticsearch,簡稱為es,es是一個開源的高擴展的分散式全文檢索引擎,它可以近乎實時的存儲、檢索數據;本身擴展性很好,可以擴展到上百台伺服器,處理 ...
  • 教程簡介 Instagram營銷初學者教程 - 從簡單和簡單的步驟學習Instagram營銷從基本到高級概念,包括概述,業務戰略,安裝和註冊,發佈和參與,活動審查,微調內容,營銷工具和應用程式,集成其他平臺,分析工具。 教程目錄 Instagram營銷教程 Instagram營銷 - 概述 Inst ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...