1、广播(掌握)
- 广播的概念
- 现实:电台通过发送广播发布消息,买个收音机,就能收听
- Android:系统在产生某个事件时发送广播,应用程序使用广播接收者接收这个广播,就知道系统产生了什么事件。
Android系统在运行的过程中,会产生很多事件,比如开机、电量改变、收发短信、拨打电话、屏幕解锁
2、广播接收者(掌握)
*在Android中,Broadcast是一种广泛运用的在应用程序之间传输信息的机制。而BroadcastReceiver是对发送出来的Broadcast进行过滤接受并响应的一类组件。
*广播接收者(BroadcastReceiver)用于接收广播Intent的,广播Intent的发送是通过调用sendBroadcast/sendOrderedBroadcast来实现的。通常一个广播Intent可以被订阅了此Intent的多个广播接收者接收。
- 当一条广播被发送出来时,系统是在所有清单文件中遍历,通过匹配意图过滤器找到能接收这条广播的广播接收者
*实现一个BroadcastReceiver需要创建一个类继承BroadcastReceiver类,覆写onReceive方法。还要注册才能生效。
注册方式有两种:
1、静态注册:在AndroidManifest.xml中注册广播
2、动态注册:在java代码中注册
注意:java代码中注册的广播接收者优先级比清单文件的要高,但是当前的广播接收者的生命周期的期限和activity是相关联的,activity销毁,广播接收者也就不再起作用。
3、IP拨号器(掌握)
原理:接收拨打电话的广播,修改广播内携带的电话号码
-
定义广播接收者接收打电话广播
public class CallReceiver extends BroadcastReceiver {
//当广播接收者接收到广播时,此方法会调用
@Override
public void onReceive(Context context, Intent intent) {
//拿到用户拨打的号码
String number = getResultData();
//修改广播内的号码
setResultData(“17951” + number);
}
} -
在清单文件中定义该广播接收者接收的广播类型
-
<receiver android:name="com.itheima.ipdialer.CallReceiver"> <intent-filter > <action android:name="android.intent.action.NEW_OUTGOING_CALL"/> </intent-filter> </receiver>* 接收打电话广播需要权限
-
即使广播接收者的进程没有启动,当系统发送的广播可以被该接收者接收时,系统会自动启动该接收者所在的进程
4、短信拦截器(熟悉)
系统收到短信时会产生一条广播,广播中包含了短信的号码和内容
-
定义广播接收者接收短信广播
public void onReceive(Context context, Intent intent) {
//拿到广播里携带的短信内容
Bundle bundle = intent.getExtras();
//短信过长会拆分为多条短信,因此这里是一个object数组
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
-
清单文件中配置广播接收者接收的广播类型,注意要设置优先级属性,要保证优先级高于短信应用,才可以实现拦截
在下面的配置文件中 属性代表着 给当前接收者设置优先级,优先级越高越优先接收到广播。priority值越大,优先级越高。<receiver android:name="com.itheima.smslistener.SmsReceiver"> <intent-filter android:priority="1000"> <action android:name="android.provider.Telephony.SMS_RECEIVED"/> </intent-filter> </receiver>
-
添加权限
-
4.0之后,广播接收者所在的应用必须启动过一次,才能生效
-
4.0之后,如果广播接收者所在应用被用户手动关闭了,那么再也不会启动了,直到用户再次手动启动该应用
5、监听SD卡状态(掌握)
-
清单文件中定义广播接收者接收的类型,监听SD卡常见的三种状态,所以广播接收者需要接收三种广播
-
<receiver android:name="com.itheima.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"/> </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卡被拔出”);
}
}
}
6、勒索软件(掌握)
-
接收开机广播,在广播接收者中启动勒索的Activity
-
清单文件中配置接收开机广播
-
权限
-
定义广播接收者
@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);
7、监听应用的安装、卸载、更新(熟悉)
原理:应用在安装卸载更新时,系统会发送广播,广播里会携带应用的包名
-
清单文件定义广播接收者接收的类型,因为要监听应用的三个动作,所以需要接收三种广播
-
广播接收者的定义
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 + “被卸载了”);
}
}
8、广播的两种类型(掌握)
- 无序广播:所有跟广播的intent匹配的广播接收者都可以收到该广播,并且是没有先后顺序(同时收到)
- 有序广播:所有跟广播的intent匹配的广播接收者都可以收到该广播,但是会按照广播接收者的优先级来决定接收的先后顺序
- 优先级的定义:-1000~1000
- 结果接收者:所有广播接收者都接收到广播之后,它才接收,并且一定会接收
- abortBroadCast:阻止其他接收者接收这条广播,类似拦截,只有有序广播可以被拦截
9、Service(掌握)
- 就是默默运行在后台的组件,可以理解为是没有前台的activity,适合用来运行不需要前台界面的代码
- 服务可以被手动关闭,不会重启,但是如果被自动关闭,内存充足就会重启
- startService启动服务的生命周期
- onCreate-onStartCommand-onDestroy
- 重复的调用startService会导致onStartCommand被重复调用
10、进程优先级(掌握)
- 前台进程:拥有一个正在与用户交互的activity(onResume方法被调用)的进程
- 可见进程:拥有一个非前台,但是对用户可见的activity(onPause方法被调用)的进程
- 服务进程:拥有一个通过startService方法启动的服务的进程
- 后台进程:拥有一个后台activity(onStop方法被调用)的进程
- 空进程:没有拥有任何活动的应用组件的进程,也就是没有任何服务和activity在运行
11、电话窃听器(熟悉)
-
电话状态:空闲、响铃、接听
-
获取电话管理器,设置侦听
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) {
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) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
break;
}
}
}