前段时间在开发一款应用,里面涉及到一个来电拦截的功能,于是乎就开始了对于来电如何拦截进行了探索,最后总结出了实现来电拦截的两种方法,并且经过实际的真机验证,在对比两种可以实现的方法,我们找出了其中较优的一种实现。
对于来电如何拦截,我们想象一下要拦截来电,首先我们的必须知道,有没有电话打进来,只有确定来了电话,我们才好去拦截,就像战斗中拦截导弹一样,没有雷达去捕获来袭导弹的信息,那就没法拦截掉来袭的导弹。这就是为什么某些导弹要使用隐形技术,目的就是为了尽量避开雷达的捕捉,增加拦截的难度。同样在电话拦截里面,我们也必须要有这样的一部“雷达”来捕捉来电。那么在Android里面哪些类可以实现这种雷达的效果呢?
第一种是PhoneStateListener,手机状态监听器,该类可以监听手机的各种状态,包括服务的状态、信号强度、消息等待指示(语音信箱)、通话转移、呼叫状态、设备单元位置、数据连接状态、数据流量方向等,我们正是利用了它来实现对于来电的监听,如下代码是我们继承PhoneStateListener的一个类的定义
class MyPhoneStateListener extendsPhoneStateListener{
@Override
public void onCallStateChanged(int state, String incomingNumber){
switch (state) {
case TelephonyManager.CALL_STATE_IDLE:
result+="手机空闲起来了 ";
break;
case TelephonyManager.CALL_STATE_RINGING:
result+=" 手机铃声响了,来电号码:"+incomingNumber;
break;
case TelephonyManager.CALL_STATE_OFFHOOK:
result+="电话被挂起了 ";
default:
break;
}
super.onCallStateChanged(state,incomingNumber);
}
}
在我们复写onCallStateChanged()这个方法中,我们可以获取到来电的号码,也就是上面所说的StringincomingNumber。当然我们要读取手机的状态也需要在AndroidManifest.xml文件中申明以下权限:
其中来电状态即是:TelephonyManager.CALL_STATE_RINGING,我们要拦截来电即可在这个状态下实现我们的拦截操作stopCall(),该方法将在下文实现。对于MyPhoneStateListener具体的使用方法即是在我们要拦截来电的Activity里面使用以下代码:
//获取电话服务
TelephonyManager
manager = (TelephonyManager)this.getSystemService(TELEPHONY_SERVICE);
//设置PhoneStateListener中的listen_call_state状态进行监听
manager.listen(newMyPhoneStateListener(),PhoneStateListener.LISTEN_CALL_STATE);
这样监听器设置完成后,我们就可以监听来电状态,然后实施拦截。好了,第一种拦截方式实现基本完成。
但是我们思考一下,对于这种方法虽然实现了对于来电的拦截。我们退出这个Activity再试试好像不怎么有效。我想大多数人希望的是,我们的这个来电拦截应用,可以在用户设置开启以后,然后退出了我们的应用还可以正常拦截到来电。说到这时,有的同学或许会说可以在后台开一个进程或Service,嗯,这样是可以实现的。但我想这样的方法虽然可以实现来电,但是有一点不怎么完美。这里说点题外话,有用过Android手机的同学或许有这样的经历,就是我们不论清理了多少次内存,清理完成不一会,手机的内存又被占用了百分七八十。这就是后台有很多服务进程自启动了,这样不仅消耗了系统资源,同时也消耗电源。这样手机待机时间就直线下降了,这就是很多人在抱怨电池不够用。有人或许会说可以用软件来禁止这些后台的软件,那是站在用户的角度。站在开发者的角度,那假如我们应用开启的Service不幸被用户强行禁用了或kill掉了,此时再有来电,我想就没办法拦截了!这时用户会觉得:啊,那个来电拦截的应用怎么没用啊?好吧,卸掉……
为避免这个问题,我们可以换个思路,这里我们想到第二种实现方法,使用BroadcastReceiver,通过获取来电去电的广播,从而进行拦截操作,这样就避免上面说的一系列问题。那怎么弄呢?好我们先写一个继承BroadcastReceiver的类PhoneStatReceiver,如下代码所示,然后复写他的onReceive()方法
public class PhoneStatReceiver extendsBroadcastReceiver{
private static final String TAG ="PhoneStatReceiver";
private static boolean incomingFlag =false;
private static Stringincoming_number = null;
private Context mycon;
@Override
public void onReceive(Context context, Intentintent) {
//如果是去电
if(intent.getAction().equals(Intent.ACTION_NEW_OUTGOING_CALL)){
incomingFlag = false;
String phoneNumber =intent.getStringExtra(Intent.EXTRA_PHONE_NUMBER);
Log.i(TAG, "callOUT:"+phoneNumber);
}else{
//如果是来电
TelephonyManager tm =
(TelephonyManager)context.getSystemService(Service.TELEPHONY_SERVICE);
switch (tm.getCallState()) {
case TelephonyManager.CALL_STATE_RINGING:
incomingFlag = true;//标识当前是来电
incoming_number=intent.getStringExtra("incoming_number");
if(setting_nodesrap==1) {
stopCall(incoming_number);//拦截来电
abortBroadcast();//截断广播
}
Log.i(TAG, "RINGING :"+ incoming_number);
break;
caseTelephonyManager.CALL_STATE_OFFHOOK:
if(incomingFlag){
Log.i(TAG, "incoming ACCEPT :"+incoming_number);
}
break;
caseTelephonyManager.CALL_STATE_IDLE:
if(incomingFlag){
Log.i(TAG, "incomingIDLE");
}
break;
}
}
}
}
在此我们还要记得在AndroidManifest.xml配置文件中加入我们的PhoneStatReceiver注册代码以及权限申明代码,如下:
至于在其中实现拦截操作的方法stopCall()具体源码大家可以加入自己的操作,以下是我实现的一种:
//电话拦截
public void stop(String incoming_number){
AudioManager mAudioManager = (AudioManager)mycon.getSystemService(Context.AUDIO_SERVICE);
mAudioManager.setRingerMode(AudioManager.RINGER_MODE_SILENT);//静音处理
iTelephony = getITelephony(mycon); //获取电话接口
try {
iTelephony.endCall();//结束电话
}catch (RemoteException e){
e.printStackTrace();
}
//再恢复正常铃声
mAudioManager.setRingerMode(AudioManager.RINGER_MODE_NORMAL);
Log.i("----", "来电 :"+ incoming_number);
}
在其中用到了ITelephony类的对象iTelephony。其中ITelephony是Android系统Phone类中TelephonyManager提供给上层应用程序用户与telephony进行操作交互的接口。但是在Android1.5以后的版本系统已经把Phone类给隐藏起来了,想要用代码实现挂断电话,就必须通过AIDL(Android InterfaceDefinition Language,即Android接口定义语言)才行。
我们找到ITelephone的接口还要加入以下方法getITelephony()获取接口:
private static ITelephony getITelephony(Context context) {
TelephonyManager mTelephonyManager = (TelephonyManager) context
.getSystemService(Context.TELEPHONY_SERVICE);
Class c = TelephonyManager.class;
Method getITelephonyMethod = null;
try {
getITelephonyMethod = c.getDeclaredMethod("getITelephony",
(Class[]) null); // 获取声明的方法
getITelephonyMethod.setAccessible(true);
} catch (SecurityException e) {
e.printStackTrace();
} catch (NoSuchMethodException e) {
e.printStackTrace();
}
try {
ITelephony iTelephony = (ITelephony)getITelephonyMethod.invoke(
mTelephonyManager, (Object[]) null); // 获取实例
return iTelephony;
} catch (Exception e) {
e.printStackTrace();
}
return iTelephony;
}
获取ITelephone的接口的方法在我们的工程文件的src目录下建一个package命名为com.android.internal.telephony,将我们的ITelephony.aidl文件导入到这个包。ITelephony.aidl文件下载链接:http://yunpan.cn/cZLedkSU6vzGr 访问密码 16c8
本次讲解到此结束,希望你看完本文能有所收获。这里说明一下以上讲解所用到的代码并不是完整的,是我们实现中用到的主要部分。还要说明的一个问题就是我们实现后,会发现在模拟器上可以直接挂断来电。而在真机上虽然能拦截掉来电但是,并不能直接挂断,而是将电话转移挂起,在拨打方,我们会听到:“你所拨打的电话正在通话中,请稍候再拨!(然后是英文的提示)”,这时我们的拦截方其实已经拦截掉来电了。还有真机在拦截之前会有“嘟”的一声提示,不能做到完全悄无声息的拦截。这个问题目前还没找到解决的好方法。不过360手机安全卫士拦截不知道能不能做到这样完全悄无声息的拦截,这个我没有测试,若能的话不知道他们是什么技术解决的。不过我也在网上查了,有人说这个是第三方软件没办法解决的,而要靠移动通讯服务商!若有哪位朋友有好方法可以解决这个问题,可以和大家分享一下!其他问题我们下次再讨论。若有问题,可以给我留言!希望你学习工作愉快!