大家好,上一节我讲解了Android Activity的生命周期,这一节我将讲解一下Service,首先我们要知道Service具体是干什么的,什么时候用到?以及它的生命周期等。
Service概念及用途:
Android中的服务,它与Activity不同,它是不能与用户交互的,不能自己启动的,运行在后台的程序,如果我们退出应用时,Service进程并没有结束,它仍然在后台运行,那我们什么时候会用到Service呢?比如我们播放音乐的时候,有可能想边听音乐边干些其他事情,当我们退出播放音乐的应用,如果不用Service,我们就听不到歌了,所以这时候就得用到Service了,又比如当我们一个应用的数据是通过网络获取的,不同时间(一段时间)的数据是不同的这时候我们可以用Service在后台定时更新,而不用每打开应用的时候在去获取。
Service生命周期 :
Android Service的生命周期并不像Activity那么复杂,它只继承了onCreate(),onStart(),onDestroy()三个方法,当我们第一次启动Service时,先后调用了onCreate(),onStart()这两个方法,当停止Service时,则执行onDestroy()方法,这里需要注意的是,如果Service已经启动了,当我们再次启动Service时,不会在执行onCreate()方法,而是直接执行onStart()方法,具体的可以看下面的实例。
Service与Activity通信:
Service后端的数据最终还是要呈现在前端Activity之上的,因为启动Service时,系统会重新开启一个新的进程,这就涉及到不同进程间通信的问题了(AIDL)这一节我不作过多描述,当我们想获取启动的Service实例时,我们可以用到bindService和onBindService方法,它们分别执行了Service中IBinder()和onUnbind()方法。
为了让大家 更容易理解,我写了一个简单的Demo,大家可以模仿着我,一步一步的来。
第一步:新建一个Android工程,我这里命名为ServiceDemo.
第二步:修改main.xml代码,我这里增加了四个按钮,代码如下:
view plaincopy to clipboardprint?
01.<?xml version="1.0" encoding="utf-8"?>
02.<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
03. android:orientation="vertical"
04. android:layout_width="fill_parent"
05. android:layout_height="fill_parent"
06. >
07. <TextView
08. android:id="@+id/text"
09. android:layout_width="fill_parent"
10. android:layout_height="wrap_content"
11. android:text="@string/hello"
12. />
13. <Button
14. android:id="@+id/startservice"
15. android:layout_width="fill_parent"
16. android:layout_height="wrap_content"
17. android:text="startService"
18. />
19. <Button
20. android:id="@+id/stopservice"
21. android:layout_width="fill_parent"
22. android:layout_height="wrap_content"
23. android:text="stopService"
24. />
25. <Button
26. android:id="@+id/bindservice"
27. android:layout_width="fill_parent"
28. android:layout_height="wrap_content"
29. android:text="bindService"
30. />
31. <Button
32. android:id="@+id/unbindservice"
33. android:layout_width="fill_parent"
34. android:layout_height="wrap_content"
35. android:text="unbindService"
36. />
37.</LinearLayout>
<?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:id="@+id/text"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="@string/hello"
/>
<Button
android:id="@+id/startservice"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="startService"
/>
<Button
android:id="@+id/stopservice"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="stopService"
/>
<Button
android:id="@+id/bindservice"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="bindService"
/>
<Button
android:id="@+id/unbindservice"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="unbindService"
/>
</LinearLayout>
第三步:新建一个Service,命名为MyService.java代码如下:
view plaincopy to clipboardprint?
01.package com.tutor.servicedemo;
02.import android.app.Service;
03.import android.content.Intent;
04.import android.os.Binder;
05.import android.os.IBinder;
06.import android.text.format.Time;
07.import android.util.Log;
08.public class MyService extends Service {
09. //定义个一个Tag标签
10. private static final String TAG = "MyService";
11. //这里定义吧一个Binder类,用在onBind()有方法里,这样Activity那边可以获取到
12. private MyBinder mBinder = new MyBinder();
13. @Override
14. public IBinder onBind(Intent intent) {
15. Log.e(TAG, "start IBinder~~~");
16. return mBinder;
17. }
18. @Override
19. public void onCreate() {
20. Log.e(TAG, "start onCreate~~~");
21. super.onCreate();
22. }
23.
24. @Override
25. public void onStart(Intent intent, int startId) {
26. Log.e(TAG, "start onStart~~~");
27. super.onStart(intent, startId);
28. }
29.
30. @Override
31. public void onDestroy() {
32. Log.e(TAG, "start onDestroy~~~");
33. super.onDestroy();
34. }
35.
36.
37. @Override
38. public boolean onUnbind(Intent intent) {
39. Log.e(TAG, "start onUnbind~~~");
40. return super.onUnbind(intent);
41. }
42.
43. //这里我写了一个获取当前时间的函数,不过没有格式化就先这么着吧
44. public String getSystemTime(){
45.
46. Time t = new Time();
47. t.setToNow();
48. return t.toString();
49. }
50.
51. public class MyBinder extends Binder{
52. MyService getService()
53. {
54. return MyService.this;
55. }
56. }
57.}
package com.tutor.servicedemo;
import android.app.Service;
import android.content.Intent;
import android.os.Binder;
import android.os.IBinder;
import android.text.format.Time;
import android.util.Log;
public class MyService extends Service {
//定义个一个Tag标签
private static final String TAG = "MyService";
//这里定义吧一个Binder类,用在onBind()有方法里,这样Activity那边可以获取到
private MyBinder mBinder = new MyBinder();
@Override
public IBinder onBind(Intent intent) {
Log.e(TAG, "start IBinder~~~");
return mBinder;
}
@Override
public void onCreate() {
Log.e(TAG, "start onCreate~~~");
super.onCreate();
}
@Override
public void onStart(Intent intent, int startId) {
Log.e(TAG, "start onStart~~~");
super.onStart(intent, startId);
}
@Override
public void onDestroy() {
Log.e(TAG, "start onDestroy~~~");
super.onDestroy();
}
@Override
public boolean onUnbind(Intent intent) {
Log.e(TAG, "start onUnbind~~~");
return super.onUnbind(intent);
}
//这里我写了一个获取当前时间的函数,不过没有格式化就先这么着吧
public String getSystemTime(){
Time t = new Time();
t.setToNow();
return t.toString();
}
public class MyBinder extends Binder{
MyService getService()
{
return MyService.this;
}
}
}
第四步:修改ServiceDemo.java,代码如下:
view plaincopy to clipboardprint?
01.package com.tutor.servicedemo;
02.import android.app.Activity;
03.import android.content.ComponentName;
04.import android.content.Context;
05.import android.content.Intent;
06.import android.content.ServiceConnection;
07.import android.os.Bundle;
08.import android.os.IBinder;
09.import android.view.View;
10.import android.view.View.OnClickListener;
11.import android.widget.Button;
12.import android.widget.TextView;
13.public class ServiceDemo extends Activity implements OnClickListener{
14.
15. private MyService mMyService;
16. private TextView mTextView;
17. private Button startServiceButton;
18. private Button stopServiceButton;
19. private Button bindServiceButton;
20. private Button unbindServiceButton;
21. private Context mContext;
22.
23. //这里需要用到ServiceConnection在Context.bindService和context.unBindService()里用到
24. private ServiceConnection mServiceConnection = new ServiceConnection() {
25. //当我bindService时,让TextView显示MyService里getSystemTime()方法的返回值
26. public void onServiceConnected(ComponentName name, IBinder service) {
27. // TODO Auto-generated method stub
28. mMyService = ((MyService.MyBinder)service).getService();
29. mTextView.setText("I am frome Service :" + mMyService.getSystemTime());
30. }
31.
32. public void onServiceDisconnected(ComponentName name) {
33. // TODO Auto-generated method stub
34.
35. }
36. };
37. public void onCreate(Bundle savedInstanceState) {
38. super.onCreate(savedInstanceState);
39. setContentView(R.layout.main);
40. setupViews();
41. }
42.
43. public void setupViews(){
44.
45. mContext = ServiceDemo.this;
46. mTextView = (TextView)findViewById(R.id.text);
47.
48.
49.
50. startServiceButton = (Button)findViewById(R.id.startservice);
51. stopServiceButton = (Button)findViewById(R.id.stopservice);
52. bindServiceButton = (Button)findViewById(R.id.bindservice);
53. unbindServiceButton = (Button)findViewById(R.id.unbindservice);
54.
55. startServiceButton.setOnClickListener(this);
56. stopServiceButton.setOnClickListener(this);
57. bindServiceButton.setOnClickListener(this);
58. unbindServiceButton.setOnClickListener(this);
59. }
60.
61. public void onClick(View v) {
62. // TODO Auto-generated method stub
63. if(v == startServiceButton){
64. Intent i = new Intent();
65. i.setClass(ServiceDemo.this, MyService.class);
66. mContext.startService(i);
67. }else if(v == stopServiceButton){
68. Intent i = new Intent();
69. i.setClass(ServiceDemo.this, MyService.class);
70. mContext.stopService(i);
71. }else if(v == bindServiceButton){
72. Intent i = new Intent();
73. i.setClass(ServiceDemo.this, MyService.class);
74. mContext.bindService(i, mServiceConnection, BIND_AUTO_CREATE);
75. }else{
76. mContext.unbindService(mServiceConnection);
77. }
78. }
79.
80.
81.
82.}
package com.tutor.servicedemo;
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.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.TextView;
public class ServiceDemo extends Activity implements OnClickListener{
private MyService mMyService;
private TextView mTextView;
private Button startServiceButton;
private Button stopServiceButton;
private Button bindServiceButton;
private Button unbindServiceButton;
private Context mContext;
//这里需要用到ServiceConnection在Context.bindService和context.unBindService()里用到
private ServiceConnection mServiceConnection = new ServiceConnection() {
//当我bindService时,让TextView显示MyService里getSystemTime()方法的返回值
public void onServiceConnected(ComponentName name, IBinder service) {
// TODO Auto-generated method stub
mMyService = ((MyService.MyBinder)service).getService();
mTextView.setText("I am frome Service :" + mMyService.getSystemTime());
}
public void onServiceDisconnected(ComponentName name) {
// TODO Auto-generated method stub
}
};
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
setupViews();
}
public void setupViews(){
mContext = ServiceDemo.this;
mTextView = (TextView)findViewById(R.id.text);
startServiceButton = (Button)findViewById(R.id.startservice);
stopServiceButton = (Button)findViewById(R.id.stopservice);
bindServiceButton = (Button)findViewById(R.id.bindservice);
unbindServiceButton = (Button)findViewById(R.id.unbindservice);
startServiceButton.setOnClickListener(this);
stopServiceButton.setOnClickListener(this);
bindServiceButton.setOnClickListener(this);
unbindServiceButton.setOnClickListener(this);
}
public void onClick(View v) {
// TODO Auto-generated method stub
if(v == startServiceButton){
Intent i = new Intent();
i.setClass(ServiceDemo.this, MyService.class);
mContext.startService(i);
}else if(v == stopServiceButton){
Intent i = new Intent();
i.setClass(ServiceDemo.this, MyService.class);
mContext.stopService(i);
}else if(v == bindServiceButton){
Intent i = new Intent();
i.setClass(ServiceDemo.this, MyService.class);
mContext.bindService(i, mServiceConnection, BIND_AUTO_CREATE);
}else{
mContext.unbindService(mServiceConnection);
}
}
}
第五步:修改AndroidManifest.xml代码(将我们新建的MyService注册进去如下代码第14行:)
view plaincopy to clipboardprint?
01.<?xml version="1.0" encoding="utf-8"?>
02.<manifest xmlns:android="http://schemas.android.com/apk/res/android"
03. package="com.tutor.servicedemo"
04. android:versionCode="1"
05. android:versionName="1.0">
06. <application android:icon="@drawable/icon" android:label="@string/app_name">
07. <activity android:name=".ServiceDemo"
08. android:label="@string/app_name">
09. <intent-filter>
10. <action android:name="android.intent.action.MAIN" />
11. <category android:name="android.intent.category.LAUNCHER" />
12. </intent-filter>
13. </activity>
14. <service android:name=".MyService" android:exported="true"></service>
15. </application>
16. <uses-sdk android:minSdkVersion="7" />
17.</manifest>
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.tutor.servicedemo"
android:versionCode="1"
android:versionName="1.0">
<application android:icon="@drawable/icon" android:label="@string/app_name">
<activity android:name=".ServiceDemo"
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=".MyService" android:exported="true"></service>
</application>
<uses-sdk android:minSdkVersion="7" />
</manifest>
第六步:执行上述工程,效果图如下:
点击startServie按钮时先后执行了Service中onCreate()->onStart()这两个方法,打开Logcat视窗效果如下图:
我们这时可以按HOME键进入Settings(设置)->Applications(应用)->Running Services(正在运行的服务)看一下我们新启动了一个服务,效果如下:
点击stopService按钮时,Service则执行了onDestroy()方法,效果图如下所示:
这时候我们再次点击startService按钮,然后点击bindService按钮(通常bindService都是bind已经启动的Service),我们看一下Service执行了IBinder()方法,以及TextView的值也有所变化了,如下两张图所示:
最后点击unbindService按钮,则Service执行了onUnbind()方法,如下图所示:
Ok,今天就先讲到这里了,谢谢大家关注