Android应用开发基础之七:广播与服务(一)
广播
- 广播的概念
- 现实:电台通过发送广播发布消息,买个收音机,就能收听
- Android:系统在产生某个事件时发送广播,应用程序使用广播接收者接收这个广播,就知道系统产生了什么事件。 Android系统在运行的过程中,会产生很多事件,比如开机、电量改变、收发短信、拨打电话、屏幕解锁
IP拨号器
原理:接收拨打电话的广播,修改广播内携带的电话号码 * 定义广播接收者接收打电话广播
public class CallReceiver extends BroadcastReceiver {
//当广播接收者接收到广播时,此方法会调用
@Override
public void onReceive(Context context, Intent intent) {
//拿到用户拨打的号码
String number = getResultData();
//修改广播内的号码
setResultData("17951" + number);
}
}
-
在清单文件中定义该广播接收者接收的广播类型
<receiver android:name="com.ibky.ipdialer.CallReceiver"> <intent-filter > <action android:name="android.intent.action.NEW_OUTGOING_CALL"/> </intent-filter> </receiver>
-
接收打电话广播需要权限
<uses-permission android:name="android.permission.PROCESS_OUTGOING_CALLS"/>
- 即使广播接收者的进程没有启动,当系统发送的广播可以被该接收者接收时,系统会自动启动该接收者所在的进程
短信拦截器
系统收到短信时会产生一条广播,广播中包含了短信的号码和内容
-
定义广播接收者接收短信广播
public void onReceive(Context context, Intent intent) { //拿到广播里携带的短信内容 Bundle bundle = intent.getExtras(); Object[] objects = (Object[]) bundle.get("pdus"); for(Object ob : objects ){ //通过object对象创建一个短信对象 SmsMessage sms = SmsMessage.createFromPdu((byte[])ob); System.out.println(sms.getMessageBody()); System.out.println(sms.getOriginatingAddress()); }
}
- 系统创建广播时,把短信存放到一个数组,然后把数据以pdus为key存入bundle,再把bundle存入intent
-
清单文件中配置广播接收者接收的广播类型,注意要设置优先级属性,要保证优先级高于短信应用,才可以实现拦截
<receiver android:name="com.
ibky
.smslistener.SmsReceiver">
<intent-filter android:priority="1000"> //priority优先级:-1000~1000,1000是最大的
<action android:name="android.provider.Telephony.SMS_RECEIVED"/> //谷歌把这条自动提示屏蔽掉了,需要手打。
</intent-filter>
</receiver>
-
添加权限
<uses-permission android:name="android.permission.RECEIVE_SMS"/>
-
4.0以后广播接收者安装以后必须手动启动一次,否则不生效
- 4.0以后广播接收者如果被手动关闭,就不会再启动了
监听SD卡状态
-
清单文件中定义广播接收者接收的类型,监听SD卡常见的三种状态,所以广播接收者需要接收三种广播
<receiver android:name="com.ibky.sdcradlistener.SDCardReceiver"> <intent-filter > <action android:name="android.intent.action.MEDIA_MOUNTED"/> //就绪 <action android:name="android.intent.action.MEDIA_UNMOUNTED"/> //被卸载 <action android:name="android.intent.action.MEDIA_REMOVED"/> //被移除 <data android:scheme="file"/> //这里必须带一个data,file为SD卡的路径 </intent-filter> </receiver>
-
广播接收者的定义
public class SDCardReceiver extends BroadcastReceiver { @Override public void onReceive(Context context, Intent intent) { // 区分接收到的是哪个广播 String action = intent.getAction(); if(action.equals("android.intent.action.MEDIA_MOUNTED")){ System.out.println("sd卡就绪"); } else if(action.equals("android.intent.action.MEDIA_UNMOUNTED")){ System.out.println("sd卡被移除"); } else if(action.equals("android.intent.action.MEDIA_REMOVED")){ System.out.println("sd卡被拔出"); } } }
勒索软件
- 接收开机广播,在广播接收者中启动勒索的Activity
-
清单文件中配置接收开机广播
<receiver android:name="com.ibky.lesuo.BootReceiver"> <intent-filter > <action android:name="android.intent.action.BOOT_COMPLETED"/> </intent-filter> </receiver>
-
权限
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED"/>
-
定义广播接收者
@Override public void onReceive(Context context, Intent intent) { //开机的时候就启动勒索软件 Intent it = new Intent(context, MainActivity.class); context.startActivity(it); }
- 以上代码还不能启动MainActivity,因为广播接收者的启动,并不会创建任务栈,那么没有任务栈,就无法启动activity
-
手动设置创建新任务栈的flag
it.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
监听应用的安装、卸载、更新
原理:应用在安装卸载更新时,系统会发送广播,广播里会携带应用的包名 * 清单文件定义广播接收者接收的类型,因为要监听应用的三个动作,所以需要接收三种广播
<receiver android:name="com.ibky.app.AppReceiver">
<intent-filter >
<action android:name="android.intent.action.PACKAGE_ADDED"/>
<action android:name="android.intent.action.PACKAGE_REPLACED"/>
<action android:name="android.intent.action.PACKAGE_REMOVED"/>
<data android:scheme="package"/>
</intent-filter>
</receiver>
-
广播接收者的定义
public void onReceive(Context context, Intent intent) { //区分接收到的是哪种广播 String action = intent.getAction(); //获取广播中包含的应用包名 Uri uri = intent.getData(); if(action.equals("android.intent.action.PACKAGE_ADDED")){ System.out.println(uri + "被安装了"); } else if(action.equals("android.intent.action.PACKAGE_REPLACED")){ System.out.println(uri + "被更新了"); } else if(action.equals("android.intent.action.PACKAGE_REMOVED")){ System.out.println(uri + "被卸载了"); } }
广播的两种类型
- 无序广播:所有跟广播的intent匹配的广播接收者都可以收到该广播,并且是没有先后顺序(同时收到)
- 有序广播:所有跟广播的intent匹配的广播接收者都可以收到该广播,但是会按照广播接收者的优先级来决定接收的先后顺序
- 优先级的定义:-1000~1000
- 最终接收者:所有广播接收者都接收到广播之后,它才接收,并且一定会接收
- abortBroadCast:阻止其他接收者接收这条广播,类似拦截,只有有序广播可以被拦截
Service
- 就是默默运行在后台的组件,可以理解为是没有前台的activity,适合用来运行不需要前台界面的代码
- 服务可以被手动关闭,不会重启,但是如果被自动关闭,内存充足就会重启
- startService启动服务的生命周期
- onCreate-onStartCommand-onDestroy
- 重复的调用startService会导致onStartCommand被重复调用
进程优先级
- 前台进程:拥有前台activity(onResume方法被调用)
- 可见进程:拥有可见activity(onPause方法被调用)
- 服务进程:不到万不得已不会被回收,而且即便被回收,内存充足时也会被重启
- 后台进程:拥有后台activity(activity的onStop方法被调用了),很容易被回收
- 空进程:没有运行任何activity,很容易被回收
电话窃ting器
- 电话状态:空闲、响铃、接听
-
获取电话管理器,设置侦听
TelephonyManager tm = (TelephonyManager) getSystemService(TELEPHONY_SERVICE); tm.listen(new MyPhoneStateListener(), PhoneStateListener.LISTEN_CALL_STATE);
-
侦听对象的实现
class MyPhoneStateListener extends PhoneStateListener{ //当电话状态改变时,此方法调用 @Override public void onCallStateChanged(int state, String incomingNumber) { // TODO Auto-generated method stub super.onCallStateChanged(state, incomingNumber); switch (state) { case TelephonyManager.CALL_STATE_IDLE://空闲 if(recorder != null){ recorder.stop(); recorder.release(); } break; case TelephonyManager.CALL_STATE_OFFHOOK://摘机 if(recorder != null){ recorder.start(); } break; case TelephonyManager.CALL_STATE_RINGING://响铃 recorder = new MediaRecorder(); //设置声音来源 recorder.setAudioSource(MediaRecorder.AudioSource.MIC); //设置音频文件格式 recorder.setOutputFormat(MediaRecorder.OutputFormat.THREE_GPP); recorder.setOutputFile("sdcard/haha.3gp"); //设置音频文件编码 recorder.setAudioEncoder(MediaRecorder.AudioEncoder.AMR_NB); try { recorder.prepare(); } catch (IllegalStateException e) { // TODO Auto-generated catch block e.printStackTrace(); } catch (IOException e) { // TODO Auto-generated catch block e.printStackTrace(); } break; } } }
随堂笔记
Android四大组件
- Activity
- BroadcastReceiver
- service
- ContentProvider
广播
- 现实
- 电台通过播放广播,发布消息
- 买个收音机,就能收到消息
- Android
- 系统在运行过程中,会产生很多事件:开机、拨打电话、收发短信、电量改变、屏幕解锁
- 一旦发生这种事件,系统会发送一个广播,为了接收到这个广播,知道系统发生了什么事件,就需要广播接收者,功能类似收音机
IP拨号器
功能:拨打电话时,自动添加ip线路的号码前缀
原理:系统打电话时,会发送一个打电话广播,iP拨号器只要定义广播接收者接收这个广播,那么就可以知道用户什么时候打电话,打电话广播中还会包含用户拨打的号码,ip拨号器可以从广播中获取此号码,并且作出修改,也就是添加号码前缀
短信拦截器
功能:类似短信防火墙系统在收到短信时,会产生一条短信广播,短信广播里,包含了短信的发信人号码和短信的内容,短信应用之所以能收到短信,其实是收到了短信广播,那我们的短信拦截器,只要在短信应用拿到广播之前,把短信广播拦截,那么短信应用就不会收到广播
监听sd卡状态
sd卡状态改变时,系统会发送广播
应用安装、更新、卸载的监听
原理,系统在安装更新卸载时,系统都会产生广播,广播中带有应用的包名
两种广播
无序广播
- 所有intent-filter与无序广播匹配的广播接收者都可以收到这条广播,并且没有先后顺序(可以理解为同时)
有序广播
- 所有intent-filter与有序广播匹配的广播接收者都可以收到这条广播,但是有先后顺序,按照intent-filter中定义优先级来决定(-1000~1000)
服务
- 默默运行在后台的组件,可以理解为没有前台的Activity,服务的作用就是用来运行需要在后台默默运行的代码
进程优先级
- 前台进程:拥有一个可以与用户交互的Activity(Activity的onResume方法调用)的进程
- 可见进程:拥有一个可见但无焦点的Activity(Activity的onPause方法被调用)的进程
- 服务进程:拥有一个通过startService方法启动的服务的进程
- 后台进程:拥有一个不可见Activity(Activity的onStop方法被调用)的进程
- 空进程:!拥有任何活动的组件(Activity、service)的进程
电话窃.听器
用户接到来电,一旦接听,立即开始录音,电话挂掉,录音结束,生成音频文件
- 电话的状态
- 空闲
- 响铃
- 摘机
心如猛虎,细嗅蔷薇。