Service是什么?Service又不是什么?
在Android王国中,Service是一个劳动模范,总是默默的在后台运行,无怨无悔,且总是干最脏最累的活,比如下载文件,倾听音乐,网络操作等这些耗时的操作,所以我们请尊重的叫他一声:"劳模,您辛苦了".
- Service是什么?
- A Service is an application component. ☆ Service 是一个应用程序组件
- that can perform long-running operations in the background. ☆ 它能在后台执行一些耗时较长的操作.
- and does not provide a user interface. ☆ 并且不提供用户界面
- Another application component can start a service and it will continue to run in the background event if the user switches to another application. ☆ 服务能被其它的应用程序组件启动,即使用户切换到其他的应用程序时还能保持在后台运行.
- Additionally,a component can bind to a service to interact with it and even perform interprocess communcation(IPC). ☆ 此外,组件还能绑定服务,并与服务交互,甚至执行进程间通信(IPC).
- For example,a service might handle network transactions,play music,perform file I/O,or interact with a content provider,all from the background. ☆ 比如,一个服务可以处理网络传输,听音乐,执行文件操作,或者与内容提供者进行交互,所有这些都在后台进行.
- A service can essentially tack two forms.☆ 服务有以下两种基本类型
- Started --> startService()
- Bound --> bindService()
- Service又不是什么?
- A service is not a separate process.☆ 服务不是一个单独的进程.
- A service is not a thread.it runs in the main thread of its hosting process. ☆ 服务不是一个线程,它运行在主线程.
- the service does not create its own thread and does not run in a separate process(unless you specify otherwise). ☆ 服务不能自己创建并且不能运行在单独的进程中(除非你明确指定).
- This means that, if your service is going to do any CPU intensive work ot blocking operations(such as MP3 playback or network). ☆ 这意味着如果你的服务要执行一些很耗CUP的工作或者阻塞的操作(比如播放mp3或网络操作),you should create a new thread within the service to do that work. ☆ 你应该在服务中创建一个新的线程来执行这些工作.
- By using a separate thread, you will reduce the risk of Application Not Responding(ANR) errors and the application's main thread can remain dedicated to user interaction with your activities. ☆ 利用一个分离的进程,将减少你的activities发生应用程序停止响应(ANR)错误的风险.
- 如何创建一个Started服务
- 继承service
- publicclassFirstServiceextendsService{
privatestaticfinalString TAG ="--FirstService-->";publicFirstService(){Log.i(TAG,"Service is running.");}@Overridepublicvoid onCreate(){Log.i(TAG,"onCreate is running.");super.onCreate();}@Overridepublicint onStartCommand(Intent intent,int flags,int startId){Log.i(TAG,"onStartCommand is running.");returnsuper.onStartCommand(intent, flags, startId);}@OverridepublicIBinder onBind(Intent intent){Log.i(TAG,"IBinder is running.");returnnull;}}
- 四大组件都需要在manifests.xml中注册,这个也不例外.
- 如何启动它
- Intent intent =newIntent(ServiceActivity.this,FirstService.class);
startService(intent);
- 生命周期onCreate(), onStartCommand(), onDestory()就这三个生命周期
- --FirstService-->:Service is running.
--FirstService-->: onCreate is running.--FirstService-->: onStartCommand is running.
- 我们在onStartCommand方法中打印下当前线程
- @Override
publicint onStartCommand(Intent intent,int flags,int startId){Log.i(TAG,"onStartCommand is running.Thread:"+Thread.currentThread());returnsuper.onStartCommand(intent, flags, startId);}
打印结果如下:- onStartCommand is running.Thread:Thread[main,5,main]
验证了两点:①由于是第二次运行,构造方法和onStart都没有打印,说明服务一旦启动是默默运行在后台的;②服务是运行在主线程的 - 如何结束服务:①调用stopService()方法 ,会回调service的onDestory()方法;
- Intent intent2 =newIntent(ServiceActivity.this,FirstService.class);
stopService(intent2);
还可以在Android系统设置-->应用-->正在运行-->找到后是如下样式,会告诉我们有一个服务在运行; 
- onStartCommand的返回值:
- START_STICKY:粘性的,被意外中止后自动重启,重新调用onStartCommand(),但会丢失原来激活它的Intent,会用一个null intent来调用onStartCommand(),可以用于播放器.值为1
- START_NOT_STICKY:非粘性的,被意外中止后不会重启,除非还存在未发送的Intent,这是避免服务运行的最安全选项; 值为2
- START_REDELIVER_INTENT:粘性的且重新发送Intent,被意外中止后重新启动,且该service组件将得到用于激活它的Intent对象,这中服务适用于需要立即恢复工作的活跃服务,比如下载文件; 值为3
- onStartCommand的参数:
- @Override //第一个参数:为我们传入的intent;第二个flags:启动服务的方式,与返回值有关;第三个为我们启动service的次数.
publicint onStartCommand(Intent intent,int flags,int startId){Log.i(TAG,"onStartCommand is running.Thread:"+Thread.currentThread());Log.i(TAG,"flags:"+flags);Log.i(TAG,"startId:"+startId);returnsuper.onStartCommand(intent, flags, startId);}
因为前面服务已经启动了,这次我们连续点了三次启动服务的按钮,打印日志如下:- 11-1602:56:28.3859039-9039/com.wanghx.androidstudy I/--FirstService-->: onStartCommand is running.Thread:Thread[main,5,main]
11-1602:56:28.3859039-9039/com.wanghx.androidstudy I/--FirstService-->: flags:011-1602:56:28.3859039-9039/com.wanghx.androidstudy I/--FirstService-->: startId:211-1602:56:33.9859039-9039/com.wanghx.androidstudy I/--FirstService-->: onStartCommand is running.Thread:Thread[main,5,main]11-1602:56:33.9859039-9039/com.wanghx.androidstudy I/--FirstService-->: flags:011-1602:56:33.9859039-9039/com.wanghx.androidstudy I/--FirstService-->: startId:311-1602:56:35.5359039-9039/com.wanghx.androidstudy I/--FirstService-->: onStartCommand is running.Thread:Thread[main,5,main]11-1602:56:35.5359039-9039/com.wanghx.androidstudy I/--FirstService-->: flags:011-1602:56:35.5359039-9039/com.wanghx.androidstudy I/--FirstService-->: startId:4
我们发现flags的值没有发生改变,而startId再按顺序增加. - 如何启动一个绑定服务
- 在activity中创建一个内部类,继承ServiceConnection.
- classMyServiceConnectionimplementsServiceConnection{
@Overridepublicvoid onServiceConnected(ComponentName name,IBinder service){Log.i(TAG,"onServiceConnected");}@Overridepublicvoid onServiceDisconnected(ComponentName name){Log.i(TAG,"onServiceDisconnected");}}
- 在activity中定义一个成员connection
privateMyServiceConnection connection =newMyServiceConnection();
- 在绑定服务按钮中加入绑定代码
Intent intent3 =newIntent(ServiceActivity.this,FirstService.class);bindService(intent3, connection, BIND_AUTO_CREATE);
按钮点击后打印如下日志:I/--FirstService-->:Service is running.I/--FirstService-->: onCreate is running.I/--FirstService-->:IBinder is running.
- 在解绑服务中加入代码,这里的connection必须和上边的绑定服务的connection实例一致.
unbindService(connection);
- Service和Thread的关系
- 其实他两个没有一毛钱关系.只是因为service需要做耗时操作,需要重新建立一线程来处理工作,而不阻塞主线程;
- service是运行在主线程的;
- activity启动service后,即使activity被销毁了,如果没有主动关闭服务,服务还是会在后台默默运行的;
- 如何连接远程的service,只需要在manifests.xml中这样写即可
- <service
android:name="com.example.servicetest.MyService"android:process=":remote"></service>
如何让activity与远程的service进行通信呢?这就要使用AIDL进行跨进程通信(IPC)了. - AIDL:Android Interface Definition Language:Android接口定义语言,它可以用于让多个service与多个应用程序组件之间进行跨进程通信;
- 这些都不是重点,我们还是弄一下在我们自己的程序中service与activity之间的通信吧;
- activity-->service 通过intent传递数据给service;
- activity调用onServiceConnected()中的IBind对象来访问service中的方法;
- IBinder通信的关键是利用activity中的IBinder对象获得service对象,然后调用方法;
- publicclassFirstServiceextendsService{
privatestaticfinalString TAG ="FirstService-->";privateMyBinder myBinder =newMyBinder();publicFirstService(){Log.i(TAG,"Service is running.");}@Overridepublicvoid onCreate(){Log.i(TAG,"onCreate is running.");super.onCreate();}@Overridepublicint onStartCommand(Intent intent,int flags,int startId){String name = intent.getStringExtra("name");Log.i(TAG,"onStartCommand is running.Thread:"+Thread.currentThread());Log.i(TAG,"flags:"+flags);Log.i(TAG,"startId:"+startId);Log.i(TAG,"name:"+name);return START_STICKY;}@Overridepublicvoid onDestroy(){Log.i(TAG,"onDestroy is running.");super.onDestroy();}@OverridepublicIBinder onBind(Intent intent){Log.i(TAG,"IBinder is running.");return myBinder;}publicclassMyBinderextendsBinder{publicFirstService getService(){returnFirstService.this;}}publicint getRandomNumber(){returnnewRandom().nextInt(10)+1;}}
- publicclassServiceActivityextendsAppCompatActivityimplementsView.OnClickListener{
privateMyServiceConnection connection =newMyServiceConnection();privatestaticfinalString TAG ="ServiceActivity-->";privateFirstService mFirstService;privateboolean isBinder;// 服务是否绑定@Overrideprotectedvoid onCreate(Bundle savedInstanceState){super.onCreate(savedInstanceState);setContentView(R.layout.activity_service);findViewById(R.id.btn_start_service).setOnClickListener(this);findViewById(R.id.btn_stop_service).setOnClickListener(this);findViewById(R.id.btn_bound_service).setOnClickListener(this);findViewById(R.id.btn_unbound_service).setOnClickListener(this);findViewById(R.id.btn_get_number).setOnClickListener(this);}@Overridepublicvoid onClick(View v){switch(v.getId()){case R.id.btn_start_service:Intent intent =newIntent(ServiceActivity.this,FirstService.class);intent.putExtra("name","Zhangsan");startService(intent);break;case R.id.btn_stop_service:Intent intent2 =newIntent(ServiceActivity.this,FirstService.class);stopService(intent2);break;case R.id.btn_bound_service:if(!isBinder){Intent intent3 =newIntent(ServiceActivity.this,FirstService.class);bindService(intent3, connection, BIND_AUTO_CREATE);isBinder =true;}break;case R.id.btn_unbound_service:if(isBinder){unbindService(connection);isBinder =false;}break;case R.id.btn_get_number:if(mFirstService ==null){Toast.makeText(getApplicationContext(),"请先绑定服务",Toast.LENGTH_SHORT).show();}else{Toast.makeText(getApplicationContext(),"得到的随机数为:"+ mFirstService.getRandomNumber(),Toast.LENGTH_SHORT).show();}break;}}classMyServiceConnectionimplementsServiceConnection{@Overridepublicvoid onServiceConnected(ComponentName name,IBinder service){Log.i(TAG,"onServiceConnected");FirstService.MyBinder myBinder =(FirstService.MyBinder) service;mFirstService = myBinder.getService();}@Overridepublicvoid onServiceDisconnected(ComponentName name){Log.i(TAG,"onServiceDisconnected");}}}
这里有一个小插曲:一起写出来大家分享下:记得以前学java基础时,老师曾说过一个java文件中只能有一个public类,类名称必须与java文件名相同,为什么这个FirstService中有两个public类,只是因为MyBinder虽然是一个public class,但是MyBinder是一个内部类,这里必须要用public修饰,否则其他包就访问不到这个内部类了;
Service是什么?Service又不是什么?的更多相关文章
- Failed to stop iptables.service: Unit iptables.service not loaded.
redhat 7 [root@lk0 ~]# service iptables stop Redirecting to /bin/systemctl stop iptables.service Fai ...
- Local System、Local Service與Network Service
CreateService参数介绍SC_HANDLE CreateService( SC_HANDLE hSCManager, //服务控制管理程序维护的登记数据库的句柄,由系统函数OpenSCMan ...
- 关于Failed to check the status of the service com.taotao.service.ItemService. No provider available fo
原文:http://www.bubuko.com/infodetail-2250226.html 项目中用dubbo发生: Failed to check the status of the serv ...
- Failed to stop iptables.service: Unit iptables.service not loaded.解决方法
CentOS7中执行 service iptables start/stop 会报错Failed to start iptables.service: Unit iptables.service fa ...
- 从Web Service和Remoting Service引出WCF服务
本篇先通过Web Service和Remoting Service创建服务,抛砖引玉,再体验WCF服务.首先一些基本面: 什么是WCF? Windows Communication Foundatio ...
- Android Service总结02 service介绍
Android Service总结02 service介绍 版本 版本说明 发布时间 发布人 V1.0 介绍了Service的种类,常用API,生命周期等内容. 2013-03-16 Skywang ...
- Service Fabric —— Stateful Service 概念
作者:潘罡 (Van Pan) @ Microsoft 上节中我们谈到了Service Fabric最底层的两个概念,一个是针对硬件层面而言的Node Type和Node.另一个是Applicatio ...
- CentOS 7 防火墙 出现Failed to start iptables.service: Unit iptables.service failed to load
错误信息如下: [root]# service iptables start Redirecting to /bin/systemctl start iptables.service Failed t ...
- service: no such service mysqld 与MySQL的开启,关闭和重启
1.问题原因与解决办法 因为修改了MySQL临时文件的目录后,使用service mysqld restart重启MySQL出现如下错误: service: no such service mysql ...
- dubbo Failed to check the status of the service com.user.service.UserService. No provider available for the service
Caused by: org.springframework.beans.factory.BeanCreationException: Error creating bean with name 'u ...
随机推荐
- 如何快速开发SPA应用
前言 web早已经进入了2.0时代了,如今的网页大有往系统应用级别的方向发展的趋势,再也不是以前的简单展示信息的界面了.如今很多webapp已经做到了原生应用的功能,并且运用自身的优势逐步取代之.HT ...
- Android开发学习之路-RecyclerView滑动删除和拖动排序
Android开发学习之路-RecyclerView使用初探 Android开发学习之路-RecyclerView的Item自定义动画及DefaultItemAnimator源码分析 Android开 ...
- Java环境变量-Linux环境
环境变量说明 JAVA_HOME 它指向jdk的安装目录,Eclipse/NetBeans/Tomcat等软件就是通过搜索JAVA_HOME变量来找到并使用安装好的jdk. PATH 作用是指定命令搜 ...
- JavaScript中函数函数的定义与变量的声明<基础知识一>
1.JavaScript中函数的三种构造方式 a.function createFun(){ } b.var createFun=function (){ } c.var createFun=new ...
- jQuery 插件-(初体验一)
1.jquery有2个扩展方法: jquery.fn.extend=jquery.prototype.extend jquery.extend (两者的区别放在后面文章说) 2.具体实例结构: //创 ...
- 3. 解析 struts.xml 文件
1. struts.xml 文件基本配置: 主要放在资源路径下,配置 sturts2相关的 Action , 拦截器等配置 <struts> <!-- 设置常量 --> < ...
- Log4net入门(日志文件篇)
在上一篇Log4net入门(控制台篇)中,我们将日志信息输出到控制台中,在这一篇中,我们将描述如何将日志信息写到文件中.要将日志信息写入文件非常简单,只需要在Log4net.config配置文件中添加 ...
- ASP.NET MVC RouteExistingFiles
遇到这样一个问题:项目是 MVC,但也包含 WebForm 的页面,RouteConfig 中设置了这样一个路由: routes.MapRoute( name: "SubjectIndex& ...
- Linux 平台静默安装 Oracle客户端
需求:Linux平台,安装完整版Oracle客户端 Tips:如果只是用到sqlldr,sqlplus功能,可以参考<Linux上oracle精简版客户端快速部署>快速部署精简版:如果需要 ...
- Python基础(三)
本章内容: 深浅拷贝 函数(全局与局部变量) 内置函数 文件处理 三元运算 lambda 表达式 递归(斐波那契数列) 冒泡排序 深浅拷贝 一.数字和字符串 对于 数字 和 字符串 而言,赋值.浅拷贝 ...