前段时间在开发一款应用,里面涉及到一个来电拦截的功能,于是乎就开始了对于来电如何拦截进行了探索,最后总结出了实现来电拦截的两种方法,并且经过实际的真机验证,在对比两种可以实现的方法,我们找出了其中较优的一种实现。

 对于来电如何拦截,我们想象一下要拦截来电,首先我们的必须知道,有没有电话打进来,只有确定来了电话,我们才好去拦截,就像战斗中拦截导弹一样,没有雷达去捕获来袭导弹的信息,那就没法拦截掉来袭的导弹。这就是为什么某些导弹要使用隐形技术,目的就是为了尽量避开雷达的捕捉,增加拦截的难度。同样在电话拦截里面,我们也必须要有这样的一部“雷达”来捕捉来电。那么在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手机安全卫士拦截不知道能不能做到这样完全悄无声息的拦截,这个我没有测试,若能的话不知道他们是什么技术解决的。