思路:
要想挂断电话,必然会用到电话服务:getSystemService(TELEPHONY_SERVICE);
但查看源码时,发现
@Override public Object getSystemService(String name) { return mBase.getSystemService(name); }
继续查找:
Context mBase; public ContextWrapper(Context base) { mBase = base; }
接着查看Context的源码
public abstract class Context {........}
崩溃了,经常使用的Context,竟然是个抽象类,那么它的具体实现呢?
这时采用断点的方式进行查找:
随便弄个测试代码,获取到要查找的对象,在下面打个断点
Debug之后
哦了,接着就是去找到ContextImpl类了,这个时候用到了一个搜索神器:search everything,很强大,只要你的电脑里有什么,就可以搜到什么。
本人已经在CSDN上传了该工具,这里是链接,想下的朋友可以去瞅瞅http://download.csdn.net/detail/xushuaic/5346268
右键open path,就可以找到了。
打开ContextImpl,接着又开始犯愁了,该从哪找起呢,两千多行代码,疯了,
既然咱们是要找Service的,只是跟踪的时候跟到了ContextImpl,那么它一定有getSystemService()的方法,Ctrl+F查找,一下就定位到来,往下看几行,一大串类似于下面的代码
private AccountManager getAccountManager() { synchronized (mSync) { if (mAccountManager == null) { IBinder b = ServiceManager.getService(ACCOUNT_SERVICE); IAccountManager service = IAccountManager.Stub.asInterface(b); mAccountManager = new AccountManager(this, service); } return mAccountManager; } }
从这就可以推断出,咱要想获取服务,就得使用ServiceManager,那么哦了,回到我们自己的代码中调用: ServiceManager.getService(TELEPHONY_SERVICE);
发现报错,原因是,该类是放在android.os.ServiceManager包中的,对于一些重要的服务,Android工程师是不希望我们自己直接拿到的,不让咱拿咱偏要拿,只好使用暴力了,这时候二话不说,上反射,使用反射得到了getService方法之后,根据上面的代码,照葫芦画瓢,很容易能写出下面的代码:
Class ServiceManager = getClass().getClassLoader().loadClass( "android.os.ServiceManager"); Method getServiceMethod = ServiceManager.getMethod( "getService", new String[] {String.class}); IBinder ibinder = (IBinder) getServiceMethod.invoke(null, new String[] { TELEPHONY_SERVICE }); //TELEPHONY 比对 ACCOUNT_SERVICE,也可以去找到ITelephony ITelephony itelephony = ITelephony.Stub.asInterface(ibinder); itelephony.endCall();
如果对AIDL不太了解的朋友,可以看看笔者的另一篇博客http://blog.csdn.net/xushuaic/article/details/8559022
通过ITelephony.Stub.asInterface 就能看出一定是AIDL的通信,那么必须使用到AIDL协议,而且该协议文件的名称就是ITelephony,那么easy,Search Everything,又哦了,新建一个包,将ITelephony.aidl放进去,本以为这样就哦了,谁知道又报错,找到红叉叉,鼠标移上去
没问题,Search一把
找到,Ctrl+C、Ctrl+V一把,导入包,终于没有红叉叉了,大功告成
注意包的创建,如果你把两个两个文件放错包了,不好意思,eclipse可不客气
下面是结束电话的服务,只要在另一个Activity中开启该服务,那么就可以自动地挂断指定的电话了,这里有的朋友一定会发现,这不就是黑名单拦截么,哦了,黑名单拦截就是这样的原理,只要提供一个黑名单的数据库,当来电的时候,到数据库里去查找然后进行判断就可以
记得权限:
<!-- 监听电话状态 --> <uses-permission android:name="android.permission.READ_PHONE_STATE" />
注册服务
<service android:name="com.example.service.EndCallService" ></service>
结束电话的服务
import java.lang.reflect.InvocationTargetException; import java.lang.reflect.Method; import com.android.internal.telephony.ITelephony; import android.app.Service; import android.content.Intent; import android.os.Handler; import android.os.IBinder; import android.os.RemoteException; import android.provider.CallLog; import android.telephony.PhoneStateListener; import android.telephony.TelephonyManager; import android.util.Log; public class EndCallService extends Service { private static final String TAG = "EndCallService"; private TelephonyManager telephonyManager; private MyPhoneListener listener; private Intent callSmsSafeService; @Override public IBinder onBind(Intent intent) { return null; } @Override public void onCreate() { Log.i(TAG,"启动服务"); // 开启电话拦截服务 telephonyManager = (TelephonyManager) getSystemService(TELEPHONY_SERVICE); listener = new MyPhoneListener(); telephonyManager.listen(listener, PhoneStateListener.LISTEN_CALL_STATE); super.onCreate(); } @Override public void onDestroy() { // 取消电话监听服务 telephonyManager.listen(listener, PhoneStateListener.LISTEN_NONE); listener = null; super.onDestroy(); } // 自定义电话状态监听 class MyPhoneListener extends PhoneStateListener { @Override public void onCallStateChanged(int state, String incomingNumber) { super.onCallStateChanged(state, incomingNumber); switch (state) { case TelephonyManager.CALL_STATE_RINGING: Log.i(TAG, "电话到来" + incomingNumber); if ("10086".equals(incomingNumber)) {// 拦截电话 Log.i(TAG, "挂断电话" + incomingNumber); endCall(); break; } } } // 挂断电话功能的API隐藏了,无法直接获取服务,通过反射 private void endCall() { // ServiceManager.getService(TELEPHONY_SERVICE); getSystemService(TELEPHONY_SERVICE); try { Class ServiceManager = getClass().getClassLoader().loadClass( "android.os.ServiceManager"); Method getServiceMethod = ServiceManager.getMethod( "getService", String.class); IBinder ibinder = (IBinder) getServiceMethod.invoke(null, new String[] { TELEPHONY_SERVICE }); ITelephony iTelephony = ITelephony.Stub.asInterface(ibinder); iTelephony.endCall(); Log.i(TAG, "end call-------------"); } catch (ClassNotFoundException e) { e.printStackTrace(); } catch (NoSuchMethodException e) { e.printStackTrace(); } catch (IllegalArgumentException e) { e.printStackTrace(); } catch (IllegalAccessException e) { e.printStackTrace(); } catch (InvocationTargetException e) { e.printStackTrace(); } catch (RemoteException e) { e.printStackTrace(); } } } }