为了更好地理解这个Demo,我先向大家介绍一下需求与功能。

需求:

每天都会有很多无聊的电话,比如推销商品等,占用我们大量时间不说,有时候还会打乱我们的思路,扰乱我们的正常生活。所以实现一个对某些号码(比如陌生号码,指定号码/黑名单等)进行拦截以避免受到骚扰,是很有现实用途的。

为了避免程序过分复杂,造成不易学习的麻烦我在这里只实现“如果来电号码没在联系人中,则进行挂断,并存入xml文件(SharedPreferences)中,并在首页显示”,以期达到抛砖引玉的效果。

其实在android在1.1版本后就已经把Phone类的相关API给隐藏起来了,想要用代码实现挂断电话的功能,就必须通过AIDL才行,然后利用反射来使用其方法。

第一步:在程序中新建一个包,包名必须为:com.android.internal.telephony,因为要使用aidl。

第二步:在这个包里面新建一个名为ITelephony.aidl的文件,然后在文件里面写入代码:

  1. package com.android.internal.telephony;
  2. interface ITelephony{
  3. boolean endCall();
  4. void answerRingingCall();
  5. }

然后是要监听电话状态,当来电时,检测来电号码是否符合拦截标准(这个拦截标准是我们自己定的,你可以拦截指定号码如实现一个黑名单的功能,我们在这里拦截所有不在联系人里的号码,并把此号码存入文件,以方便在首页显示),代码如下:

  1. import java.lang.reflect.Method;
  2. import java.util.ArrayList;
  3. import com.android.internal.telephony.ITelephony;
  4. import android.app.Service;
  5. import android.content.BroadcastReceiver;
  6. import android.content.ContentResolver;
  7. import android.content.Context;
  8. import android.content.Intent;
  9. import android.content.SharedPreferences;
  10. import android.content.SharedPreferences.Editor;
  11. import android.database.Cursor;
  12. import android.provider.ContactsContract;
  13. import android.telephony.TelephonyManager;
  14. import android.util.Log;
  15. public class PhoneStatReceiver extends BroadcastReceiver{
  16. String TAG = "tag";
  17. TelephonyManager telMgr;
  18. @Override
  19. public void onReceive(Context context, Intent intent) {
  20. telMgr = (TelephonyManager) context.getSystemService(Service.TELEPHONY_SERVICE);
  21. switch (telMgr.getCallState()) {
  22. case TelephonyManager.CALL_STATE_RINGING:
  23. String number = intent.getStringExtra(TelephonyManager.EXTRA_INCOMING_NUMBER);
  24. Log.v(TAG,"number:"+number);
  25. if (!getPhoneNum(context).contains(number)) {
  26. SharedPreferences phonenumSP = context.getSharedPreferences("in_phone_num", Context.MODE_PRIVATE);
  27. Editor editor = phonenumSP.edit();
  28. editor.putString(number,number);
  29. editor.commit();
  30. endCall();
  31. }
  32. break;
  33. case TelephonyManager.CALL_STATE_OFFHOOK:
  34. break;
  35. case TelephonyManager.CALL_STATE_IDLE:
  36. break;
  37. }
  38. }
  39. /**
  40. * 挂断电话
  41. */
  42. private void endCall()
  43. {
  44. Class<TelephonyManager> c = TelephonyManager.class;
  45. try
  46. {
  47. Method getITelephonyMethod = c.getDeclaredMethod("getITelephony", (Class[]) null);
  48. getITelephonyMethod.setAccessible(true);
  49. ITelephony iTelephony = null;
  50. Log.e(TAG, "End call.");
  51. iTelephony = (ITelephony) getITelephonyMethod.invoke(telMgr, (Object[]) null);
  52. iTelephony.endCall();
  53. }
  54. catch (Exception e)
  55. {
  56. Log.e(TAG, "Fail to answer ring call.", e);
  57. }
  58. }
  59. private ArrayList<String>  getPhoneNum(Context context) {
  60. ArrayList<String> numList = new ArrayList<String>();
  61. //得到ContentResolver对象
  62. ContentResolver cr = context.getContentResolver();
  63. //取得电话本中开始一项的光标
  64. Cursor cursor = cr.query(ContactsContract.Contacts.CONTENT_URI, null, null, null, null);
  65. while (cursor.moveToNext())
  66. {
  67. // 取得联系人ID
  68. String contactId = cursor.getString(cursor.getColumnIndex(ContactsContract.Contacts._ID));
  69. Cursor phone = cr.query(ContactsContract.CommonDataKinds.Phone.CONTENT_URI, null,
  70. ContactsContract.CommonDataKinds.Phone.CONTACT_ID + " = " + contactId, null, null);
  71. // 取得电话号码(可能存在多个号码)
  72. while (phone.moveToNext())
  73. {
  74. String strPhoneNumber = phone.getString(phone.getColumnIndex(ContactsContract.CommonDataKinds.Phone.NUMBER));
  75. numList.add(strPhoneNumber);
  76. Log.v("tag","strPhoneNumber:"+strPhoneNumber);
  77. }
  78. phone.close();
  79. }
  80. cursor.close();
  81. return numList;
  82. }
  83. }

这里我们要注意以下几点:

1.PhoneStatReceiver一定要在清单文件(AndroidManifest.xml)中注册。

2.一定要添加权限

AndroidManifest文件如下:

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <manifest xmlns:android="http://schemas.android.com/apk/res/android"
  3. package="com.xxxx.xxxx"
  4. android:versionCode="1"
  5. android:versionName="1.0" >
  6. <uses-sdk android:minSdkVersion="8" />
  7. <!-- 挂断手机的权限 -->
  8. <uses-permission android:name="android.permission.CALL_PHONE"/>
  9. <!-- 读取手机状态的权限 -->
  10. <uses-permission android:name="android.permission.READ_PHONE_STATE" />
  11. <!-- 读content的权限 -->
  12. <uses-permission android:name="android.permission.READ_CONTACTS" />
  13. <application
  14. android:icon="@drawable/ic_launcher"
  15. android:label="@string/app_name" >
  16. <activity
  17. android:name=".MainActivity"
  18. android:label="@string/app_name" >
  19. <intent-filter>
  20. <action android:name="android.intent.action.MAIN" />
  21. <category android:name="android.intent.category.LAUNCHER" />
  22. </intent-filter>
  23. </activity>
  24. <!-- 注册监听手机状态 -->
  25. <receiver android:name=".PhoneStatReceiver">
  26. <intent-filter android:priority="1000" >
  27. <action android:name="android.intent.action.PHONE_STATE" />
  28. </intent-filter>
  29. </receiver>
  30. </application>
  31. </manifest>

其实最到这里,整个拦截功能就已经实现了,但是呢,我们的首页也不能让它光秃秃的显示个Hello World!吧。所以,在MainActivity中,再给大家加点料,就是在listView中显示所有已经被拦截的电话号码,代码如下:

  1. import java.util.Map;
  2. import android.app.ListActivity;
  3. import android.content.Context;
  4. import android.content.SharedPreferences;
  5. import android.os.Bundle;
  6. import android.util.Log;
  7. import android.widget.ArrayAdapter;
  8. public class MainActivity extends ListActivity {
  9. @Override
  10. public void onCreate(Bundle savedInstanceState) {
  11. super.onCreate(savedInstanceState);
  12. SharedPreferences phonenumSP = getSharedPreferences("in_phone_num", Context.MODE_PRIVATE);
  13. Map map = phonenumSP.getAll();
  14. Object[] array = map.keySet().toArray();
  15. Log.v("tag",map.toString()+map.size());
  16. ArrayAdapter adapter = new ArrayAdapter(this, android.R.layout.simple_list_item_1,array);
  17. setListAdapter(adapter);
  18. }
  19. }

好了,整个项目就完成了,我们可以拦截骚扰电话了,这只是一个小例子,你可以添加一些控制功能以更加人性化,比如开启和关闭拦截,可选的拦截时间段,给ListView添加点击事件使用户可以把拦截到的电话添加到通讯录等功能。

不过,这里只是完成了对来电的拦截功能,未对外拨电话进行拦截,下篇博客再写吧!

参考:http://blog.163.com/wu_zefeng/blog/static/1826291752011312114420975/

http://www.cotrun.net/blog/1572.html

Android 代码实现来电拦截的更多相关文章

  1. Android实战_来电拦截专家

    1 项目演示: 2 代码演示: 1)MainActivity类代码: MainActivity类代码: package com.example.phoneinteceptor_one;import j ...

  2. 实现android手机来电拦截系统页面弹出自定义页面特效

    如何实现android手机来电拦截系统页面弹出自定义页面特效, 首先:    我们需要注册一个监听来电的广播PhoneStateReceiver 类:其次:    在onReceive里面我们获取an ...

  3. Android监听来电和去电

    要监听android打电话和接电话,只需下面2步骤1.第一步,写一个Receiver继承自BroadcastReceiver import android.app.Service; import an ...

  4. 下载最新Android代码的方法

    之前我是去Android官方网站下载最新Android代码,但是这种方法需要FQ,而且有时候FQ又不太方便,今天我发现一个不错的网站,是清华大学搞的,跟Android官方的代码基本保持同步,而且下载方 ...

  5. Android代码混淆官方实现方法

    首先查看一下 “project.properties” 这个文件: # This file is automatically generated by Android Tools.# Do not m ...

  6. 编写高效的Android代码

    编写高效的Android代码 毫无疑问,基于Android平台的设备一定是嵌入式设备.现代的手持设备不仅仅是一部电话那么简单,它还是一个小型的手持电脑,但是,即使是最快的最高端的手持设备也远远比不上一 ...

  7. Android代码内存优化建议-OnTrimMemory优化

    原文  http://androidperformance.com/2015/07/20/Android代码内存优化建议-OnTrimMemory优化/ OnTrimMemory 回调是 Androi ...

  8. Android代码混淆和项目宣布步骤记录器

    原本放在一起Android项目与发布的文件相混淆.我突然想到,为什么不写博客,分享.有这篇文章的情况下,. Android代码混淆及项目公布步骤记录 一.清理代码中的调试信息,如Log.System. ...

  9. Android 代码混淆 混淆方案

    本篇文章:自己在混淆的时候整理出比较全面的混淆方法,比较实用,自己走过的坑,淌出来的路.请大家不要再走回头路,可能只要我们代码加混淆,一点不对就会导致项目运行崩溃等后果,有许多人发现没有打包运行好好地 ...

随机推荐

  1. 【SpringBoot】关闭HttpClient无用日志

    环境: SpringBoot pom依赖了apache.commons.HttpClient: <!--httpclient--> <dependency> <group ...

  2. 【Leetcode】583. Delete Operation for Two Strings

    583. Delete Operation for Two Strings Given two words word1 and word2, find the minimum number of st ...

  3. 检验Xcode是否被改动过的简单方法,不妨试试!!!

    检验Xcode是否被改动过的简单方法,不妨试试!!!       在终端系统上运行以下命令启用检测: spctl --assess --verbose /Applications/Xcode.app  ...

  4. django配置templates、static、media和连接mysql数据库

    1.模板文件 # =======templates配置======= if os.path.exists(os.path.join(BASE_DIR, 'templates')) is False: ...

  5. 【UOJ 80】 二分图最大权匹配

    [分析] 之前打的那种KM会TLE... why??明明说n^3的啊? #include<cstdio> #include<cstdlib> #include<cstri ...

  6. Codeforces 521 E cycling city

    cf的一道题,非常有意思,题目是问图中是否存在两个点,使得这两个点之间有三条路径,而且三条路径没有公共点. 其实就是判断一下是否为仙人掌就行了,如果不是仙人掌的话肯定就存在,题目难在输出路径上,改了半 ...

  7. Codeforces 1129 E.Legendary Tree

    Codeforces 1129 E.Legendary Tree 解题思路: 这题好厉害,我来复读一下官方题解,顺便补充几句. 首先,可以通过询问 \(n-1​\) 次 \((S=\{1\},T=\{ ...

  8. [NOI2015]寿司晚宴 --- 状压DP

    [NOI2015]寿司晚宴 题目描述 为了庆祝NOI的成功开幕,主办方为大家准备了一场寿司晚宴. 小G和小W作为参加NOI的选手,也被邀请参加了寿司晚宴. 在晚宴上,主办方为大家提供了n−1种不同的寿 ...

  9. Mac 显示隐藏的文件

    要显示隐藏文件: 在终端中输入代码:defaults write com.apple.finder AppleShowAllFiles -bool true 隐藏文件: 在终端输入代码:default ...

  10. Educational Codeforces Round 11 E. Different Subsets For All Tuples 动态规划

    E. Different Subsets For All Tuples 题目连接: http://www.codeforces.com/contest/660/problem/E Descriptio ...