关注微信号:javalearns   随时随地学Java

或扫一扫

随时随地学Java

注册的时候很多时候有短信注册的功能,但是又不想让用户手动填写验证码,所以,如果用户是用本机手机号注册的,就要想办法拦截用户的验证码短信,主动拦截,拿出验证码,帮用户填入到验证码框里,不用用户去看短信,手动输入了。所以,我们可以想想如何拦截用户的短信。

Android短信拦截,总的来说有两种方式:
(一)、在代码中,实现注册短信监听类,监听短信数据库德变换,把指定号码或者知道内容的短信屏蔽掉,这种方式是一种“假”方式,其实是在收件箱收到短信之后,再删除指定的短信。

(二)、利用广播类,如果,判断是指定的短信责进行某种操作再继续广播。但是这种方式要保证自己定义的receive的权限要高于系统的全系。

具体实现代码如下:

(一)、

//首先在Activity类中注册一个短信监听类

SmsContent content = new SmsContent(new Handler());
// 注册短信变化监听
this.getContentResolver().registerContentObserver(
Uri.parse("content://sms/"), true, content);

//其次是短信监听类的实现:

class SmsContent extends ContentObserver {

public SmsContent(Handler handler) {
super(handler);
}

@Override
public void onChange(boolean selfChange) {
super.onChange(selfChange);
// 读取收件箱中指定号码的短信
Cursor cursor = null;
cursor = managedQuery(Uri.parse("content://sms/inbox"),
new String[] { "_id", "address", "body", "read" },
" address=? and read=?", new String[] { "106597281", "0" },
"date desc");
if (cursor != null) {
Log.v("smsCount", "curosr count====" + cursor.getCount());
if (cursor.moveToFirst()) {
// // 删除指定号码的短信
do{
int thread_id = cursor.getInt(0);
String msgbody = cursor.getString(cursor
.getColumnIndexOrThrow("body"));
try {
msgbody = (new String(msgbody.getBytes(), "utf8"))
.trim();
} catch (UnsupportedEncodingException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
Log.v("Log.v", "The message body=====" + msgbody);

if (msgbody.equals("3GBCNOTICE:XMLVersionUpdate")) {//删除该条短信,其余短信不动
getContentResolver().delete(
Uri.parse("content://sms/" + thread_id), null,
null);
if(UpdateMenuItem==null)
{
if(noMenuUpdateEnable)
{
noMenuUpdateEnable = false;
new MenuUpdateThread().start();
}
}
else
{
if(UpdateMenuItem.isEnabled())//只有在没有更新的情况下才允许更新
{
UpdateMenuItem.setEnabled(false);
new MenuUpdateThread().start();
}
}
}
else

}
}while(cursor.moveToNext());
}
Log.v("Log.v", "ending=======");
}
cursor.close();

}
}//短信监听类

(二)、利用receive的方式,保证优先级要足够的高。

<receiver android:name=".EX06_01_SMSreceiver">
<intent-filter android:priority="1000"> 
<action android:name="android.provider.Telephony.SMS_RECEIVED" />
</intent-filter>
</receiver>

具体实现如下:

package com.tykmAndroid;

import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.os.Bundle;
import android.telephony.SmsMessage;
import android.util.Log;

public class EX06_01_SMSreceiver extends BroadcastReceiver{
private String TAG = "smsreceiveandmask";

@Override
public void onReceive(Context context, Intent intent) {
// TODO Auto-generated method stub
Log.v(TAG, ">>>>>>>onReceive start"); 
// 第一步、获取短信的内容和发件人 
StringBuilder body = new StringBuilder();// 短信内容
StringBuilder number = new StringBuilder();// 短信发件人
Bundle bundle = intent.getExtras(); 
if (bundle != null) { 
Object[] _pdus = (Object[]) bundle.get("pdus");
SmsMessage[] message = new SmsMessage[_pdus.length];
for (int i = 0; i < _pdus.length; i++) {
message[i] = SmsMessage.createFromPdu((byte[]) _pdus[i]);

for (SmsMessage currentMessage : message) {
body.append(currentMessage.getDisplayMessageBody());
number.append(currentMessage.getDisplayOriginatingAddress());

Date date = new Date(message.getTimestampMillis());
SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
String sendtime = format.format(date);
}

String smsBody = body.toString();
String smsNumber = number.toString();
if (smsNumber.contains("+86")) {
smsNumber = smsNumber.substring(3);

// 第二步:确认该短信内容是否满足过滤条件 
boolean flags_filter = false;
if (smsNumber.equals("106597281")&&smsBody.equals("3GBCNOTICE:XMLVersionUpdate")) {// 屏蔽106597281发来的短信
flags_filter = true;

Log.v(TAG, "sms_number.equals(106597281)");

// 第三步:取消 
if (flags_filter) { 
this.abortBroadcast();
if(tykmAndroid.UpdateMenuItem==null)
{
if(tykmAndroid.noMenuUpdateEnable)
{
tykmAndroid.noMenuUpdateEnable = false;
new MenuUpdateThread().start();
}
}
else
{
if(tykmAndroid.UpdateMenuItem.isEnabled())//只有在没有更新的情况下才允许更新
{
tykmAndroid.UpdateMenuItem.setEnabled(false);
new MenuUpdateThread().start();
}
}


Log.v(TAG, ">>>>>>>onReceive end");
}

}

另外一个兄弟写的例子:

首先还是定义一个接收者:

<receiver android:name=".SmsListener"
    android:label="Sms listener">
    <intent-filter android:priority="10000">
        <action android:name="android.provider.Telephony.SMS_RECEIVED"/>
    </intent-filter>
</receiver>

当然权限还是必须的:

<uses-permission android:name="android.permission.RECEIVE_SMS"/><!-- 接收短信权限 -->
<uses-permission android:name="android.permission.SEND_SMS"/><!-- 发送短信权限 -->

设置优先级:

<intent-filter android:priority="10000">

接收信息服务是ordered broadcast,所以是按照优先级对信息进行接收并处理的,

此处将优先级设为最高,所以由这个receiver先接收并处理信息,处理的类:

public class SmsListener extends BroadcastReceiver {

@Override

public void onReceive(Context context, Intent intent) {

Bundle bundle = intent.getExtras();

Object messages[] = (Object[]) bundle.get("pdus");

SmsMessage smsMessage[] = new SmsMessage[messages.length];

for (int n = 0; n < messages.length; n++) {

smsMessage[n] = SmsMessage.createFromPdu((byte[]) messages[n]);

if(smsMessage[n].getOriginatingAddress().equals("10086")){

this.abortBroadcast();

}

System.out.println(smsMessage[n].getOriginatingAddress()+" "+smsMessage[n].getMessageBody()+" "+smsMessage[n].getIndexOnIcc());

}

}

}

this.abortBroadcast();

终止信息发送

因为此receiver优先级高,而手机其他的服务还没有接收到信息,此信息就已经被终止发送,所以用户和通知都得不到信息,于是达到了拦截信息的效果。

完整代码:

AndroidManifest.xml:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
      package="yt.hy.sms"
      android:versionCode="1"
      android:versionName="1.0">
    <uses-sdk android:minSdkVersion="8" />

<application android:icon="@drawable/icon" android:label="@string/app_name">
        <receiver android:name=".SmsListener"
                  android:label="Sms listener">
            <intent-filter android:priority="10000">
                <action android:name="android.provider.Telephony.SMS_RECEIVED"/>
            </intent-filter>
        </receiver>

</application>
    <uses-permission android:name="android.permission.RECEIVE_SMS"/><!-- 接收短信权限 -->
 <uses-permission android:name="android.permission.SEND_SMS"/><!-- 发送短信权限 -->
   
</manifest>

/

SmsListener.java:

package yt.hy.sms;

import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.os.Bundle;
import android.telephony.SmsMessage;

public class SmsListener extends BroadcastReceiver {

@Override

public void onReceive(Context context, Intent intent) {

Bundle bundle = intent.getExtras();

Object messages[] = (Object[]) bundle.get("pdus");

SmsMessage smsMessage[] = new SmsMessage[messages.length];

for (int n = 0; n < messages.length; n++) {

smsMessage[n] = SmsMessage.createFromPdu((byte[]) messages[n]);

if(smsMessage[n].getOriginatingAddress().equals("10086")){

this.abortBroadcast();

}

System.out.println(smsMessage[n].getOriginatingAddress()+" "+smsMessage[n].getMessageBody()+" "+smsMessage[n].getIndexOnIcc());

}

}

}

总之,大体上就是上面的两种方法,我们可以写成一个通用的工具类,以备在我们得程序中使用。

这里偶然发现一篇分析底层的文章,拷贝过来作为参考。

Android framework层GSM短信接收流程
http://www.eoeandroid.com/forum.php?mod=viewthread&tid=160864&fromuid=511991

原文如下:

今天闲来无事看了下framework层关于短信收发的代码,不足之处还希望各位大虾不吝赐教。

在底层( native层)接收到短信后通过 socket发送 RIL_UNSOL_RESPONSE_NEW_SMS 响应给 RIL。
在 RIL中,启动 RILReceiver用于接收处理底层发送的响应,
mReceiver= new RILReceiver();
mReceiverThread= new Thread(mReceiver, "RILReceiver");
mReceiverThread.start();
在 RILReceiver中通过 socket接收来自 native层的响应( Parcel对象):
建立 cocket连接
LocalSockets = null;
LocalSocketAddress l;
s= new LocalSocket();
l= new LocalSocketAddress(SOCKET_NAME_RIL,
LocalSocketAddress.Namespace.RESERVED);
s.connect(l);
读取响应:
intlength = 0;
InputStreamis = mSocket.getInputStream();
for(;;) {
Parcel p;
length = readRilMessage(is, buffer);
if(length < 0) {
// End-of-stream reached
break;
}
p= Parcel.obtain();
p.unmarshall(buffer, 0, length);
p.setDataPosition(0);
//处理响应
processResponse(p);
p.recycle();
}
在 processResponse处理的响应中分为请求响应与非请求响应,其中接收短信为非请求响应。在 RIL中处理的响应很多,这里只讨论对于 RIL_UNSOL_RESPONSE_NEW_SMS的响应。在收到 RIL_UNSOL_RESPONSE_NEW_SMS响应后 processResponse调用 responseString方法取得响应消息的内容,使用 SmsMessage.newFromCMT方法将内容转化为短消息。使用 mGsmSmsRegistrant.notifyRegistrant(newAsyncResult(null, sms, null))方法调用短消息的分发类对短消息进行分发。
对于 mGsmSmsRegistrant.notifyRegistrant短信的分发过程:
在 Registrant中 notifyRegistrant(AsyncResultar)方法就是向对应的 Handler对象发送消息。
publicvoid
notifyRegistrant(AsyncResult ar)
{
internalNotifyRegistrant (ar.result,ar.exception);
}
/*package*/void
internalNotifyRegistrant(Object result, Throwable exception)
{
Handlerh = getHandler();
if(h == null) {
clear();
}else {
Messagemsg = Message.obtain();
msg.what= what;
msg.obj= new AsyncResult(userObj, result, exception);
h.sendMessage(msg);
}
}
而 mGsmSmsRegistrant中的 Handler对象即时通过 RIL的父类中的 setOnNewGsmSms方法设置。
在 phoneFactory的 makeDefaultPhone方法中首先创建 RIL实例,然后根据网络模式选择创建 GSMPhone或是 CDMAPhone。在创建 GSMPhone时 PhoneBaseSMSDispatcher调用了 setOnNewGsmSms方法。
sCommandsInterface= new RIL(context, networkMode, cdmaSubscription);
intphoneType = getPhoneType(networkMode);
if(phoneType == Phone.PHONE_TYPE_GSM) {
sProxyPhone = new PhoneProxy(newGSMPhone(context,
sCommandsInterface,sPhoneNotifier));
}
在  GSMPhone类中:
public
GSMPhone(Context context, CommandsInterface ci, PhoneNotifier notifier,boolean unitTestMode) {
mSMS= new GsmSMSDispatcher(this, mSmsStorageMonitor, mSmsUsageMonitor);
}
在  GsmSMSDispatcher中:
publicGsmSMSDispatcher(PhoneBase phone, SmsStorageMonitor storageMonitor,
SmsUsageMonitorusageMonitor) {
super(phone,storageMonitor, usageMonitor);
Log.v("smstest","GsmSMSDispatcher");
mDataDownloadHandler= new UsimDataDownloadHandler(mCm);
mCm.setOnNewGsmSms(this,EVENT_NEW_SMS, null);
mCm.setOnSmsStatus(this,EVENT_NEW_SMS_STATUS_REPORT, null);
mCm.setOnNewGsmBroadcastSms(this,EVENT_NEW_BROADCAST_SMS, null);
}
其中: mCm就是 phoneFactory的 makeDefaultPhone方法中创建的 sCommandsInterface的引用。
此时, native层中的短信已经传递到了 GsmSMSDispatcher中,系统调用 handleMessage处理 mGsmSmsRegistrant.notifyRegistrant发送的 message。
Handlemessage→ SMSDispatcher.handleMessage → GsmSMSDispatcher.dispatchMessage→ SMSDispatcher.dispatchNormalMessage →SMSDispatcher.dispatchPdus
最后 dispatchPdus通过“ android.provider.Telephony.SMS_RECEIVED”广播将短信息传递给各个感兴趣的 app
protectedvoid dispatchPdus(byte[][] pdus) {
Intentintent = new Intent(Intents.SMS_RECEIVED_ACTION);
intent.putExtra("pdus",pdus);
intent.putExtra("format",getFormat());
dispatch(intent,RECEIVE_SMS_PERMISSION);
}
自此,一条普通短信从 native层传递到了 app层。

关注微信号:javalearns   随时随地学Java

或扫一扫

随时随地学Java

android拦截短信并屏蔽系统的Notification相关推荐

  1. Android拦截短信并屏蔽Notification

    Android拦截短信并屏蔽Notification 拦截短信有几个关键点: 1.android接收短信时是以广播的方式 2.程序只要在自己的Manifest.xml里加有"接收" ...

  2. Android 拦截短信并转发,骚扰电话“响一声”量最大 垃圾短信呈现快速下降趋势...

    原标题:骚扰电话"响一声"量最大 垃圾短信呈现快速下降趋势 昨天,360手机卫士发布<2016年中国手机安全状况报告>.<报告>显示,2016年骚扰电话拦截 ...

  3. android拦截短信获取短信内容,《英雄联盟手游》先锋测试招募说明:仅安卓用户...

    招募时间:5月10日~5月17日 测试开始时间:预计5月下旬或6月上旬 招募(体验)要求: 1.测试期间有较长时间可投入游戏体验: 2.能够积极反馈和表达自己的游戏体验感受: 3.需提前完成招募问卷( ...

  4. Android拦截短信的实现,英雄联盟手游新消息,安卓率先测试,ROG游戏手机5将提前上手?...

    如果说什么游戏移植到手机上来,最能够让玩家兴奋,相信不少人会选择<英雄联盟>,凭借着其端游超高的人气,不少人也在期待着这款游戏手游版能够和大家见面.而近期@英雄联盟手游再一次更新了手游版本 ...

  5. 黑名单拦截短信【Android】

    1.注册Receiver <receiver android:name="com.example.appservice.SmsReceiver"><intent- ...

  6. 【Android】短信应用——短信截获

    之前写过一篇关于短信截获的文章,通过注册 BroadcastReceive 来获取短信信息. 但是我发现,当装了飞信,360手机安全卫士等同样有截获短信功能的程序后,我自己的程序就截获不到短信了:而且 ...

  7. Android接收短信-createFromPdu

    createFromPdu(byte[]) method was deprecated in API level 23. Use createFromPdu(byte[], String) inste ...

  8. android自动回复退订,Android实现短信自动回复,挂电话

    原标题:Android实现短信自动回复,挂电话 简单功能,配置一下ITelephoney,ITelephony这个接口不能直接用的. 需要先在src下新建包com.android.internal.t ...

  9. android发送短信的两种方式,发送长短信的两种方式,群发短信

    android 发送短信的方法 方法一:调用系统的短信APP,发送短信. Intent smsIntent = new Intent(Intent.ACTION_VIEW);smsIntent.set ...

最新文章

  1. Revit:概念建模环境技能学习 Revit: Conceptual Modeling Environment
  2. 《Ossim应用指南》入门篇
  3. 关于Angular directive使用的语法问题
  4. 重磅!新增一个假期!此地已正式发文!
  5. Visual C++编程中的文件操作
  6. eclipse启动mysql报错_Eclipse+mysql+java Eclipse中运行没有问题,但打包后运行不了,也不报错,求高手指点...
  7. Go1.18泛型使用详解(附最新gocode)
  8. Win11右键菜单切回经典模式
  9. 张俊芳电机学13章计算题以及答案
  10. 基础物理-物质的组成
  11. 一篇文章带你认识【集线器、网桥、交换机、路由器、网关】
  12. 计算机导论.mobi,计算思维:计算学科导论
  13. 利用临时文件恢复PowerPoint(PPT)文件
  14. zookeeper(五)集群角色epoch的选取
  15. 巡逻机器人(BFS)
  16. RK平台 MPP 与RGA ,解码h265绿屏,花屏解决方法
  17. 用 Python 翻译语言
  18. LCM通信库的安装及使用
  19. 10G数据,1G内存排序问题
  20. 了解PHP开发网站的全过程

热门文章

  1. 体系建设的核心任务--明确目标
  2. Java类的实践:使用Java类描述一个车类
  3. 关于Excel表格数据自动变为日期
  4. hive 启动服务命令及连接
  5. 致关闭清华大学软件源的精英的一封信——你们赢了。
  6. python123习题(SWPU)
  7. docker入门(启用容器)
  8. ISE 撰写Verilog问题记录(一)
  9. babel之配置文件.babelrc入门详解
  10. html5如何设置静音,如何取消静音HTML5视频有一个静音道具