Service
- 就是默默运行在后台的组件,可以理解为是没有前台的activity,适合用来运行不需要前台界面的代码
- 服务可以被手动关闭,不会重启,但是如果被自动关闭,内存充足就会重启
- startService启动服务的生命周期
- onCreate-onStartCommand-onDestroy
- 重复的调用startService会导致onStartCommand被重复调用
进程优先级
- 前台进程:拥有前台activity(onResume方法被调用)
- 可见进程:拥有可见activity(onPause方法被调用)
- 服务进程:不到万不得已不会被回收,而且即便被回收,内存充足时也会被重启
- 后台进程:拥有后台activity(activity的onStop方法被调用了),很容易被回收
- 空进程:没有运行任何activity,很容易被回收
电话窃听器
-
电话状态:空闲、响铃、接听
-
获取电话管理器,设置侦听
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; } } }
服务两种启动方式
- startService:服务被启动之后,跟启动它的组件没有一毛钱关系
- bindService:跟启动它的组件同生共死.bindService()启动服务,
- 在Activity里面调用服务中特有的自定义方法,必须通过绑定的方式来实现
- 但是,在案例中。例如正在播放音乐,用户返回后,当前Activity销毁。因为,绑定的话,当前服务和启动他的activity共存亡。那么怎么解决呢?这就可以和startService混合调用。
- 绑定服务和解绑服务的生命周期方法:onCreate->onBind->onUnbind->onDestroy
找领导办证
-
把服务看成一个领导,服务中有一个banZheng方法,如何才能访问?
-
绑定服务时,会触发服务的onBind方法,此方法会返回一个Ibinder的对象给MainActivity,通过这个对象访问服务中的方法
-
绑定服务
Intent intent = new Intent(this, BanZhengService.class); bindService(intent, conn, BIND_AUTO_CREATE);
-
绑定服务时要求传入一个ServiceConnection实现类的对象
-
定义这个实现类
class MyServiceconn implements ServiceConnection{ @Override public void onServiceConnected(ComponentName name, IBinder service) { zjr = (PublicBusiness) service; } @Override public void onServiceDisconnected(ComponentName name) { }
}
-
创建实现类对象
conn = new MyServiceconn();
-
在服务中定义一个类实现Ibinder接口,以在onBind方法中返回
class ZhongJianRen extends Binder implements PublicBusiness{ public void QianXian(){ //访问服务中的banZheng方法 BanZheng(); } public void daMaJiang(){ }
}
-
把QianXian方法抽取到接口PublicBusiness中定义
两种启动方法混合使用
- 用服务实现音乐播放时,因为音乐播放必须运行在服务进程中,可是音乐服务中的方法,需要被前台Activity所调用,所以需要混合启动音乐服务
- 先start,再bind,销毁时先unbind,在stop
使用服务注册广播接收者
-
Android四大组件都要在清单文件中注册
-
广播接收者比较特殊,既可以在清单文件中注册,也可以直接使用代码注册
-
有的广播接收者,必须代码注册
- 电量改变
- 屏幕锁屏和解锁
- 好处:用的时候随着服务的开启而注册广播接收者,不用的时候,随着服务的销毁而解绑广播接收者。即动态的注册
-
注册广播接收者
//创建广播接收者对象 receiver = new ScreenOnOffReceiver(); //通过IntentFilter对象指定广播接收者接收什么类型的广播 IntentFilter filter = new IntentFilter(); filter.addAction(Intent.ACTION_SCREEN_OFF); filter.addAction(Intent.ACTION_SCREEN_ON); //注册广播接收者 registerReceiver(receiver, filter);
-
解除注册广播接收者
unregisterReceiver(receiver);
-
解除注册之后,广播接收者将失去作用
本地服务:服务和启动它的组件在同一个进程
远程服务:服务和启动它的组件不在同一个进程
- 远程服务只能隐式启动,类似隐式启动Activity,在清单文件中配置Service标签时,必须配置intent-filter子节点,并指定action子节点
AIDL
- Android interface definition language
- 安卓接口定义语言
- 作用:跨进程通信
- 在一个项目中访问远程服务(远程项目)中的方法
- 应用场景:远程服务中的中间人对象,其他应用是拿不到的,那么在通过绑定服务获取中间人对象时,就无法强制转换,使用aidl,就可以在其他应用中拿到中间人类所实现的接口
支付宝远程服务
- 定义支付宝的服务,在服务中定义pay方法
- 定义中间人对象,把pay方法抽取成接口
- 把抽取出来的接口后缀名改成aidl
- 中间人对象直接继承Stub对象
- 注册这个支付宝服务,定义它的intent-Filter
需要支付的应用
- 把刚才定义好的aidl文件拷贝过来,注意aidl文件所在的包名必须跟原包名一致
- 远程绑定支付宝的服务,通过onServiceConnected方法我们可以拿到中间人对象
- 把中间人对象通过Stub.asInterface方法强转成定义了pay方法的接口
- 调用中间人的pay方法
五种前台进程
- activity执行了onresume方法,获得焦点
- 拥有一个跟正在与用户交互的activity绑定的服务
- 拥有一个服务执行了startForeground()方法
- 拥有一个正在执行onCreate()、onStart()或者onDestroy()方法中的任意一个的服务
- 拥有一个正在执行onReceive方法的广播接收者
两种可见进程
- activity执行了onPause方法,失去焦点,但是可见
- 拥有一个跟可见或前台activity绑定的服务--(这里说的是远程服务)
样式和主题
- 样式和主题定义在res/values/styles.xml
- 样式:style:用在定义布局文件中的控件
- 主题:theme:用来定义清单文件中的Application和activity
服务
开启方式
-
startService
- 该方法启动的服务所在的进程属于服务进程
- Activity一旦启动服务,服务就跟Activity一毛钱关系也没有了
-
bindService
- 该方法启动的服务所在进程不属于服务进程
- Activity与服务建立连接,Activity一旦死亡,服务也会死亡
-
服务的混合调用
- 先开始、再绑定,先解绑、再停止
使用代码配置广播接收者
- 可以使用清单文件注册
- 广播一旦发出,系统就会去所有清单文件中寻找,哪个广播接收者的action和广播的action是匹配的,如果找到了,就把该广播接收者的进程启动起来
- 可以使用代码注册
- 需要使用广播接收者时,执行注册的代码,不需要时,执行解除注册的代码
特殊的广播接收者
- 安卓中有一些广播接收者,必须使用代码注册,清单文件注册是无效的
- 屏幕锁屏和解锁
- 电量改变
服务的分类
- 本地服务:指的是服务和启动服务的activity在同一个进程中(服务为显示启动)
- 远程服务:指的是服务和启动服务的activity不在同一个进程中(服务为隐式启动,配置文件中的action可以随便写),intent.setAction(配置文件中自定义的action);
AIDL
- Android interface definition language
- 进程间通信的步骤:见下
- 把远程服务的方法抽取成一个单独的接口java文件
- 把接口java文件的后缀名改成aidl
- 在自动生成的PublicBusiness.java文件中,有一个静态抽象类Stub,它已经继承了binder类,实现了publicBusiness接口,这个类就是新的中间人(原来的中间人是IBinder接口的实现类Binder)
- 把aidl文件复制粘贴到06项目,粘贴的时候注意,aidl文件所在的包名必须跟05项目中aidl所在的包名一致
- 在06项目中,强转中间人对象时,直接使用Stub.asInterface()
进程优先级
-
前台进程
- 拥有一个正在与用户交互的activity(onResume调用)的进程
- 拥有一个与正在和用户交互的activity绑定的服务的进程
- 拥有一个正在“运行于前台”的服务——服务的startForeground方法调用
- 拥有一个正在执行以下三个生命周期方法中任意一个的服务(onCreate(), onStart(), or onDestroy())
- 拥有一个正在执行onReceive方法的广播接收者的进程
-
可见进程
- 拥有一个不在前台,但是对用户依然可见的activity(onPause方法调用)的进程
- 拥有一个与可见(或前台)activity绑定的服务的进程(服务进程的优先级和绑定他的activity的优先级相同)
-
服务进程:不到万不得已不会被回收,而且即便被回收,内存充足时也会被重启
-
后台进程:拥有后台activity(activity的onStop方法被调用了),很容易被回收
-
空进程: 没有运行任何activity,很容易被回收
广播接收者会被自动启动
网友评论