Latest web development tutorials

Android 服務(Service)

服務是一個後台運行的組件,執行長時間運行且不需要用戶交互的任務。 即使應用被銷毀也依然可以工作。 服務基本上包含兩種狀態-

狀態 描述
Started Android的應用程序組件,如活動,通過startService()啟動了服務,則服務是Started狀態。 一旦啟動,服務可以在後台無限期運行,及時啟動它的組件已經被銷毀。
Bound 當Android的應用程序組件通過bindService()綁定了服務,則服務是Bound狀態。 Bound狀態的服務提供了一個客戶服務器接口來允許組件與服務進行交互,如發送請求,獲取結果,甚至通過IPC來進行跨進程通信。

服務擁有生命週期方法,可以實現監控服務狀態的變化,可以在合適的階段執行工作。 下面的左圖展示了當服務通過startService()被創建時的聲明周期,右圖則顯示了當服務通過bindService()被創建時的生命週期:

圖片圖片

要創建服務,你需要創建一個繼承自Service基類或者它的已知子類的Java類。 Service基類定義了不同的回調方法和多數重要方法。 你不需要實現所有的回調方法。 雖然如此,理解所有的方法還是非常重要的。 實現這些回調能確保你的應用以用戶期望的方式實現。

回調 描述
onStartCommand() 其他組件(如活動)通過調用startService()來請求啟動服務時,系統調用該方法。 如果你實現該方法,你有責任在工作完成時通過stopSelf()或者stopService()方法來停止服務。
onBind 當其他組件想要通過bindService()來綁定服務時,系統調用該方法。 如果你實現該方法,你需要返回IBinder對象來提供一個接口,以便客戶來與服務通信。 你必須實現該方法,如果你不允許綁定,則直接返回null。
onUnbind() 當客戶中斷所有服務發布的特殊接口時,系統調用該方法。
onRebind() 當新的客戶端與服務連接,且此前它已經通過onUnbind(Intent)通知斷開連接時,系統調用該方法。
onCreate() 當服務通過onStartCommand()和onBind()被第一次創建的時候,系統調用該方法。 該調用要求執行一次性安裝。
onDestroy() 當服務不再有用或者被銷毀時,系統調用該方法。 你的服務需要實現該方法來清理任何資源,如線程,已註冊的監聽器,接收器等。

下面的主服務演示了每個方法的生命週期-

package cn.uprogrammer.androidservices;

import android.app.Service;
import android.os.IBinder;
import android.content.Intent;
import android.os.Bundle;

public class HelloService extends Service {

    /** 标识服务如果被杀死之后的行为 */
    int mStartMode;

    /** 绑定的客户端接口 */
    IBinder mBinder;

    /** 标识是否可以使用onRebind */
    boolean mAllowRebind;

    /** 当服务被创建时调用. */
    @Override
    public void onCreate() {

    }

    /** 调用startService()启动服务时回调 */
    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        return mStartMode;
    }

    /** 通过bindService()绑定到服务的客户端 */
    @Override
    public IBinder onBind(Intent intent) {
        return mBinder;
    }

    /** 通过unbindService()解除所有客户端绑定时调用 */
    @Override
    public boolean onUnbind(Intent intent) {
        return mAllowRebind;
    }

    /** 通过bindService()将客户端绑定到服务时调用*/
    @Override
    public void onRebind(Intent intent) {

    }

    /** 服务不再有用且将要被销毁时调用 */
    @Override
    public void onDestroy() {

    }
}

實例

這個例子將通過簡單地步驟為你展示如何創建自己的Android服務。 按照下面的步驟來修改之前在Hello World實例章節中創建的Android應用程序:

步驟 描述
1 使用Android Studio IDE來創建Android應用程序並在cn.uprogrammer.androidservices包下命名為androidservices。 類似Hello World實例章節。
2 修改主活動文件MainActivity.java來添加startService()和stopService()方法。
3 在包cn.uprogrammer.androidservices下創建新的Java文件MyService.java。 這個文件將實現Android服務相關的方法。
4 在AndroidManifest.xml文件中使用<service.../>標籤來定義服務。 應用程序可以有一個或多個服務,沒有任何限制。
5 修改res/layout/activity_main.xml文件中的默認佈局,在線性佈局中包含兩個按鈕。
6 不要對res/values/strings.xml文件中的任何常量進行修改。 Android Studio會注意字符串值。
7 啟動Android模擬器來運行應用程序,並驗證應用程序所做改變的結果。

下面是主活動文件src/cn.uprogrammer.androidservices/MainActivity.java文件所修改的內容。 這個文件包含所有基本的生命週期方法。 我們添加了startService()和stopService()方法來啟動和停止服務。

package cn.uprogrammer.androidservices;

import android.app.Activity;
import android.os.Bundle;
import android.view.Menu;


import android.content.Intent;
import android.view.View;

public class MainActivity extends Activity {

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
    }

    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
        getMenuInflater().inflate(R.menu.menu_main, menu);
        return true;
    }

    // Method to start the service
    public void startService(View view) {
        startService(new Intent(getBaseContext(), MyService.class));
    }

    // Method to stop the service
    public void stopService(View view) {
        stopService(new Intent(getBaseContext(), MyService.class));
    }
}

以下是src/cn.uprogrammer.androidservices/MyService.java的內容。 這個文件可以基於需求實現一個或多個服務關聯的方法。 對於新人,我們只實現onStartCommand()和onDestroy() -

package cn.uprogrammer.androidservices;

import android.app.Service;
import android.content.Intent;
import android.os.IBinder;
import android.widget.Toast;

public class MyService extends Service {

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

    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        // Let it continue running until it is stopped.
        Toast.makeText(this, "服务已经启动", Toast.LENGTH_LONG).show();
        return START_STICKY;
    }

    @Override
    public void onDestroy() {
        super.onDestroy();
        Toast.makeText(this, "服务已经停止", Toast.LENGTH_LONG).show();
    }
}

下面將修改AndroidManifest.xml文件。 這裡添加<service.../>標籤來包含我們的服務:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="cn.uprogrammer.androidservices"
    android:versionCode="1"
    android:versionName="1.0">

    <uses-sdk
        android:minSdkVersion="13"
        android:targetSdkVersion="22" />

    <application
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name"
        android:theme="@style/AppTheme" >

        <activity
            android:name=".MainActivity"
            android:label="@string/title_activity_main" >

            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.LAUNCHER"/>
            </intent-filter>

        </activity>

        <service android:name=".MyService" />

    </application>

</manifest>

以下是res/layout/activity_main.xml文件的內容,包含兩個按鈕:

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent"
    android:layout_height="match_parent" android:paddingLeft="@dimen/activity_horizontal_margin"
    android:paddingRight="@dimen/activity_horizontal_margin"
    android:paddingTop="@dimen/activity_vertical_margin"
    android:paddingBottom="@dimen/activity_vertical_margin" tools:context=".MainActivity">

    <TextView
        android:id="@+id/textView1"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="Android 服务实例"
        android:layout_alignParentTop="true"
        android:layout_centerHorizontal="true"
        android:textSize="30dp" />

    <TextView
        android:id="@+id/textView2"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="www.uprogrammer.cn"
        android:textColor="#ff87ff09"
        android:textSize="30dp"
        android:layout_above="@+id/imageButton"
        android:layout_centerHorizontal="true"
        android:layout_marginBottom="40dp" />

    <ImageButton
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:id="@+id/imageButton"
        android:src="@drawable/ic_launcher"
        android:layout_centerVertical="true"
        android:layout_centerHorizontal="true" />

    <Button
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:id="@+id/button2"
        android:text="启动服务"
        android:onClick="startService"
        android:layout_below="@+id/imageButton"
        android:layout_centerHorizontal="true" />

    <Button
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="停止服务"
        android:id="@+id/button"
        android:onClick="stopService"
        android:layout_below="@+id/button2"
        android:layout_alignLeft="@+id/button2"
        android:layout_alignStart="@+id/button2"
        android:layout_alignRight="@+id/button2"
        android:layout_alignEnd="@+id/button2" />

</RelativeLayout>

下面是res/values/strings.xml的內容,來定義兩個新的常量:

<?xml version="1.0" encoding="utf-8"?>
<resources>

    <string name="app_name">Android Services</string>
    <string name="title_activity_main">MainActivity</string>
    <string name="menu_settings">Settings</string>
    <string name="action_settings">Settings</string>

</resources>

讓我們運行剛剛修改的My Application應用程序。 我假設你已經在安裝環境時創建了AVD。 打開你的項目中的活動文件,點擊工具欄中的 圖片 圖標來在Android Studio中運行應用程序。 Android Studio在AVD上安裝應用程序並啟動它。 如果一切順利,將在模擬器窗口上顯示如下:

圖片

現在點擊"啟動服務"按鈕來啟動服務,這將執行我們編寫的onStartCommand()方法,一條"服務已經啟動"的消息在模擬器的底部出現,如下:

圖片

點擊底部的"停止服務"按鈕,可以停止服務。