Android Service使用方法--简单音乐播放实例

鉴客 发布于 2011/11/14 14:03
阅读 40K+
收藏 47

Service翻译成中文是服务,熟悉Windows 系统的同学一定很熟悉了。Android里的Service跟Windows里的Service功能差不多,就是一个不可见的进程在后台执行。

Android中的服务,它与Activity不同,它是不能与用户交互的,不能自己启动的,运行在后台的程序,如果我们退出应用时,Service进程并没有结束,它仍然在后台运行,例如我们打开一个音乐播放器来听音乐,在听音乐的同时也想做下其它的事情,比如上网聊Q、或者上网浏览新闻之类的事情。这样的话,我们就需要用到Service服务了。下面我们以一个简单的音乐播放器的实例来说明下Service的生命周期和Service的使用。

下面是音乐播放器Demo的程序结构图:

Android Service 的生命周期:

Android中Service的生命周期并不是很复杂,只是继承了onCreate(), onStart(), onDestory()三个方法。当我们第一次启动Service服务时,调用onCreate() --> onStart()两个方法,当停止Service服务时,调用onDestory()方法。如果Service已经启动了,第二次再启动同一个服务时,就只是调用 onStart() 这个方法了。

Android Service 的使用:

[1] 参照上面的程序结构图,我们可以创建一个Android程序,在src目录下创建一个Activity,一个继承自Service类的服务类;同时在资源文件夹res目录下创建一个raw的文件夹存放音频文件,如把music.mp3音乐文件放在该目录下。该程序的主界面如下:

[2] layout目录下的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"
    >
    <TextView  
       android:layout_width="fill_parent" 
       android:layout_height="wrap_content" 
       android:text="Welcome to Andy's blog!"
       android:textSize="16sp"/>   
    <TextView  
       android:layout_width="fill_parent" 
       android:layout_height="wrap_content" 
       android:text="音乐播放服务"/>
    <Button
       android:id="@+id/startMusic" 
       android:layout_width="wrap_content"
       android:layout_height="wrap_content"
       android:text="开启音乐播放服务"/>
    <Button
       android:id="@+id/stopMusic" 
       android:layout_width="wrap_content"
       android:layout_height="wrap_content"
       android:text="停止音乐播放服务"/>
   <Button
      android:id="@+id/bindMusic" 
      android:layout_width="wrap_content"
      android:layout_height="wrap_content"
      android:text="绑定音乐播放服务"/>
   <Button
      android:id="@+id/unbindMusic" 
      android:layout_width="wrap_content"
      android:layout_height="wrap_content"
      android:text="解除 ——绑定音乐播放服务"/>
</LinearLayout>

[3] src目录下MusicService.java源码:
package com.andyidea.service;
import android.app.Service;
import android.content.Intent;
import android.media.MediaPlayer;
import android.os.IBinder;
import android.util.Log;
import android.widget.Toast;
public class MusicService extends Service {
	//为日志工具设置标签
	private static String TAG = "MusicService";
	//定义音乐播放器变量
	private MediaPlayer mPlayer;
	
	//该服务不存在需要被创建时被调用,不管startService()还是bindService()都会启动时调用该方法
	@Override
	public void onCreate() {
		Toast.makeText(this, "MusicSevice onCreate()"
				, Toast.LENGTH_SHORT).show();
		Log.e(TAG, "MusicSerice onCreate()");
		
		mPlayer = MediaPlayer.create(getApplicationContext(), R.raw.music);
		//设置可以重复播放
		mPlayer.setLooping(true);
		super.onCreate();
	}
	
	@Override
	public void onStart(Intent intent, int startId) {
		Toast.makeText(this, "MusicSevice onStart()"
				, Toast.LENGTH_SHORT).show();
		Log.e(TAG, "MusicSerice onStart()");
		
		mPlayer.start();
		
		super.onStart(intent, startId);
	}
	@Override
	public void onDestroy() {
		Toast.makeText(this, "MusicSevice onDestroy()"
				, Toast.LENGTH_SHORT).show();
		Log.e(TAG, "MusicSerice onDestroy()");
		
		mPlayer.stop();
		
		super.onDestroy();
	}
	//其他对象通过bindService 方法通知该Service时该方法被调用
	@Override
	public IBinder onBind(Intent intent) {
		Toast.makeText(this, "MusicSevice onBind()"
				, Toast.LENGTH_SHORT).show();
		Log.e(TAG, "MusicSerice onBind()");
		
		mPlayer.start();
		
		return null;
	}
	//其它对象通过unbindService方法通知该Service时该方法被调用
	@Override
	public boolean onUnbind(Intent intent) {
		Toast.makeText(this, "MusicSevice onUnbind()"
				, Toast.LENGTH_SHORT).show();
		Log.e(TAG, "MusicSerice onUnbind()");
		
		mPlayer.stop();
		
		return super.onUnbind(intent);
	}
	
}

[4] src目录下MusicServiceActivity源码:

package com.andyidea.service;
import android.app.Activity;
import android.content.ComponentName;
import android.content.Context;
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;
import android.widget.Toast;
public class MusicServiceActivity extends Activity {
	
	//为日志工具设置标签
	private static String TAG = "MusicService";
	
    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        
        //输出Toast消息和日志记录
        Toast.makeText(this, "MusicServiceActivity",
        		Toast.LENGTH_SHORT).show();
        Log.e(TAG, "MusicServiceActivity");
        
        initlizeViews();
    }
    
    private void initlizeViews(){
    	Button btnStart = (Button)findViewById(R.id.startMusic);
    	Button btnStop = (Button)findViewById(R.id.stopMusic);
    	Button btnBind = (Button)findViewById(R.id.bindMusic);
    	Button btnUnbind = (Button)findViewById(R.id.unbindMusic);
    	
    	//定义点击监听器
    	OnClickListener ocl = new OnClickListener() {
			
			@Override
			public void onClick(View v) {
				//显示指定	intent所指的对象是个	service
				Intent intent = new Intent(MusicServiceActivity.this,MusicService.class);
				switch(v.getId()){
				case R.id.startMusic:
					//开始服务
					startService(intent);
					break;
				case R.id.stopMusic:
					//停止服务
					stopService(intent);
					break;
				case R.id.bindMusic:
					//绑定服务
					bindService(intent, conn, Context.BIND_AUTO_CREATE);
					break;
				case R.id.unbindMusic:
					//解绑服务
					unbindService(conn);
					break;
				}
			}
		};
		
		 //绑定点击监听
		btnStart.setOnClickListener(ocl);
		btnStop.setOnClickListener(ocl);
		btnBind.setOnClickListener(ocl);
		btnUnbind.setOnClickListener(ocl);
    }
    
    //定义服务链接对象
    final ServiceConnection conn = new ServiceConnection() {
		
		@Override
		public void onServiceDisconnected(ComponentName name) {
			Toast.makeText(MusicServiceActivity.this, "MusicServiceActivity onSeviceDisconnected"
					, Toast.LENGTH_SHORT).show();
			Log.e(TAG, "MusicServiceActivity onSeviceDisconnected");
		}
		
		@Override
		public void onServiceConnected(ComponentName name, IBinder service) {
			Toast.makeText(MusicServiceActivity.this, "MusicServiceActivity onServiceConnected"
					,Toast.LENGTH_SHORT).show();
			Log.e(TAG, "MusicServiceActivity onServiceConnected");
		}
	};
}

[5] 最后,我们别忘了在AndroidManifest.xml配置文件中添加对Service的注册。即在application节点中添加

      <service android:name=".MusicService"/> 进行注册。

[6] 我们来看下程序运行后的Log.e中显示的Service生命周期

[7] 我们在Android终端设备中查看下刚才启动的音乐播放服务,看看我们退出程序后,是不是该程序的服务还在运行的呢?按如下步骤:Menu --> Settings --> Applications --> Running services 。在弹出的 Running services 中可以看到有哪些服务在运行。

这样我们就看到我们退出程序后,是由于该服务还在后台运行着,所以我们的音乐还可以继续播放着。就这样,我们就可以一边享受音乐,一边可以聊QQ,或者浏览新闻等等。

文章出处:http://blog.csdn.net/cjjky/article/details/6552852

加载中
0
夜雪
写的很好啊 
0
lzsCommunity
lzsCommunity
Service并不是一个进程,也不是一个线程,它运行在应用进程的主线程中。。。。
lzsCommunity
lzsCommunity
回复 @forrey09 : 是的。。
forrey09
forrey09
对啊,service是运行在主线程中的。。。不过可以在service中可以开启一个工作现场去做其他的工作
0
simonws
simonws
运行在主线程中的话,为什么程序推出后服务还可以继续运行?,主线程还没退出?
lwm1368
lwm1368
兄弟,我也有这样的疑惑,请问你是这么理解的?
色魔张大妈
色魔张大妈
需要结束服务
0
团队合作
我认为是进程的可能性大点
0
阳光飞鸟
Service是运行在主线程之中的,程序之所以退出后服务还能继续运行,是因为系统还没把主线程给杀死了,还在继续运行着,只有等到系统需要内存的时候,才有可能杀死主线程。
0
s
stillhere

引用来自“阳光飞鸟”的答案

Service是运行在主线程之中的,程序之所以退出后服务还能继续运行,是因为系统还没把主线程给杀死了,还在继续运行着,只有等到系统需要内存的时候,才有可能杀死主线程。
并且 Service的优先级比较高,系统不到迫不得已是不会回收掉的、
0
惠儿
惠儿

mPlayer = MediaPlayer.create(getApplicationContext(), R.raw.music);

后面的R.raw.music出错music cannot be resolved or is not a field

这是为什么啊?哪位帮帮忙

惠儿
惠儿
回复 @鉴客 : 我在run()里面startServer,提示空指针异常,请问这个怎么解决呢?
惠儿
惠儿
回复 @鉴客 : 知道原因了 是R自动引入时引错了
鉴客
鉴客
这是基础 android 知识,建议先补一补 :)
0
徐岳
徐岳
谢谢楼主分享~
0
G
GinRyan
Service是一种另类的进程,它的地位和进程是一样的
返回顶部
顶部