Android Service 服务(一)—— Service

Android Service 服务(一)—— Service

大家好,又见面了,我是全栈君,祝每个程序员都可以多学几门语言。

一、 Service简单介绍

Service是android 系统中的四大组件之中的一个(Activity、Service、BroadcastReceiver、ContentProvider),它跟Activity的级别差点儿相同,但不能自己执行仅仅能后台执行,而且能够和其它组件进行交互。service能够在非常多场合的应用中使用,比方播放多媒体的时候用户启动了其它Activity这个时候程序要在后台继续播放,比方检測SD卡上文件的变化,再或者在后台记录你地理信息位置的改变等等,总之服务总是藏在后台的。

Service的启动有两种方式:context.startService() context.bindService()


二、 Service启动流程

context.startService() 启动流程:

context.startService()  -> onCreate()  -> onStart()  -> Service running  -> context.stopService()  -> onDestroy()  -> Service stop 


假设Service还没有执行,则android先调用onCreate(),然后调用onStart();

假设Service已经执行,则仅仅调用onStart(),所以一个Service的onStart方法可能会反复调用多次。 

假设stopService的时候会直接onDestroy,假设是调用者自己直接退出而没有调用stopService的话,Service会一直在后台执行,该Service的调用者再启动起来后能够通过stopService关闭Service。

所以调用startService的生命周期为:
onCreate –>
onStart (可多次调用) –>
onDestroy


context.bindService()启动流程:

context.bindService()  -> onCreate()  -> onBind()  -> Service running  -> onUnbind()  -> onDestroy()  -> Service stop

 

onBind()将返回给client一个IBind接口实例,IBind同意client回调服务的方法,比方得到Service的实例、执行状态或其它操作。这个时候把调用者(Context,比如Activity)会和Service绑定在一起,Context退出了,Srevice就会调用onUnbind->onDestroy对应退出。 

所以调用bindService的生命周期为:onCreate –> onBind(仅仅一次,不可多次绑定) –> onUnbind –> onDestory。

在Service每一次的开启关闭过程中,仅仅有onStart可被多次调用(通过多次startService调用),其它onCreate,onBind,onUnbind,onDestory在一个生命周期中仅仅能被调用一次。

 Android Service 服务(一)—— Service

三、 Service生命周期 

Service的生命周期并不像Activity那么复杂,它仅仅继承了onCreate()、onStart()、onDestroy()三个方法

当我们第一次启动Service时,先后调用了onCreate()、onStart()这两个方法;当停止Service时,则执行onDestroy()方法。

这里须要注意的是,假设Service已经启动了,当我们再次启动Service时,不会在执行onCreate()方法,而是直接执行onStart()方法。

它能够通过Service.stopSelf()方法或者Service.stopSelfResult()方法来停止自己,仅仅要调用一次stopService()方法便能够停止服务,不管调用了多少次的启动服务方法。

四、 Service演示样例

以下我做了一个简单的音乐播放的应用,分别使用startService和bindService来启动本地的服务。

Activity

public class PlayMusicService extends Activity implements OnClickListener {

	private Button playBtn;
	private Button stopBtn;
	private Button pauseBtn;
	private Button exitBtn;
	private Button closeBtn;

	private Intent intent;
	
	@Override
	public void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		setContentView(R.layout.music_service);

		playBtn = (Button) findViewById(R.id.play);
		stopBtn = (Button) findViewById(R.id.stop);
		pauseBtn = (Button) findViewById(R.id.pause);
		exitBtn = (Button) findViewById(R.id.exit);
		closeBtn = (Button) findViewById(R.id.close);
		
		playBtn.setOnClickListener(this);
		stopBtn.setOnClickListener(this);
		pauseBtn.setOnClickListener(this);
		exitBtn.setOnClickListener(this);
		closeBtn.setOnClickListener(this);

	}

	@Override
	public void onClick(View v) {
		int op = -1;
		intent = new Intent("com.homer.service.musicService");

		switch (v.getId()) {
		case R.id.play:								// play music
			op = 1;
			break;
		case R.id.stop:								// stop music
			op = 2;
			break;
		case R.id.pause:							// pause music
			op = 3;
			break;
		case R.id.close:							// close activity
			this.finish();
			break;
		case R.id.exit:								// stopService
			op = 4;
			stopService(intent);
			this.finish();
			break;
		}

		Bundle bundle = new Bundle();
		bundle.putInt("op", op);
		intent.putExtras(bundle);
		
		startService(intent);							// startService
	}
	
	@Override
	public void onDestroy(){
		super.onDestroy();

		if(intent != null){
			stopService(intent);
		}
	}
}

Service

public class MusicService extends Service {
	private static final String TAG = "MyService";
	
	private MediaPlayer mediaPlayer;

	@Override
	public IBinder onBind(Intent arg0) {
		return null;
	}

	@Override
	public void onCreate() {
		Log.v(TAG, "onCreate");
		Toast.makeText(this, "show media player", Toast.LENGTH_SHORT).show();

		if (mediaPlayer == null) {
			mediaPlayer = MediaPlayer.create(this, R.raw.tmp);
			mediaPlayer.setLooping(false);
		}
	}

	@Override
	public void onDestroy() {
		Log.v(TAG, "onDestroy");
		Toast.makeText(this, "stop media player", Toast.LENGTH_SHORT);
		if (mediaPlayer != null) {
			mediaPlayer.stop();
			mediaPlayer.release();
		}
	}

	@Override
	public void onStart(Intent intent, int startId) {
		Log.v(TAG, "onStart");
		if (intent != null) {
			Bundle bundle = intent.getExtras();
			if (bundle != null) {
				int op = bundle.getInt("op");
				switch (op) {
				case 1:
					play();
					break;
				case 2:
					stop();
					break;
				case 3:
					pause();
					break;
				}
			}
		}
	}

	public void play() {
		if (!mediaPlayer.isPlaying()) {
			mediaPlayer.start();
		}
	}

	public void pause() {
		if (mediaPlayer != null && mediaPlayer.isPlaying()) {
			mediaPlayer.pause();
		}
	}

	public void stop() {
		if (mediaPlayer != null) {
			mediaPlayer.stop();
			try {
				mediaPlayer.prepare();	// 在调用stop后假设须要再次通过start进行播放,须要之前调用prepare函数
			} catch (IOException ex) {
				ex.printStackTrace();
			}
		}
	}
}

AndroidManifest.xml

注冊activity

        <activity
            android:name=".service.PlayMusicService"
            android:label="@string/app_name" />

注冊service

        <service
            android:name=".service.MusicService"
            android:enabled="true" >
            <intent-filter>
                <action android:name="com.homer.service.musicService" />
            </intent-filter>
        </service>

五、 代码解析

1、Activity中,PlayMusicService中通过重写OnClickListener 接口onClick()方法实现对播放音乐的控制,把音乐各种操作用数字通过Intent传递给service

然后通过构造一个Intent , intent = new Intent(“com.homer.service.musicService“); 

当中,com.homer.service.musicService是 AndroidManifest.xml 对service的定义,即上面“注冊service”

2、Activity中,音乐播放的控制,利用Bundle绑定数字op后,通过 startService(intent); 服务后发送出去


Bundle bundle = new Bundle();
bundle.putInt(“op”, op);
intent.putExtras(bundle);

startService(intent);

3、 Service中,会处理Activity启动的 startService(intent);服务,依次调用service的启动过程:onCreate –> onStart(可多次调用) –> onDestroy

onCreate(),  创建mediaPlayer

onStart(),      通过获取Bundle bundle = intent.getExtras();,提取int op = bundle.getInt(“op”);,然后执行响应的音乐播放操作

onDestroy(),停止并释放mediaPlayer音乐资源,假设当执行context.stopService()时调用此方法

4、Activity中,onClick()函数中close与exit是执行含义是不同的:

close : 仅仅是执行了this.finish(); 关闭了本Activity窗口,service并没有被关掉,音乐依旧会继续在后台播放

exit  : 先调用了stopService(intent); 关闭了service服务,在Service中会调用3中的onDestroy()停止并释放音乐资源,后才执行this.finish(); 关闭了本Activity窗口

源代码下载

六、 拓展知识(进程和声明周期)

Android操作系统尝试尽可能长时间的保持应用的进程,但当可用内存非常低时终于要移走一部分进程。如何确定那些程序能够执行,那些要被销毁,Android让每个进程在一个重要级的基础上执行,重要级低的进程最有可能被淘汰,一共同拥有5级,以下这个列表就是依照重要性排列的:

1 一个前台进程显示的是用户此时须要处理和显示的。下列的条件有不论什么一个成立,这个进程都被觉得是在前台执行的。
        a 与用户正发生交互的。
        b 它控制一个与用户交互的必须的主要的服务。
        c 有一个正在调用生命周期的回调函数的service(如onCreate()、onStar()、onDestroy())
        d 它有一个正在执行onReceive()方法的广播接收对象。
仅仅有少数的前台进程能够在不论什么给定的时间内执行,销毁他们是系统万不得已的、最后的选择——当内存不够系统继续执行下去时。通常,在这一点上,设备已经达到了内存分页状态,所以杀掉一些前台进程来保证能够响应用户的需求。

2 一个可用进程没有不论什么前台组件,但它仍然能够影响到用户的界面。以下两种情况发生时,能够称该进程为可用进程。
        它是一个非前台的activity,但对用户仍然可用(onPause()方法已经被调用)这是可能发生的,比如:前台的activity是一个同意上一个activity可见的对话框,即当前activity半透明,能看到前一个activity的界面,它是一个服务于可用activity的服务。

3 一个服务进程是一个通过调用startService()方法启动的服务,而且不属于前两种情况。虽然服务进程没有直接被用户看到,但他们确实是用户所关心的,比方后台播放音乐或网络下载数据。所以系统保证他们的执行,直到不能保证全部的前台可见程序都正常执行时才会终止他们。

4 一个后台进程就是一个非当前正在执行的activity(activity的onStop()方法已经被调用),他们不会对用户体验造成直接的影响,当没有足够内存来执行前台可见程序时,他们将会被终止。通常,后台进程会有非常多个在执行,所以他们维护一个LRU近期使用程序列表来保证常常执行的activity能最后一个被终止。假设一个activity正确的实现了生命周期的方法,而且保存它当前状态,杀死这些进程将不会影响到用户体验。

5 一个空线程没有执行不论什么可用应用程序组,保留他们的唯一原因是为了设立一个缓存机制,来加快组件启动的时间。系统常常杀死这些内存来平衡系统的整个系统的资源,进程缓存和基本核心缓存之间的资源。
Android把进程里优先级最高的activity或服务,作为这个进程的优先级。比如,一个进程拥有一个服务和一个可见的activity,那么这个进程将会被定义为可见进程,而不是服务进程。

此外,假设别的进程依赖某一个进程的话,那么被依赖的进程会提高优先级。一个进程服务于还有一个进程,那么提供服务的进程不会低于获得服务的进程。比如,假设进程A的一个内容提供商服务于进程B的一个client,或者进程A的一个service被进程B的一个组件绑定,那么进程A至少拥有和进程B一样的优先级,或者更高。

由于一个执行服务的进程的优先级高于执行后台activity的进程,一个activity会准备一个长时间执行的操作来启动一个服务,而不是启动一个线程–尤其是这个操作可能会拖垮这个activity。比如后台播放音乐的同一时候,通过照相机向server发送一张照片,启动一个服务会保证这个操作至少执行在service 进程的优先级下,不管这个activity发生了什么,广播接收者应该作为一个空服务而不是简单的把耗时的操作单独放在一个线程里。 

參考推荐:

android service 学习

Android Service生命周期及使用方法

Android生命周期之service/Broadcast

Android BroadcastReceiver 学习

Android之BroadcastReceiver的使用

Android BroadcastReceiver启动Service 

Service (android developer)

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/118924.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • ideal2019 30天试用结束了,在线激活码【最新永久激活】

    (ideal2019 30天试用结束了,在线激活码)这是一篇idea技术相关文章,由全栈君为大家提供,主要知识点是关于2021JetBrains全家桶永久激活码的内容IntelliJ2021最新激活注册码,破解教程可免费永久激活,亲测有效,下面是详细链接哦~https://javaforall.net/100143.html2JTX0APX6F-eyJsa…

    2022年3月30日
    42
  • linux -umask

    linux -umask

    2021年9月9日
    53
  • H2 数据库入门和基本使用「建议收藏」

    H2 数据库入门和基本使用「建议收藏」1、下载安装包H2官网:http://h2database.com/html/main.html2、安装H2数据库选择安装目录->点击下一步->安装->完成

    2022年10月12日
    2
  • HTML的dpr

    HTML的dpr只是为了自己以后查找方便 看不懂勿怪 DPI PPI 屏幕像素密度或屏幕每英寸的像素数量 dips 手机尺寸或独立像素物理像素 手机分辨率 dpr nbsp window devicePixelR js 可以拿到 css 定义 psd 转换 rem 单位 rem px 基准值基准值 手机宽高 dpr 10 乘以 dpr 是因为页面有可能缩放除以 10 是为了取整

    2025年9月22日
    4
  • CIFAR-10 数据集「建议收藏」

    CIFAR-10 数据集「建议收藏」CIFAR-10数据集简介CIFAR-10是由Hinton的学生AlexKrizhevsky和IlyaSutskever整理的一个用于识别普适物体的小型数据集。一共包含10个类别的RGB彩色图片:飞机(a叩lane)、汽车(automobile)、鸟类(bird)、猫(cat)、鹿(deer)、狗(dog)、蛙类(frog)、马(hor…

    2022年4月19日
    40
  • MacPorts_mac查看使用电脑时间

    MacPorts_mac查看使用电脑时间1.MacPort中第三方软件下载包存放的默认路径是:/opt/local/var/macports/distfiles/为了提高安装速度,可以在安装新port时直接将此目录下的文件拷贝到新的MacPort相同的目录中就可以避免Port去网上下载。2.使用MacPort前应该首先更新Port的index输入:$ sudoport-vselfupdate (强烈推荐第一

    2022年9月21日
    3

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号