Android ContentObserver 监听短信思路详解

 更新时间:2024年09月29日 10:04:43   作者:大风起兮云飞扬丶  
ContentObserver允许在Android中监控特定数据的变化,可用于短信等应用的数据监听,开发者可通过继承ContentObserver并实现onChange方法来定义当目标内容变化时的响应行为,感兴趣的朋友一起看看吧

概述 

内容观察器ContentObserver给目标内容注册一个观察器,目标内容的数据一旦发生变化,观察器规定好的动作马上触发,从而执行开发者预先定义的代码。

思路

注册一个监听

getContentResolver().registerContentObserver(uri, true, mObserver);

继承 ContentObserver 实现一个用于回调的监听类

 private static class SmsGetObserver extends ContentObserver {
        private final Context mContext;
        public SmsGetObserver(Context context) {
            super(new Handler(Looper.getMainLooper()));
            this.mContext = context;
        }
        @SuppressLint("Range")
        @Override
        public void onChange(boolean selfChange, @Nullable Uri uri) {
            super.onChange(selfChange, uri);
            // onChange会多次调用,收到一条短信会调用两次onChange
            // mUri===content://sms/raw/20
            // mUri===content://sms/inbox/20
            // 安卓7.0以上系统,点击标记为已读,也会调用一次
            // mUri===content://sms
            // 收到一条短信都是uri后面都会有确定的一个数字,对应数据库的_id,比如上面的20
            Log.d("aabb",uri.toString());
            if (uri == null) {
                return;
            }
            if (uri.toString().contains("content://sms/raw") ||
                    uri.toString().equals("content://sms")) {
                return;
            }
            // 通过内容解析器获取符合条件的结果游标集
            Cursor cursor = mContext.getContentResolver().query(uri, new String[]{"address", "body", "date"}, null, null, "date Desc");
            if (cursor.moveToNext()) {
                // 短信的发送号码
                String sender = cursor.getString(cursor.getColumnIndex("address"));
                // 短信内容
                String content = cursor.getString(cursor.getColumnIndex("body"));
                Log.d("AAAA", String.format("sender:%s,content:%s", sender, content));
            }
            cursor.close();
        }
    }

短信相关权限

    <uses-permission android:name="android.permission.SEND_SMS" />
    <uses-permission android:name="android.permission.READ_SMS" />

完整代码

package com.example.cpclient;
import androidx.annotation.Nullable;
import androidx.appcompat.app.AppCompatActivity;
import android.annotation.SuppressLint;
import android.content.Context;
import android.database.ContentObserver;
import android.database.Cursor;
import android.net.Uri;
import android.os.Bundle;
import android.os.Handler;
import android.os.Looper;
import android.util.Log;
public class MonitorSmsActivity extends AppCompatActivity {
    private SmsGetObserver mObserver;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_monitor_sms);
        // 给指定Uri注册内容观察器,一旦发生数据变化,就触发观察器的onChange方法
        Uri uri = Uri.parse("content://sms");
        // notifyForDescendents:
        // false :表示精确匹配,即只匹配该Uri,true :表示可以同时匹配其派生的Uri
        // 假设UriMatcher 里注册的Uri共有以下类型:
        // 1.content://AUTHORITIES/[table]
        // 2.content://AUTHORITIES/[table]/#
        // 3.content://AUTHORITIES/[table]/[subtable]
        // 假设我们当前需要观察的Uri为content://AUTHORITIES/student:
        // 如果发生数据变化的 Uri 为 3。
        // 当notifyForDescendents为false,那么该ContentObserver会监听不到,但是当notifyForDescendents 为ture,能捕捉该Uri的数据库变化。
        mObserver = new SmsGetObserver(this);
        getContentResolver().registerContentObserver(uri, true, mObserver);
    }
    @Override
    protected void onDestroy() {
        super.onDestroy();
        getContentResolver().unregisterContentObserver(mObserver);
    }
    private static class SmsGetObserver extends ContentObserver {
        private final Context mContext;
        public SmsGetObserver(Context context) {
            super(new Handler(Looper.getMainLooper()));
            this.mContext = context;
        }
        @SuppressLint("Range")
        @Override
        public void onChange(boolean selfChange, @Nullable Uri uri) {
            super.onChange(selfChange, uri);
            // onChange会多次调用,收到一条短信会调用两次onChange
            // mUri===content://sms/raw/20
            // mUri===content://sms/inbox/20
            // 安卓7.0以上系统,点击标记为已读,也会调用一次
            // mUri===content://sms
            // 收到一条短信都是uri后面都会有确定的一个数字,对应数据库的_id,比如上面的20
            Log.d("aabb",uri.toString());
            if (uri == null) {
                return;
            }
            if (uri.toString().contains("content://sms/raw") ||
                    uri.toString().equals("content://sms")) {
                return;
            }
            // 通过内容解析器获取符合条件的结果游标集
            Cursor cursor = mContext.getContentResolver().query(uri, new String[]{"address", "body", "date"}, null, null, "date Desc");
            if (cursor.moveToNext()) {
                // 短信的发送号码
                String sender = cursor.getString(cursor.getColumnIndex("address"));
                // 短信内容
                String content = cursor.getString(cursor.getColumnIndex("body"));
                Log.d("AAAA", String.format("sender:%s,content:%s", sender, content));
            }
            cursor.close();
        }
    }
}

拓展

当在你的provider中,别人insert了一条数据,你要告知他是否成功了

使用 notifyChange进行通知回调

getContext().getContentResolver().notifyChange()

案例

   @Override
    public Uri insert(Uri uri, ContentValues values) {
        if (URI_MATCHER.match(uri) == USERS) {
            SQLiteDatabase db = dbHelper.getWritableDatabase();
            long rowId = db.insert(UserDBHelper.TABLE_NAME, null, values);
            if (rowId > 0) {
                // 如果添加成功,就利用新记录的行号生成新的地址
                Uri newUri = ContentUris.withAppendedId(UserInfoContent.CONTENT_URI, rowId);
                // 通知监听器,数据已经改变
                getContext().getContentResolver().notifyChange(newUri, null);
            }
        }
        return uri;
    }

案例代码

到此这篇关于Android ContentObserver 监听短信的文章就介绍到这了,更多相关Android ContentObserver 监听短信内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 详解OpenGL Shader抗锯齿的实现

    详解OpenGL Shader抗锯齿的实现

    普通绘制圆形形状时可以看到图形边缘会有明显锯齿现象并不像真实圆形形状一样圆润边缘平滑。本文将介绍如何通过自制函数实现抗锯齿,需要的可以参考一下
    2022-02-02
  • Android OpenGLES2.0绘制三角形(二)

    Android OpenGLES2.0绘制三角形(二)

    这篇文章主要为大家详细介绍了Android OpenGLES2.0绘制三角形的相关资料,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2016-12-12
  • Android App开发中创建Fragment组件的教程

    Android App开发中创建Fragment组件的教程

    这篇文章主要介绍了Android App开发中创建Fragment的教程,Fragment是用以更灵活地构建多屏幕界面的可UI组件,需要的朋友可以参考下
    2016-05-05
  • android自定义ProgressDialog加载效果

    android自定义ProgressDialog加载效果

    这篇文章主要为大家详细介绍了android自定义ProgressDialog加载效果,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-06-06
  • Android气泡效果实现方法

    Android气泡效果实现方法

    这篇文章主要介绍了Android气泡效果实现方法,结合实例形式详细分析了Android实现气泡效果的页面布局及功能代码,涉及RelativeLayout布局,TextView控件及对话框Dialog相关使用技巧,需要的朋友可以参考下
    2016-01-01
  • Android 仿微信朋友圈点赞和评论弹出框功能

    Android 仿微信朋友圈点赞和评论弹出框功能

    这篇文章主要介绍了Android 仿微信朋友圈点赞和评论弹出框功能的相关资料,非常不错,具有参考解决价值,需要的朋友可以参考下
    2016-11-11
  • Android编程防止进程被第三方软件杀死的方法

    Android编程防止进程被第三方软件杀死的方法

    这篇文章主要介绍了Android编程防止进程被第三方软件杀死的方法,涉及Android进程操作的相关技巧,具有一定参考借鉴价值,需要的朋友可以参考下
    2015-10-10
  • 自定义Android注解系列教程之注解变量

    自定义Android注解系列教程之注解变量

    这篇文章主要给大家介绍了关于自定义Android注解系列教程之注解变量的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2018-07-07
  • Android返回键功能的实现方法

    Android返回键功能的实现方法

    这篇文章主要介绍了Android返回键功能的实现方法,实例分析了Android返回键的原理与具体的功能实现代码,非常具有实用价值,需要的朋友可以参考下
    2015-02-02
  • Android 自定义图片地图坐标功能的实现

    Android 自定义图片地图坐标功能的实现

    最近项目要求实现一个在自定义地图图片上添加坐标信息的功能,类似于在图片做标注的功能,这种功能纠结该如何实现呢?下面小编通过实例代码给大家介绍Android 自定义地图的实现,需要的朋友参考下吧
    2021-07-07

最新评论