前段时间在开发一款应用,里面涉及到一个来电拦截的功能,于是乎就开始了对于来电如何拦截进行了探索,最后总结出了实现来电拦截的两种方法,并且经过实际的真机验证,在对比两种可以实现的方法,我们找出了其中较优的一种实现。
对于来电如何拦截,我们想象一下要拦截来电,首先我们的必须知道,有没有电话打进来,只有确定来了电话,我们才好去拦截,就像战斗中拦截导弹一样,没有雷达去捕获来袭导弹的信息,那就没法拦截掉来袭的导弹。这就是为什么某些导弹要使用隐形技术,目的就是为了尽量避开雷达的捕捉,增加拦截的难度。同样在电话拦截里面,我们也必须要有这样的一部“雷达”来捕捉来电。那么在Android里面哪些类可以实现这种雷达的效果呢?
第一种是PhoneStateListener,手机状态监听器,该类可以监听手机的各种状态,包括服务的状态、信号强度、消息等待指示(语音信箱)、通话转移、呼叫状态、设备单元位置、数据连接状态、数据流量方向等,我们正是利用了它来实现对于来电的监听,如下代码是我们继承PhoneStateListener的一个类的定义
class MyPhoneStateListener extends PhoneStateListener{
@Override
public void onCallStateChanged(int state, String incomingNumber) {
switch (state) {
case TelephonyManager.CALL_STATE_IDLE:
";
break;
case TelephonyManager.CALL_STATE_RINGING:
手机铃声响了,来电号码:"+incomingNumber;
break;
case TelephonyManager.CALL_STATE_OFFHOOK:
result+=" 电话被挂起了 ";
default:
break;
}
super.onCallStateChanged(state, incomingNumber);
}
}
在我们复写onCallStateChanged()这个方法中,我们可以获取到来电的号码,也就是上面所说的String incomingNumber。当然我们要读取手机的状态也需要在AndroidManifest.xml文件中申明以下权限:
其中来电状态即是:TelephonyManager.CALL_STATE_RINGING,我们要拦截来电即可在这个状态下实现我们的拦截操作stopCall(),该方法将在下文实现。对于MyPhoneStateListener具体的使用方法即是在我们要拦截来电的Activity里面使用以下代码:
//获取电话服务
TelephonyManager
manager = (TelephonyManager) this.getSystemService(TELEPHONY_SERVICE);
// 设置PhoneStateListener中的listen_call_state状态进行监听
manager.listen(new MyPhoneStateListener(),PhoneStateListener.LISTEN_CALL_STATE);
这样监听器设置完成后,我们就可以监听来电状态,然后实施拦截。好了,第一种拦截方式实现基本完成。
但是我们思考一下,对于这种方法虽然实现了对于来电的拦截。我们退出这个Activity再试试好像不怎么有效。我想大多数人希望的是,我们的这个来电拦截应用,可以在用户设置开启以后,然后退出了我们的应用还可以正常拦截到来电。说到这时,有的同学或许会说可以在后台开一个进程或Service,嗯,这样是可以实现的。但我想这样的方法虽然可以实现来电,但是有一点不怎么完美。这里说点题外话,有用过Android手机的同学或许有这样的经历,就是我们不论清理了多少次内存,清理完成不一会,手机的内存又被占用了百分七八十。这就是后台有很多服务进程自启动了,这样不仅消耗了系统资源,同时也消耗电源。这样手机待机时间就直线下降了,这就是很多人在抱怨电池不够用。有人或许会说可以用软件来禁止这些后台的软件,那是站在用户的角度。站在开发者的角度,那假如我们应用开启的Service不幸被用户强行禁用了或kill掉了,此时再有来电,我想就没办法拦截了!这时用户会觉得:啊,那个来电拦截的应用怎么没用啊?好吧,卸掉……
为避免这个问题,我们可以换个思路,这里我们想到第二种实现方法,使用Broadcast Receiver,通过获取来电去电的广播,从而进行拦截操作,这样就避免上面说的一系列问题。那怎么弄呢?好我们先写一个继承BroadcastReceiver的类PhoneStatReceiver,如下代码所示,然后复写他的onReceive()方法
public class PhoneStatReceiver extends BroadcastReceiver{
private static final String TAG = "PhoneStatReceiver";
private static boolean incomingFlag = false;
private static String incoming_number = null;
private Context mycon;
@Override
public void onReceive(Context context, Intent intent) {
//如果是去电
if(intent.getAction().equals(Intent.ACTION_NEW_OUTGOING_CALL)){
incomingFlag = false;
String phoneNumber = intent.getStringExtra(Intent.EXTRA_PHONE_NUMBER);
}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");
{
stopCall(incoming_number);//拦截来电
abortBroadcast();//截断广播
}
Log.i(TAG, "RINGING :"+ incoming_number);
break;
case TelephonyManager.CALL_STATE_OFFHOOK:
if(incomingFlag){
Log.i(TAG, "incoming ACCEPT :"+ incoming_number);
}
break;
case TelephonyManager.CALL_STATE_IDLE:
if(incomingFlag){
Log.i(TAG, "incoming IDLE");
}
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();
}
//再恢复正常铃声
Log.i("----", "来电 :"+ incoming_number);
}
在其中用到了ITelephony 类的对象iTelephony。其中ITelephony是Android系统Phone类中TelephonyManager提供给上层应用程序用户与telephony进行操作交互的接口。但是在Android 1.5以后的版本系统已经把Phone类给隐藏起来了,想要用代码实现挂断电话,就必须通过AIDL(Android Interface Definition 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文件导入到这个包。
本次讲解到此结束,希望你看完本文能有所收获。这里说明一下以上讲解所用到的代码并不是完整的,是我们实现中用到的主要部分。还要说明的一个问题就是我们实现后,会发现在模拟器上可以直接挂断来电。而在真机上虽然能拦截掉来电但是,并不能直接挂断,而是将电话转移挂起,在拨打方,我们会听到:“你所拨打的电话正在通话中,请稍候再拨!(然后是英文的提示)”,这时我们的拦截方其实已经拦截掉来电了。还有真机在拦截之前会有“嘟”的一声提示,不能做到完全悄无声息的拦截。这个问题目前还没找到解决的好方法。不过360手机安全卫士拦截不知道能不能做到这样完全悄无声息的拦截,这个我没有测试,若能的话不知道他们是什么技术解决的。