安卓四大組件之服務

来源:http://www.cnblogs.com/huangjie123/archive/2016/11/16/6069405.html
-Advertisement-
Play Games

startService()和bindService()是開啟服務的兩種方式 ...


       Service 是Android 中的(四大)組件之一。服務是沒有界面的組件,運行在後臺,服務是運行在當前應用程式進程里。如果有耗時的操作,不想有界面、而且還不想程式退出就停止運行的邏輯,放在服務里。要註意的是,服務也是運行在主線程中,如果有耗時操作,要放在子線程里,如果服務被系統殺死了,會預設重啟。另外,組件也可以通過綁定的形式跟一個Service進行交互,甚至完成進程間通信。比如:Service 可以在後臺處理網路傳輸、播放音樂、進行I/O 流的讀寫或者跟內容提供者進行交互。startService()和bindService()是開啟服務的兩種方式,startService()不能調用服務里的方法,不可以與服務進行通信,服務一旦開啟,會長時間在後臺運行,與開啟在不再有關係, 開啟在退出了,服務還是在運行的,而且不能調用服務里的方法;bindService()可以間接的調用服務里的方法,可以與服務進程通信,服務開啟了是和開啟在的生命周期綁定的,如果開啟在關閉了,服務也就關閉了, 開啟者可以間接的調用服務里的方法。如果服務同時被開啟和綁定,那麼服務就停不掉了,需要解除綁定服務才能停止服務。需要服務長期在後臺運行,還需要調用服務里的方法,用混合方式開啟服務,即採用兩種方式,但調用方式需要嚴格的順序。首先,用start方式開啟服務,目的是保證服務在後臺能夠長時間運行;其次,用bind方式綁定服務,綁定了服務,方便調用服務里的方法;然後解綁時,要先用unBind方式解綁,最後才stop方式停止服務。

     以上就是對服務的基本描述,接下來實現服務的具體操作。首先,需要明確一下服務的具體操作步驟。 

     startService()的方式開啟服務的大體流程如下:
      1. 寫一個類繼承 Service;
      2. 重寫onCreate()方法;
      3. 在清單文件的下麵聲明service,在name標簽中寫下服務的包名加類名;

     bindService()的方式綁定服務的具體寫法:
      1. 創建服務類, 繼承 Service;
      2. 定義一個介面,暴露對外提供的方法;          

public interface IService{
     public void callServiceMethed();
 }

      3. 在服務類里定義代理對象,定義一個方法可以間接的調用服務的方法, 這樣寫可以防止不想被暴露的方法被別人調用了,將希望被調用的方法寫在介面中,其他方法不被其他類調用;    

private class MyBinder extends Binder implements IService{ 
     public void callServiceMethed(){ 調用服務的方法 } ... 
}

 

      4. 在onBinder方法里返回代理對象,如果不返回,調用方拿到的對象就是空的       

public IBinder onBind(Intent intent) { return new MyBinder(); }

      5. 創建類實現 ServiceConnection,實現裡面的兩個方法         

private class MyConn implements ServiceConnection{ 
@Override 
public void onServiceConnected(ComponentName name, IBinder service) {
    //當服務連接成功時候調用 
} 
@Override
 public void onServiceDisconnected(ComponentName name) {
    //當服務斷開連接時調用
 } 
}

 

     6. activity採用綁定的方式開啟服務,bindService()方法綁定服務;
     7. 調用代理對象的方法,間接的調用了服務里的方法

 

     接下來瞭解一下服務的生命周期
        跟Activity 一樣,Service 也是有生命周期的,不一樣的是Service 的生命周期,從它被創建開始,到它被銷毀為止,可以有兩條不同的路徑:標準開啟模式和綁定模式。兩種生命周期的可以用下麵箭頭表示:

          標準開啟模式的生命周期:
          startService()->onCreate() -> onstartcommand() -> onDestroy()

          綁定模式的生命周期:
          bindService()->onBind() -> onunBind() -> onDestroy()

      被開啟的service 通過其他組件調用startService()被創建。這種service 可以無限地運行下去,必須調用stopSelf()方法或者其他組件調用stopService()方法來停止它。當service 被停止時,系統會銷毀它。用start方法開啟服務,服務只會被創建一次,執行一次onCreate方法,一旦服務創建完成,後續調用start去開啟服務只會執行onstart和onstartcommand方法,當調用了stop方法,服務只會調用一次onDestroy方法。

     被綁定的service 是當其他組件模擬一個客戶時,調用bindService()來創建的。客戶可以通過一個IBinder介面和service 進行通信。客戶可以通過unbindService()方法來關閉這種連接。一個service 可以同時和多個客戶綁定,當多個客戶都解除綁定之後,系統會銷毀service。你可以和一個已經調用了startService()而被開啟的service 進行綁定。比如,一個後臺音樂service 可能因調用startService()方法而被開啟了,稍後可能用戶想要控制播放器或者得到一些當前歌曲的信息,可以通過bindService()將一個activity 和service 綁定。這種情況下,stopService()或stopSelf()實際上並不能停止這個service,除非所有的客戶都解除綁定。

      瞭解服務的生命周期是為了更好的理解服務在整個工程下的運行原理。因此有必要瞭解一下安卓中進程的工作原理。

      在Android 中進程優先順序由高到低,依次分為:前臺進程(Foreground process),可視化進程(Visible process),服務進程(Service process),後臺進程(Background process),空進程(Empty process)。下麵一次給出每一種進程的概念和應用場景。
      前臺進程(Foreground process): 用戶正在操作的應用程式進程叫做前臺進程。通常情況下,在任何時候系統只存在一小部分前臺進程。這些進程只會作為最後的手段才會被殺死,即當記憶體不足以繼續運行他們的時候。在這個時刻,設備已經達到記憶體分頁狀態,當系統達到記憶體分頁狀態時只能通過虛擬地址訪問記憶體,可理解為達到這個狀態時系統已經無法繼續分配新的記憶體空間即可,因此殺死一些前臺進程,釋放記憶體空間以保證應用能夠繼續響應用戶的交互是必要的手段。

      可視化進程(Visible process): 用戶已經不能操作這個應用程式了,但是用戶還能看到應用程式界面。一個可視進程被認為是極其重要的並且一般不會被系統殺死,除非為了保證所有的前臺進程去運行不得已為之。

      服務進程(Service process): 應用程式服務在後臺運行。一個擁有正在運行的Service,並且該Service 是被startService()方法啟動起來的進程,並且該進程沒有被歸類到前面的兩種(前置進程和可視進程)類型,那麼該進程就是服務進程。儘管服務進程沒有與用戶可見的控制項直接綁定,但是這些進程乾的工作依然是用戶關心的(比如在後臺播放音樂或者從網路上下載數據),因此系統保留這些進程一直運行除非系統沒有足夠的記憶體去運行前臺進程和可視進程。
      後臺進程(Background process):應用程式界面被用戶最小化。一個擁有對用戶不可見的Activity,該Activity 已經被執行了onStop()方法進程叫做後臺進程。後臺進程對用戶體驗沒有直接的影響,並且系統會在任何需要為前臺進程,可視進程,或服務進程申請記憶體的時候殺死後臺進程。通常系統中運行著大量的後臺進程,這些後臺進程保存在一個LRU(最少最近使用的)列表中,使用LRU 規則是為了保證讓最近被用戶使用的Activity 進程最後被殺死,就是誰最近被使用了,誰最後再被殺死。如果一個Activity 正確實現了它的生命周期方法,並且保存了它的狀態,通常這個狀態是系統自動保存的,那麼當系統殺死它的進程的時候是對用戶的體驗沒有看得見的影響的,因為當用戶導航到之前的Activity 的時候,這個Activity 會自動恢復之前保存的視圖狀態。查看Activity 文檔去獲取更多關於Activity狀態的保存和恢覆信息。

     空進程(Empty process): 應用程式沒有任何的activity和service運行。不擁有任何系統四大組件的進程叫空進程。保持空進程存活的唯一理由是為了緩存,這樣可以提高下次啟動組件的打開速度。當系統需要維持緩存進程和底層內核緩存的資源均衡的時候系統經常會或者隨時會殺死該類進程。
     Android 系統有一套記憶體回收機制,會根據優先順序進行回收。Android 系統會儘可能的維持程式的進程,但是終究還是需要回收一些舊的進程節省記憶體提供給新的或者重要的進程使用。進程的回收順序是:從低到高,即當系統記憶體不夠用時, 會把空進程一個一個回收掉,當系統回收所有的完空進程不夠用時, 繼續向上回收後臺進程, 依次類推。但是當回收服務, 可視, 前臺這三種進程時, 系統非必要情況下不會輕易回收, 如果需要回收掉這三種進程, 那麼在系統記憶體夠用時, 會再給重新啟動進程;但是服務進程如果用戶手動的關閉服務, 這時服務不會再重啟了。

     文章最後給出兩個案例來加深對服務的理解,一個是用startService()開啟服務,一個是利用綁定服務的原理實現遠程服務。

      用startService()開啟服務。流程就不再介紹,直接給出相關的代碼。

      開啟服務和停止服務的java代碼:

import android.app.Activity;
import android.content.Intent;
import android.os.Bundle;
import android.view.View;

public class MainActivity extends Activity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
    }
    /**
*開啟服務
*/
public void start(View view){ Intent intent = new Intent(this,DemoService.class); startService(intent); } /**
    *停止服務
    */
public void stop(View viwe){ Intent intent = new Intent(this,DemoService.class); stopService(intent); } }

 

         startService()方式開啟服務時,服務的java代碼:

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

public class DemoService extends Service {
    private boolean flag;

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

    /**
     * 服務一般用於檢測設備
     * 
     */
    @Override
    public void onCreate() {
        super.onCreate();
        new Thread() {
            public void run() {
                flag = true;
                while (flag) {
                    System.out.println("檢查是否有設備插入進來了.");
                    SystemClock.sleep(2000);
                    System.out.println("服務被創建了,運行在:"
                            + Thread.currentThread().getName() + "線程中");
                }
            };
        }.start();
    }

    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        System.out.println("服務被開啟");
        return super.onStartCommand(intent, flags, startId);
    }

    @Override
    public void onDestroy() {
        super.onDestroy();
        flag = false;
        System.out.println("服務被銷毀");
    }
}

        相應得佈局文件較為簡單,僅僅需要兩個按鍵控制服務的開和關,在此也給出佈局文件的相關代碼。

<LinearLayout 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:orientation="vertical"
    tools:context=".MainActivity" >

    <Button
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:onClick="start"
        android:text="開始服務" />

    <Button
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:onClick="stop"
        android:text="結束服務" />

</LinearLayout>

     

      用bindService()綁定服務的案例,利用服務實現遠程通信。在Android 平臺中,各個組件運行在自己的進程中,他們之間是不能相互訪問的,但是在程式之間是不可避免的要傳遞一些對象,在進程之間相互通信。為了實現進程之間的相互通信,Android 採用了一種輕量級的實現方式RPC(Remote Procedure Call 遠程進程調用)來完成進程之間的通信,並且Android 通過介面定義語言(Android Interface Definition Language ,AIDL)來生成兩個進程之間相互訪問的代碼。如果想讓我們的Service 可以提供遠程服務,那麼就必須定義一個.aidl 文件,該文件使用的是java 語法,類似java 的介面。然後將該文件在客戶端和服務端的src 目錄下各自保存一份,這樣編譯器就會根據aidl 文件自動生成一個java 類,也就說在客戶端和服務端都擁有了相同的類文件了。

       遠程服務需要服務提供者以及服務調用者。因此需要兩個安卓項目。

      遠程服務提供者的java代碼包括三個部分:

      綁定服務的JAVA代碼:

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

public class remote extends Service {
    
    public class myBind extends IService.Stub {
        @Override
        public void callMethodInService() throws RemoteException {
            methodInService();
        }
    }
    
    @Override
    public IBinder onBind(Intent intent) {
        return new myBind();
    }

    @Override
    public void onCreate() {
        System.out.println("服務被創建");
        super.onCreate();
    }
    
    @Override
    public void onDestroy() {
        System.out.println("服務被銷毀");
        super.onDestroy();
    }
    
    private void methodInService(){
        System.out.println("服務中的方法被調用");
    }
}

        綁定服務的主界面,服務是運行在後臺的不可見,因此為了顯示兩個程式之間傳遞數據,在此寫了一個主界面。界面的佈局可以隨意寫。


 

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

 

public class MainActivity extends Activity {

 

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

 

     aidl文件和捷口類似:

 package com.example.remote;
 
  interface IService {
       void callMethodInService();
   }

 

        接下來是服務調用者中的代碼。

import com.example.remote.IService;

import android.app.Activity;
import android.content.ComponentName;
import android.content.Intent;
import android.content.ServiceConnection;
import android.os.Bundle;
import android.os.IBinder;
import android.os.RemoteCallbackList;
import android.os.RemoteException;
import android.view.View;

public class MainActivity extends Activity {
    private IService iservice;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
    }
    /**
     * 綁定服務
     * @param view
     */
    public void bind(View view) {
        Intent intent = new Intent();
        intent.setAction("com.example.remote");
        bindService(intent, new myConn(), BIND_AUTO_CREATE);
    }
    /**
     * 調用遠程服務的方法 
     * @param view
     */
    public void call(View view) {
        try {
            iservice.callMethodInService();
        } catch (RemoteException e) {
            e.printStackTrace();
        }
    }

    private class myConn implements ServiceConnection {
        //鏈接成功
        @Override
        public void onServiceConnected(ComponentName name, IBinder service) {
            iservice = IService.Stub.asInterface(service);
        }
        //鏈接失敗
        @Override
        public void onServiceDisconnected(ComponentName name) {
            System.out.println("拒絕鏈接");
        }
    }
}

        單獨定義一個文件夾,文件夾的名字和遠程服務的包名一直,裡面存放aidl文件:

 package com.example.remote;
 
  interface IService {
     void callMethodInService();
   }

      此外,在服務調用者的佈局問價中寫兩個按鈕,用來綁定扶服務和調用服務中的方法。

   

<LinearLayout 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:orientation="vertical"
    android:paddingBottom="@dimen/activity_vertical_margin"
    android:paddingLeft="@dimen/activity_horizontal_margin"
    android:paddingRight="@dimen/activity_horizontal_margin"
    android:paddingTop="@dimen/activity_vertical_margin"
    tools:context=".MainActivity" >

    <Button
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:onClick="bind"
        android:text="綁定服務" />

    <Button
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:onClick="call"
        android:text="調用服務的方法" />

</LinearLayout>

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 1. 可以說幾乎每個做過Web開發的人都問過,到底元素的ID和Name有什麼區別阿?為什麼有了ID還要有Name呢?! 而同樣我們也可以得到最classical的答案:ID就像是一個人的身份證號碼,而Name就像是他的名字,ID顯然是唯一的,而Name是可以重覆的。 上周我也遇到了ID和Name的問 ...
  • 這是一個令人激動的革新。 CSS 變數,顧名思義,也就是由網頁的作者或用戶定義的實體,用來指定文檔中的特定變數。 更準確的說法,應該稱之為 CSS 自定義屬性 ,不過下文為了好理解都稱之為 CSS 變數。 一直以來我們都知道,CSS 中是沒有變數而言的,要使用 CSS 變數,只能藉助 SASS 或者 ...
  • 方法一: 方法二:正則法 ...
  • webix的表格,頭部沒有tooltip的屬性支持 onmousemove只監聽了數據部分,對列頭沒有監聽。官網上演示的是在header屬性上寫個span 加個title的屬性,但是樣式不好看。然後我就直接參照寫了個。用的是監聽載入完成後的事件。用於載入後確定列的情況。 1、效果如下 代碼如下,拷貝 ...
  • 前言 作為一名前端開發人員,如果你告訴我你沒有看過任何關於前端的書籍,那麼我完全可以認為你不是一名合格的前端開發工程師。為什麼我要以“看書”來衡量合格前端的標準?因為前端作為一個特殊的極具變化與開拓性的工種,沒有較強的自我學習與思考能力,很難在這激烈又紛雜的環境里存活而不被淘汰,而“看書”則是最基本 ...
  • 基本演示 背景演示 迴圈演示 回調函數演示 綁定菜單演示 項目導航演示 自動滾動 slide自動滾動 響應式 下載地址 實例代碼 <!DOCTYPE html> <html lang="zh-CN"> <head> <meta charset="utf-8"> <title>fullPage.js ...
  • 現在很多類似以微博發佈動態的效果,下麵為一個用 JavaScript寫的小小的類似微博發佈信息的案例 佈局出來的樣式,點擊藍色的刪除鏈接,會刪除對應的那一行內容 陌陌說:重要知識點:獲取輸入框的數值,創建子節點和給子節點添加內容,刪除對應的節點 ...
  • 模塊系統 Node根據CommonJS規範實現了一套自己的模塊機制,可以使用require()導入一個模塊,使用module.exports導出一個模塊。 require使用 在Node中我們可以使用require()導入一個模塊,此時我們就會獲得一個被導入模塊的對象,我們就可以利用這個對象來完成一 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...