首页 > 浅谈Android四大组件之Service

浅谈Android四大组件之Service

一:Service简介

Android开发中,当需要创建在后台运行的程序的时候,就要使用到Service。

1:Service(服务)是一个没有用户界面的在后台运行执行耗时操作的应用组件。其他应用组件能够启动Service,并且当用户切换到另外的应用场景,Service将持续在后台运行。另外,一个组件能够绑定到一个service与之交互(IPC机制),例如,一个service可能会处理网络操作,播放音乐,操作文件I/O或者与内容提供者(content provider)交互,所有这些活动都是在后台进行。

 

2:Service有两种状态,“启动的”和“绑定:

通过startService()启动的服务处于“启动的”状态,一旦启动,service就在后台运行,即使启动它的应用组件已经被销毁了。通常started状态的service执行单任务并且不返回任何结果给启动者。比如当下载或上传一个文件,当这项操作完成时,service应该停止它本身。

 

还有一种“绑定”状态的service,通过调用bindService()来启动,一个绑定的service提供一个允许组件与service交互的接口,可以发送请求、获取返回结果,还可以通过夸进程通信来交互(IPC)。绑定的service只有当应用组件绑定后才能运行,多个组件可以绑定一个service,当调用unbind()方法时,这个service就会被销毁了。

 

3:serviceactivity一样都存在与当前进程的主线程中,所以,一些阻塞UI的操作,比如耗时操作不能放在service里进行,比如另外开启一个线程来处理诸如网络请求的耗时操作。如果在service里进行一些耗CPU和耗时操作,可能会引发ANR警告,这时应用会弹出是强制关闭还是等待的对话框。所以,对service的理解就是和activity平级的,只不过是看不见的,在后台运行的一个组件,这也是为什么和activity同被说为Android的基本组件。

 

二:Service的生命周期

Android Service生命周期与Activity生命周期是相似的,但是也存在一些细节上也存在着重要的不同:

onCreateonStart是不同的

通过从客户端调用Context.startService(Intent)方法我们可以启动一个服务。如果这个服务还没有运行,Android将启动它并且在onCreate方法之后调用它的onStart方法。如果这个服务已经在运行,那么它的onStart方法将被新的Intent再次调用。所以对于单个运行的Service它的onStart方法被反复调用是完全可能的并且是很正常的。

onResumeonPause以及onStop是不需要的

回调一个服务通常是没有用户界面的,所以我们也就不需要onPauseonResume或者onStop方法了。无论何时一个运行中的Service它总是在后台运行。

onBind

如果一个客户端需要持久的连接到一个服务,那么他可以调用Context.bindService方法。如果这个服务没有运行方法将通过调用onCreate方法去创建这个服务但并不调用onStart方法来启动它。相反,onBind方法将被客户端的Intent调用,并且它返回一个IBind对象以便客户端稍后可以调用这个服务。同一服务被客户端同时启动和绑定是很正常的。

onDestroy

Activity一样,当一个服务被结束是onDestroy方法将会被调用。当没有客户端启动或绑定到一个服务时Android将终结这个服务。与很多Activity时的情况一样,当内存很低的时候Android也可能会终结一个服务。如果这种情况发生,Android也可能在内存够用的时候尝试启动被终止的服务,所以你的服务必须为重启持久保存信息,并且最好在onStart方法内来做。

 

三.Service的启动方式

Service的启动方式有如下2种

CstartService()和bindService(),这两种方式对Service生命周期的影响是不同的

startservice是由其他组件调用startService()方法启动的,这导致服务的onStartCommand()方法被调用,当服务是started状态时,其生命周期与启动它的组件无关,并且可以在后台无限期运行,即使启动服务的组件已经被销毁。因此,服务需要在完成任务后调用stopself()方法停止,或者有其他组件调用stopService()方法停止。

使用bindService()方法启用服务,调用者与服务绑定在了一起,调用者一旦退出,服务也就终止,大有‘不求同生,但求同死’的特点

Android应用中每个Service都必须要在AndroidMainifest.Xml配置文件中声明。

 四:2种启动方式的具体例子

 

startService()方式的生命周期: 

启动时,startService –> onCreate() –> onStart()


停止时,stopService –> onDestroy()

 

如果调用者直接退出而没有停止Service,则Service 会一直在后台运行
Context.startService()方法启动服务,在服务未被创建时,系统会先调用服务的onCreate()方法,接着调用onStart()方法。如果调用startService()方法前服务已经被创建,多次调用startService()方法并不会导致多次创建服务,但会导致多次调用onStart()方法。采用startService()方法启动的服务,只能调用Context.stopService()方法结束服务,服务结束时会调用onDestroy()方法。

 

startService()方式启动 Service的方法:

xml version="1.0" encoding="utf-8"?> 
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:orientation="vertical" android:layout_width="fill_parent" android:layout_height="fill_parent" > <Button android:id="@+id/startBtn" android:layout_width="match_parent" android:layout_height="wrap_content" android:text="启动 Service" /> <Button android:id="@+id/stopBtn" android:layout_width="match_parent" android:layout_height="wrap_content" android:text="停止 Service" /> 
LinearLayout> 

MainActivity.java

package com.android.service.activity;  import android.app.Activity;  
import android.content.Intent;  
import android.os.Bundle;  
import android.view.View;  
import android.view.View.OnClickListener;  
import android.widget.Button;  public class MainActivity extends Activity  
{  private Button startBtn;  private Button stopBtn;  @Override public void onCreate(Bundle savedInstanceState)  {  super.onCreate(savedInstanceState);  setContentView(R.layout.main);  startBtn = (Button) findViewById(R.id.startBtn);  stopBtn = (Button) findViewById(R.id.stopBtn);  //添加监听器  
        startBtn.setOnClickListener(listener);  stopBtn.setOnClickListener(listener);  }  //启动监听器  private OnClickListener listener=new OnClickListener()  {         @Override public void onClick(View v)  {  Intent intent=new Intent(MainActivity.this, ServiceDemo.class);  switch (v.getId())  {  case R.id.startBtn:  startService(intent);  break;  case R.id.stopBtn:  stopService(intent);  break;  default:  break;  }             }  };  
} 

新建ServiceDemo类,ServiceDemo.java如下

package com.android.service.activity;  import android.app.Service;  
import android.content.Intent;  
import android.os.IBinder;  
import android.util.Log;  public class ServiceDemo extends Service  
{  private static final String TAG="Test";  @Override //Service时被调用  public void onCreate()  {  Log.i(TAG, "Service onCreate--->");  super.onCreate();  }  @Override //当调用者使用startService()方法启动Service时,该方法被调用  public void onStart(Intent intent, int startId)  {  Log.i(TAG, "Service onStart--->");  super.onStart(intent, startId);  }  @Override //当Service不在使用时调用  public void onDestroy()  {  Log.i(TAG, "Service onDestroy--->");  super.onDestroy();  }  @Override //当使用startService()方法启动Service时,方法体内只需写return null  public IBinder onBind(Intent intent)  {  return null;  }  
}  

在AndroidManifest.xml文件中添加16~21行的声明

xml version="1.0" encoding="utf-8"?> 
<manifest xmlns:android="http://schemas.android.com/apk/res/android" package="com.android.service.activity" android:versionCode="1" android:versionName="1.0"> <uses-sdk android:minSdkVersion="10" /> <application android:icon="@drawable/icon" android:label="@string/app_name"> <activity android:name=".MainActivity" android:label="@string/app_name"> <intent-filter> <action android:name="android.intent.action.MAIN" /> <category android:name="android.intent.category.LAUNCHER" /> intent-filter> activity> <service android:name=".ServiceDemo" >    <intent-filter> <action android:name="android.intent.action.MAIN" /> <category android:name="android.intent.category.LAUNCHER" /> intent-filter> service>    application> 
manifest> 

效果如下:

当点击按钮时,先后执行了Service中onCreate()->onStart()这两个方法,LogCat显示如下:

当点击 按钮时,Service则执行了onDestroy()方法,LogCat显示如下:

 

退出后,进入Settings(设置)->Applications(应用)->Running Services(正在运行的服务)看一下我们新启动了的服务,效果图如下:

 

 

bindService()方式的生命周期: 

绑定时,bindService -> onCreate() –> onBind()
调用者退出了,即解绑定时,Srevice就会unbindService –>onUnbind() –> onDestory()
 
用Context.bindService()方法启动服务,在服务未被创建时,系统会先调用服务的onCreate()方法,接着调用onBind()方法。这个时候调用者和服务绑定在一起,调用者退出了,系统就会先调用服务的onUnbind()方法,接着调用onDestroy()方法。如果调用bindService()方法前服务已经被绑定,多次调用bindService()方法并不会导致多次创建服务及绑定(也就是说onCreate()和onBind()方法并不会被多次调用)。如果调用者希望与正在绑定的服务解除绑定,可以调用unbindService()方法,调用该方法也会导致系统调用服务的onUnbind()-->onDestroy()方法。
 bindService()方式启动 Service的方法:
绑定Service需要三个参数:bindService(intent, conn, Service.BIND_AUTO_CREATE);
第一个:Intent对象
第二个:ServiceConnection对象,创建该对象要实现它的onServiceConnected()和 onServiceDisconnected()来判断连接成功或者是断开连接
第三个:如何创建Service,一般指定绑定的时候自动创建。
例子如下:
main.xml
xml version="1.0" encoding="utf-8"?> 
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:orientation="vertical" android:layout_width="fill_parent" android:layout_height="fill_parent" > <Button   android:text="启动Service"   android:id="@+id/startBtn1"   android:layout_width="match_parent"   android:layout_height="wrap_content" />    <Button   android:text="停止Service"   android:id="@+id/stopBtn2"   android:layout_width="match_parent"   android:layout_height="wrap_content" />    <Button   android:text="绑定Service"   android:id="@+id/bindBtn3"   android:layout_width="match_parent"   android:layout_height="wrap_content" /> <Button   android:text="解除绑定"   android:id="@+id/unbindBtn4"   android:layout_width="match_parent"   android:layout_height="wrap_content" /> 
LinearLayout> 

MainActivity.java

package com.android.bindservice.activity;  import android.app.Activity;  
import android.app.Service;  
import android.content.ComponentName;  
import android.content.Intent;  
import android.content.ServiceConnection;  
import android.os.Bundle;  
import android.os.IBinder;  
import android.util.Log;  
import android.view.View;  
import android.view.View.OnClickListener;  
import android.widget.Button;  public class MainActivity extends Activity {  // 声明Button  private Button startBtn,stopBtn,bindBtn,unbindBtn;  @Override public void onCreate(Bundle savedInstanceState) {  super.onCreate(savedInstanceState);  // 设置当前布局视图  
        setContentView(R.layout.main);  // 实例化Button  startBtn = (Button)findViewById(R.id.startBtn1);  stopBtn = (Button)findViewById(R.id.stopBtn2);  bindBtn = (Button)findViewById(R.id.bindBtn3);  unbindBtn = (Button)findViewById(R.id.unbindBtn4);  // 添加监听器  
        startBtn.setOnClickListener(startListener);  stopBtn.setOnClickListener(stopListener);  bindBtn.setOnClickListener(bindListener);  unbindBtn.setOnClickListener(unBindListener);  }  // 启动Service监听器  private OnClickListener startListener = new OnClickListener() {  @Override public void onClick(View v) {  // 创建Intent  Intent intent = new Intent();  // 设置Class属性  intent.setClass(MainActivity.this, BindService.class);  // 启动该Service  
            startService(intent);  }  };  // 停止Service监听器  private OnClickListener stopListener = new OnClickListener() {  @Override public void onClick(View v) {  // 创建Intent  Intent intent = new Intent();  // 设置Class属性  intent.setClass(MainActivity.this, BindService.class);  // 启动该Service  
            stopService(intent);  }  };  // 连接对象  private ServiceConnection conn = new ServiceConnection() {  @Override public void onServiceConnected(ComponentName name, IBinder service) {  Log.i("Service", "连接成功!");  }  @Override public void onServiceDisconnected(ComponentName name) {  Log.i("Service", "断开连接!");  }  };  // 綁定Service监听器  private OnClickListener bindListener = new OnClickListener() {  @Override public void onClick(View v) {  // 创建Intent  Intent intent = new Intent();  // 设置Class属性  intent.setClass(MainActivity.this, BindService.class);  // 绑定Service  
            bindService(intent, conn, Service.BIND_AUTO_CREATE);  }  };  // 解除绑定Service监听器  private OnClickListener unBindListener = new OnClickListener() {  @Override public void onClick(View v) {  // 创建Intent  Intent intent = new Intent();  // 设置Class属性  intent.setClass(MainActivity.this, BindService.class);  // 解除绑定Service  
            unbindService(conn);  }  };    
} 

BindService.java

package com.android.bindservice.activity;  import android.app.Service;  
import android.content.Intent;  
import android.os.IBinder;  
import android.util.Log;  public class BindService extends Service{  private static final String TAG="Test";  //返回null  public IBinder onBind(Intent intent) {  Log.i(TAG, "Service onBind--->");  return null;  }  // Service创建时调用  public void onCreate() {  Log.i(TAG, "Service onCreate--->");  }  // 当客户端调用startService()方法启动Service时,该方法被调用  public void onStart(Intent intent, int startId) {  Log.i(TAG, "Service onStart--->");  }  // 当Service不再使用时调用  public void onDestroy() {  Log.i(TAG, "Service onDestroy--->");  }  // 当解除绑定时调用  public boolean onUnbind(Intent intent) {    Log.i(TAG, "Service onUnbind--->");    return super.onUnbind(intent);    }    
}  

 

在AndroidManifest.xml文件中添加16~21行的声明

xml version="1.0" encoding="utf-8"?> 
<manifest xmlns:android="http://schemas.android.com/apk/res/android" package="com.android.bindservice.activity" android:versionCode="1" android:versionName="1.0"> <uses-sdk android:minSdkVersion="10" /> <application android:icon="@drawable/icon" android:label="@string/app_name"> <activity android:name=".MainActivity" android:label="@string/app_name"> <intent-filter> <action android:name="android.intent.action.MAIN" /> <category android:name="android.intent.category.LAUNCHER" /> intent-filter> activity> <service android:name=".BindService" >    <intent-filter> <action android:name="android.intent.action.MAIN" /> <category android:name="android.intent.category.LAUNCHER" /> intent-filter> service> application> 
manifest> 

效果如下:

当点击按钮时,先后执行了Service中onCreate()->onStart()这两个方法,LogCat显示如下:

当点击按钮,则Service执行了onUnbind()方法,LogCat显示如下:

 

当点击按钮,则Service执行了onUnbind()方法,LogCat显示如下:

 

 

转载于:https://www.cnblogs.com/FENGXUUEILIN/p/5663280.html

更多相关:

  • android:id 为控件指定相应的IDandroid:text 指定控件的文本,置尽量使用strings.xmlandroid:grivity 指定控件的基本位置 ,比如举重,居右,android:padding 指定控件的内边距,控件当中的内容android:singleLine 如果设置为真的话,则将控件的内容在同一行当中显示...

  • 布局主要分两个 其中主布局是

  • 大家平时见到的最多的可能就是Frame动画了,Android中当然也少不了它。它的使用更加简单,只需要创建一个 AnimationDrawabledF对象来表示Frame动画,然后通过addFrame 方法把每一帧要显示的内容添加进去,并设置播放间隔时间,本例子中间隔时间为5S, 最后通过start 方法就可。 以播放这个动画了,...

  • 作业要求: 作一个显示框里面分成三行 一二行占这个框的1/2 第三行独占1/2 第三行里面分成两列第一列占25%,第二列占75%。 屏幕显示效果 实现步骤:  

  • 来源:公众号|计算机视觉工坊(系投稿)作者:仲夏夜之星「3D视觉工坊」技术交流群已经成立,目前大约有12000人,方向主要涉及3D视觉、CV&深度学习、SLAM、三维重建、点云后处理、自动驾驶、CV入门、三维测量、VR/AR、3D人脸识别、医疗影像、缺陷检测、行人重识别、目标跟踪、视觉产品落地、视觉竞赛、车牌识别、硬件选型、学术交流、...

  • 点云PCL免费知识星球,点云论文速读。文章:Real-Time LIDAR-Based Urban Road and Sidewalk Detection for Autonomous Vehicles作者:Ern˝o Horváth  , Claudiu Pozna ,and Miklós Unger编译:点云PCL代码:http...

  • 文章:Semantic Histogram Based Graph Matching for Real-Time Multi-Robot Global Localization in Large Scale Environment作者:Xiyue Guo, Junjie Hu, Junfeng Chen, Fuqin Deng, T...

  • 点云PCL免费知识星球,点云论文速读。文章:Robust Place Recognition using an Imaging Lidar作者:Tixiao Shan, Brendan Englot, Fabio Duarte, Carlo Ratti, and Daniela Rus编译:点云PCL(ICRA 2021)开源代码:...

  • 文章:A Survey of Calibration Methods for Optical See-Through Head-Mounted Displays作者:Jens Grubert , Yuta Itoh, Kenneth Moser编译:点云PCL本文仅做学术分享,如有侵权,请联系删除。欢迎各位加入免费知识星球,获取PD...