深入解析Android 虚拟机
上QQ阅读APP看书,第一时间看更新

6.5 Service的生命周期

在本章前面的内容中,已经讲解了Android中Activity的生命周期。在本节的内容中,将详细讲解Android中Service的生命周期知识,为读者步入本书后面知识的学习打下基础。

6.5.1 Service的基本概念和用途

Android中的服务,它与Activity不同,它是不能与用户交互的,不能自己启动的,运行在后台的程序,如果退出应用时,Service进程并没有结束,它仍然在后台运行,那什么时候会用到Service呢?比如播放音乐的时候,有可能想边听音乐边干些其他事情,当退出播放音乐的应用,如果不用Service,就听不到歌了,所以这时候就得用到Service了。又比如当一个应用的数据是通过网络获取的,不同时间(一段时间)的数据是不同的这时候可以用Service在后台定时更新,而无需每打开应用的时候在去获取。

6.5.2 Service的生命周期详解

Android Service的生命周期并不像Activity那么复杂,它只继承了onCreate()、onStart()、onDestroy()3个方法,当第一次启动Service时,先后调用了onCreate()和onStart()这两个方法,当停止Service时,则执行onDestroy()方法,这里需要注意的是,如果Service已经启动了,当再次启动Service时,不会再执行onCreate()方法,而是直接执行onStart()方法,具体的可以看下面的实例。

6.5.3 Service与Activity通信

Service后端的数据最终还是要呈现在前端Activity之上的,因为启动Service时,系统会重新开启一个新的进程,这就涉及到不同进程间通信的问题了(AIDL)这一节不作过多描述,当想获取启动的Service实例时,可以用到bindService和onBindService方法,它们分别执行了Service中IBinder()和onUnbind()方法。

为了让大家更容易理解Service与Activity通信过程,接下来用一段演示代码来详细讲解。

(1)新建一个Android工程,在此命名为ServiceDemo。

(2)修改文件main.xml,在此增加了4个按钮,具体实现代码如下所示:

        <? 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>

(3)新建一个Service,命名为MyService.java,具体实现代码如下所示:

        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;
              }
            }
        }

(4)修改文件ServiceDemo.java,具体实现代码如下所示:

        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);
              }
            }
        }

(5)修改文件AndroidManifest.xml的代码,在此注册新建的MyService。

        <? 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>

执行上述代码后的效果如图6-5所示。

图6-5 执行效果

当单击“startServie”按钮时,先后执行了Service中onCreate()→onStart()这两个方法,如果打开DDMS的Logcat窗口,会看到如图6-6所示的界面。

图6-6 Logcat窗口

这时可以按HOME键进入Settings(设置)→Applications(应用)→Running Services(正在运行的服务)看一下新启动了一个服务,效果如图6-7所示。

图6-7 新启动了一个服务

当单击stopService按钮时,Service执行了onDestroy()方法,效果如图6-8所示。

图6-8 Logcat窗口

如果此时再次单击“startService”按钮,然后单击“bindService”按钮(通常bindService都是bind已经启动的Service),看一下Service执行了IBinder()方法,以及TextView的值也有所变化了,如图6-9和图6-10所示。

图6-9 Logcat窗口

图6-10 执行效果

最后单击“unbindService”按钮,则Service执行了onUnbind()方法,如图6-11所示。

图6-11 Logcat窗口