Android 来电拦截的开发实现

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

    对于来电如何拦截,我们想象一下要拦截来电,首先我们的必须知道,有没有电话打进来,只有确定来了电话,我们才好去拦截,就像战斗中拦截导弹一样,没有雷达去捕获来袭导弹的信息,那就没法拦截掉来袭的导弹。这就是为什么某些导弹要使用隐形技术,目的就是为了尽量避开雷达的捕捉,增加拦截的难度。同样在电话拦截里面,我们也必须要有这样的一部“雷达”来捕捉来电。那么在Android里面哪些类可以实现这种雷达的效果呢?

    第一种是PhoneStateListener,手机状态监听器,该类可以监听手机的各种状态,包括服务的状态、信号强度、消息等待指示(语音信箱)、通话转移、呼叫状态、设备单元位置、数据连接状态、数据流量方向等,我们正是利用了它来实现对于来电的监听,如下代码是我们继承PhoneStateListener的一个类的定义

class MyPhoneStateListener extends PhoneStateListener{ 

        @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()这个方法中,我们可以获取到来电的号码,也就是上面所说的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);         

                  Log.i(TAG, "call OUT:"+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; 

                  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();

                            }                    

                  //再恢复正常铃声    

            mAudioManager.setRingerMode(AudioManager.RINGER_MODE_NORMAL);                 

                 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文件导入到这个包。ITelephony.aidl文件下载链接:http://pan.baidu.com/share/link?shareid=438259&uk=421763582

    本次讲解到此结束,希望你看完本文能有所收获。这里说明一下以上讲解所用到的代码并不是完整的,是我们实现中用到的主要部分。还要说明的一个问题就是我们实现后,会发现在模拟器上可以直接挂断来电。而在真机上虽然能拦截掉来电但是,并不能直接挂断,而是将电话转移挂起,在拨打方,我们会听到:“你所拨打的电话正在通话中,请稍候再拨!(然后是英文的提示)”,这时我们的拦截方其实已经拦截掉来电了。还有真机在拦截之前会有“嘟”的一声提示,不能做到完全悄无声息的拦截。这个问题目前还没找到解决的好方法。不过360手机安全卫士拦截不知道能不能做到这样完全悄无声息的拦截,这个我没有测试,若能的话不知道他们是什么技术解决的。不过我也在网上查了,有人说这个是第三方软件没办法解决的,而要靠移动通讯服务商!若有哪位朋友有好方法可以解决这个问题,可以和大家分享一下!其他问题我们下次再讨论。若有问题,可以给我留言!希望你学习工作愉快!

转载于:https://www.cnblogs.com/wangdacui/p/3824112.html

  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值