Android探索之Service全面回顾及总结
什么是Service?
Service(服务)是Android提供的四大组件之一,是一个没有用户界面的在后台运行执行耗时操作的应用组件。其他应用组件能够启动Service,并且当用户切换到另外的应用场景,Service将持续在后台运行。为了方便记忆,我们可以把Service看做是没有页面的Activity,它总是默默的后台处理一些耗时的操作或者不干扰用户使用的后台操作,例如,一个service可能会处理网络操作,播放音乐,操作文件I/O或者与内容提供者(content provider)交互,所有这些活动都是在后台进行。
Service都有哪些?
1)按照使用范围分类:
1.本地服务(Local Service):用于应用程序内部
Local Service运行于当前app的主进程中,如果当前app的进程被Kill掉了,Local Service也会随着随之终止。使用场景举例:音乐播放器服务
2.远程服务(Remote Service):用于android系统内部的应用程序之间
Remote Service是运行于一个独立的进程中,可以被多个app复用,可以使用android:process声明进程名字,由于运行于独立的进程,Activity所在进程被Kill的时候不会影响Remote Service。
2)按照运行类别分类:
1.前台服务
前台服务是那些被认为用户知道的并且在内存低的时候不允许系统杀死的服务,通过startForeground 使服务成为 前台服务。
2.后台服务
区别于前台服务,创建的服务默认是后台服务,在内存低的时候有可能被系统杀死。
3)按照使用方式分类:
1.context.startService()
2.context.bindService()
Service的如何使用?
1.Service AndroidManifest.xml 声明
<service android:enabled=["true" | "false"]
android:exported=["true" | "false"]
android:icon="drawable resource"
android:isolatedProcess=["true" | "false"]
android:label="string resource"
android:name="string"
android:permission="string"
android:process="string" >
. . .
</service>
具体参数解说:
android:name 服务类名
android:label 服务的名字,如果此项不设置,那么默认显示的服务名则为类名
android:icon 服务的图标
android:permission 申明此服务的权限,这意味着只有提供了该权限的应用才能控制或连接此服务
android:process 表示该服务是否运行在另外一个进程,如果设置了此项,那么将会在包名后面加上这段字符串表示另一进程的名字
android:enabled 如果此项设置为 true,那么 Service 将会默认被系统启动,不设置默认此项为 false
android:exported 表示该服务是否能够被其他应用程序所控制或连接,不设置默认此项为 false
具体写个测试Service
public class TestService extends Service {
private TestBinder binder=new TestBinder();
private Thread testThread;
private boolean isStart=false;
@Override
public void onCreate() {
super.onCreate();
Log.e("TestService", "执行 onCreate()");
startForeground();
}
private void startForeground(){
Notification.Builder builder = new Notification.Builder(this);
PendingIntent contentIntent = PendingIntent.getActivity(this, 0,
new Intent(this, MainActivity.class), 0);
builder.setContentIntent(contentIntent);//设置目标跳转
builder.setSmallIcon(R.mipmap.ic_launcher);//设置显示的图片
builder.setTicker("前台服务开启");// 状态栏上显示
builder.setContentTitle("前台服务");//设置标题
builder.setContentText("这是一个前台服务");
Notification notification = builder.build();
startForeground(1, notification);
}
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
Log.e("TestService", "执行 onStartCommand()");
//startThread();
//testANR();
return super.onStartCommand(intent, flags, startId);
}
@Override
public void onDestroy() {
super.onDestroy();
Log.e("TestService", "执行 onDestroy()");
stopThread();
stopForeground(true);
}
@Override
public IBinder onBind(Intent intent) {
Log.e("TestService", "执行 onBind()");
return binder;
}
class TestBinder extends Binder {
public void testMethod() {
Log.e("TestService", "执行 testMethod()");
}
}
//测试是否ANR
private void testANR(){
try {
Thread.sleep(50000);
} catch (Exception e) {
}
}
private void startThread(){
stopThread();
isStart = true;
if (testThread == null) {
testThread = new Thread(runnable);
testThread.start();
}
}
private void stopThread(){
try {
isStart = false;
if (null != testThread && Thread.State.RUNNABLE == testThread.getState()) {
try {
Thread.sleep(500);
testThread.interrupt();
} catch (Exception e) {
testThread = null;
}
}
testThread = null;
} catch (Exception e) {
e.printStackTrace();
} finally {
testThread = null;
}
}
private Runnable runnable=new Runnable() {
@Override
public void run() {
while(isStart) {
Log.e("TestService", "runnable");
}
}
};
}
启动/关闭,绑定/解绑
//开启service
Log.e("TestService", "执行 startService()");
Intent intent =new Intent(MainActivity.this,TestService.class);
startService(intent); //停止service
Log.e("TestService", "执行 stopService()");
Intent intent =new Intent(MainActivity.this,TestService.class);
stopService(intent); //绑定service
Log.e("TestService", "执行 bindService()");
Intent intent =new Intent(MainActivity.this,TestService.class);
bindService(intent,connection,BIND_AUTO_CREATE); //解绑service
Log.e("TestService", "执行 unbindService()");
unbindService(connection); //测试ServiceConnection
private ServiceConnection connection = new ServiceConnection() { @Override
public void onServiceDisconnected(ComponentName name) {
} @Override
public void onServiceConnected(ComponentName name, IBinder service) {
TestService.TestBinder testBinder = (TestService.TestBinder) service;
testBinder.testMethod();
}
};
接下来了解下Service的生命周期:
1)startService()/stopService()方式 onCreate()---->onStartCommand()---->running()---->onDestory()

如果多次startService()会不会多次onCreate呢?Service只会调用一次onCreate(),但会多次调用onStartCommand()

如果多次调用stopService()呢?调用stopService()一次终止服务

2)bindService()/unbindService()方式 onCreate()---->onBind()---->running()---->onDestory()

3)同时调用了startService()/bindService()

首先看下我们调用stopService() 看下运行结果如下:未执行onDestory()服务无法关闭;所有两者同时使用时先调用unbindService()然后再调用stopService()才能真正关闭服务

接下来我们重点看下onStartCommand(Intent intent, int flags, int startId)方法中的flags参数。
START_NOT_STICKY:当Service因为内存不足而被系统kill后,接下来未来的某个时间内,即使系统内存足够可用,系统也不会尝试重新创建此Service。除非程序中Client明确再次调用startService(...)启动此Service。
START_STICKY:当Service因为内存不足而被系统kill后,接下来未来的某个时间内,当系统内存足够可用的情况下,系统将会尝试重新创建此Service,一旦创建成功后将回调onStartCommand(...)方法,但其中的Intent将是null,pendingintent除外。
START_REDELIVER_INTENT:与START_STICKY唯一不同的是,回调onStartCommand(...)方法时,其中的Intent将是非空,将是最后一次调用startService(...)中的intent。
START_STICKY_COMPATIBILITY:默认flags值
通常我们为了防止Service在内存不足的时候被系统杀死,把flags设置成START_STICKY 但是在4.0之后部分手机也是无效的,被杀死之后不会重启了。
我们该如何提供app服务不被杀死或者降低被杀死的概率呢?我们可以把设置服务为前台服务,比如音乐播放器,app退出之后就变成前台服务。直接看具体实现:
//设置前台服务
private void startForeground(){
Notification.Builder builder = new Notification.Builder(this);
PendingIntent contentIntent = PendingIntent.getActivity(this, 0,
new Intent(this, MainActivity.class), 0);
builder.setContentIntent(contentIntent);//设置目标跳转
builder.setSmallIcon(R.mipmap.ic_launcher);//设置显示的图片
builder.setTicker("前台服务开启");// 状态栏上显示
builder.setContentTitle("前台服务");//设置标题
builder.setContentText("这是一个前台服务");
Notification notification = builder.build();
startForeground(1, notification);
} //关闭前台服务,参数true 代表remove状态栏通知
stopForeground(true);
具体效果:

每当我们讨论Service的时候总是听到别人说:Service用来处理一个耗时的操作!!!接下来我们来验证一下Service里该如何处理耗时操作?
为了方便写了一个模拟耗时的函数:
//测试是否ANR
private void testANR(){
try {
Thread.sleep(50000);
} catch (Exception e) {
}
}
测试运行:
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
Log.e("TestService", "执行 onStartCommand()");
//startThread();
testANR();
return super.onStartCommand(intent, flags, startId);
}
运行结果:

其实结果是在我们的意料之中的,经过上面的知识我们已经得知本地服务是依托附主进程的上的,处理耗时操作肯定会造成ANR的,如果改成远程服务就不会造成ANR了。所以我们在Service处理耗时操作也是要像Activity一样采用异步处理的。需要主要的时如果我们在Service中起了一个线程,我们必须在onDestory ()函数中销毁线程。因为我们stopService()的时候线程并不会随之销毁。因此可知:Service和Thread是两码事,没有一毛钱的关系。
知识拓展:
1)检查某个Service是否在运行中:
private boolean isServiceRunning(String className) {
boolean isRunning = false;
ActivityManager activityManager = (ActivityManager) getSystemService(Context.ACTIVITY_SERVICE);
List<ActivityManager.RunningServiceInfo> serviceList = activityManager.getRunningServices(30);
if (!(serviceList.isEmpty())) {
return false;
}
for (int i=0; i<serviceList.size(); i++) {
if (serviceList.get(i).service.getClassName().equals(className) == true) {
isRunning = true;
break;
}
}
return isRunning;
}
2)本文中开启Service采用的显式Intent,同样采用隐式Intent开启Service,如果app被破译则有安全隐患。
<permission android:name="com.whoislcj.testservice.permission"/>
<uses-permission android:name="com.whoislcj.testservice.permission"/>
如果一个实体运行或绑定一个服务,必须要拥有该服务的权限。如果没有权限, startService() , bindService() 或 stopService() 方法将不执行, Intent 也不会传递到服务。
3)Service与UI之间的通讯。
UI----->Service :Intent传值
UI----->Service :绑定Service传值
UI<---->Service: 通过广播
4)Service与AIDL实现跨进程通信 总结文章地址:http://www.cnblogs.com/whoislcj/p/5509868.html
先认识几个名词:
- 进程间通信(IPC):Inter Process Communication
- AIDL:Android Interface Definition Language, Android接口定义语言
Android探索之Service全面回顾及总结的更多相关文章
- (转载)Android中的Service:Binder,Messenger,AIDL(2)
前言 前面一篇博文介绍了关于Service的一些基本知识,包括service是什么,怎么创建一个service,创建了一个service之后如何启动它等等.在这一篇博文里有一些需要前一篇铺垫的东西,建 ...
- Android 面试题--Service
1.Service 是否在 main thread 中执行, service 里面是否能执行耗时的操作?默认情况,如果没有显示的指 servic 所运行的进程, Service 和 activity ...
- Android 中的 Service 全面总结(转载)
转载地址:http://www.cnblogs.com/newcj/archive/2011/05/30/2061370.html 感谢作者 Android 中的 Service 全面总结 1.Ser ...
- java攻城狮之路(Android篇)--BroadcastReceiver&Service
四大组件:activity 显示. contentProvider 对外暴露自己的数据给其他的应用程序.BroadcastReceiver 广播接收者,必须指定要接收的广播类型.必须明确的指定acti ...
- 图解Android - Binder 和 Service
在 Zygote启动过程 一文中我们说道,Zygote一生中最重要的一件事就是生下了 System Server 这个大儿子,System Server 担负着提供系统 Service的重任,在深入了 ...
- Android 服务类Service 的详细学习
http://blog.csdn.net/vipzjyno1/article/details/26004831 Android服务类Service学习四大组建 目录(?)[+] 什么是服务 服务有 ...
- 大仙说道之Android studio实现Service AIDL
今天要开发过程中要用到AIDL的调用,之前用的eclipse有大量教程,用起来很方便,现在刚换了Android studio,不可否认studio真的很强大,只是很多功能还需要摸索. AIDL(And ...
- Android四大组件——Service
Service相关链接 Service初涉 Service进阶 Service精通 Service是Android系统中的一种组件,它跟Activity的级别差不多,但是它不能自己运行,只能后台运行, ...
- Android学习总结——Service组件
从Service的启动方式上,可以将Service分为Started Service和Bound Service.在使用Service时,要想系统能够找到此自定义Service,无论哪种类型,都需要在 ...
随机推荐
- img只显示图片一部分 或 css设置背景图片只显示图片指定区域
17:14 2016/3/22img只显示图片一部分 或 css设置背景图片只显示图片指定区域 background-position: 100% 56%; 设置背景图片显示图片的哪个坐标区域,图片左 ...
- Git使用出错:Couldn‘t reserve space for cygwin‘s heap, Win32
今天使用Git在命令行下更新代码遇到了问题,起初觉得是自己安装某软件导致冲突,从网上搜索了一下找到类似问题,成功解决问题. 错误信息如下: E:\storm-sql>git pull origi ...
- 【异常】No ManagedConnections available within configured blocking timeout
Caused by: org.jboss.util.NestedSQLException: No ManagedConnections available within configured bloc ...
- MySQL表名和数据库关键字相同解决办法
今天改他们的代码的时候,遇到了MySQL表名和数据库关键字的问题. 由于表名是关键字,导致增删改查都报错. Hibernate: select leave0_.id as id22_, leave0_ ...
- 移动前端不得不了解的html5 head 头标签
本文主要内容来自一丝的常用的 HTML 头部标签和百度FEX的HTML head 头标签. 移动端的工作已经越来越成为前端工作的重要内容,除了平常的项目开发,HTML 头部标签功能,特别是meta标签 ...
- python基础之day2
python基本数据类型 1.数字 int(整型) 在32位机器上,整数的位数为32位,取值范围为-2**31-2**31-1,即-2147483648-2147483647 在64位系统 ...
- Jenkins中构建Testcomplete项目的方法介绍
Jenkins的部署在上一篇随笔中已经和大家介绍了,下面我们介绍一下再Jenkins中构建testcomplete项目.我这里使用的是Testcomplete11,下面详细介绍一下构建步骤. 1.Je ...
- Raid 介绍以及软raid的实现
RAID: old Redundant Arrays of Inexpensive Disks (廉价磁盘冗余阵列) new Redundant Arrays of Independent Disks ...
- php数据加密
<?php/** * 简单对称加密算法之加密 * @param String $string 需要加密的字串 * @param String $skey 加密EKY * @author Anyo ...
- 最后一周psp
团队项目PSP 一:表格 C类型 C内容 S开始时间 E结束时间 I时间间隔 T净时间(mins) 预计花费时间(mins) 讨论 讨论用户界面 10:20 11:45 25 40 80 分析 ...