Android_實用技術(3)—— Service簡析(Ⅲ)

来源:http://www.cnblogs.com/pepsimaxin/archive/2016/04/26/5435958.html
-Advertisement-
Play Games

1、今天我們來分析Service中的一個小技巧:前臺服務(Forground Service) 【問題】:我們都知道服務是運行在後臺的,如果系統出現記憶體不足的情況,那麼此時,系統就可能回收後代的服務,那麼我們如何保證服務可以一直運行? 【解決】:在服務中,有一個前臺服務的概念,調用startForg ...


  1、今天我們來分析Service中的一個小技巧:前臺服務(Forground Service)

----------------------------------------------------------------------------------------------------------------------------------------

【問題】我們都知道服務是運行在後臺的,如果系統出現記憶體不足的情況,那麼此時,系統就可能回收後代的服務,那麼我們如何保證服務可以一直運行?

【解決】:在服務中,有一個前臺服務的概念,調用startForground()方法。

   我們看看官網對前臺服務startForeground()的描述:

  

    

  看了官方的解釋後,我們再來看看如何使用,上代碼:

複製代碼
public class MyService extends Service{

  ......

  @Override
  public void onCreate() {
    super.onCreate();
    Intent intent = new Intent(getBaseContext(), MainActivity.class);
    PendingIntent pi = PendingIntent.getActivity(getBaseContext(), 0 , intent, PendingIntent.FLAG_CANCEL_CURRENT);
    Notification no = new Notification.Builder(getBaseContext()) // 啟動服務後,在前臺添加一個Notification
        .setWhen(System.currentTimeMillis())
        .setSmallIcon(R.mipmap.ic_launcher)
        .setTicker("Create a front desk Service!")
        .setContentTitle("This is a front desk service")
        .setContentText("A service skill!!!")
        .setContentIntent(pi)
        .setAutoCancel(true)
        .setDefaults(Notification.DEFAULT_ALL)
        .build();
    startForeground(1, no);
    Log.d(TAG, "onCreate executed");
  }
}
複製代碼

  我們再看一下官方文檔:

  

  以上的代碼是在Service的創建中添加了一個Notification,調用startForground()就可以保證:只要服務一直存在,那麼在前臺就會一直顯示這個Notification

  如果我們在onDestroy()中調用stopForground()方法,會銷毀這個Notification但是Service還是存活的,此時Service就會面臨被System幹掉的風險

  如果直接STOP SERVICE,那麼Notification和Service都會銷毀。 

----------------------------------------------------------------------------------------------------------------------------------------

  2、接下來,我們再來看一個Service的另外一個小技巧:IntentService

【問題】我們知道服務的代碼邏輯是在主線程中執行的,如果我們在主線程中需要執行一些耗時的操作,那麼很有可能出現ANR(程式暫無響應)的狀況。

  這個時候,我們可以採用Android的多線程編程(小編在之前的 AsyncTask 貼中講解過多線程,可以回顧)的方式,我們來看一段代碼:

public class MyService extends Service{
  @Nullable
  @Override
  public IBinder onBind(Intent intent) {
    return null;
  }
  @Override
  public int onStartCommand(Intent intent, int flags, int startId) {
    new Thread(new Runnable() {
      @Override
      public void run() {
        // 處理具體的邏輯 // 開啟一個線程處理耗時操作
      }
    }).start();     
return super.onStartCommand(intent, flags, startId);   }
}

  現在,服務可以啟動起來了,但是如果不調用StopService()stopSelf()方法,服務會一直運行,現在我們修改一下代碼:

public class MyService extends Service{
  @Nullable
  @Override
  public IBinder onBind(Intent intent) {
    return null;
  }
  @Override
  public int onStartCommand(Intent intent, int flags, int startId) {
    new Thread(new Runnable() {
      @Override
      public void run() {
        // 處理具體的邏輯                     // 開啟一個線程處理耗時操作
        stopSelf(); // 讓服務執行完邏輯後自行停止       }     }).start();     return super.onStartCommand(intent, flags, startId);   } }

  上面的代碼就是一個標準的Service的書寫形式,主要包含兩個知識點:Thread子線程的創建stopSelf()方法的調用

  其實,在一般的使用過程中,一部分程式員很容易忘記以上兩個要點,存在遺忘,那麼有沒有更好的辦法能夠實現上面兩個需求呢?

【解決】在Android中,專門提供了一個IntentService類(android.app.IntentService),這個類就能很好的滿足我們的需求!我們直接通過代碼來看:

  (1)新建一個MyIntentService類繼承自IntentService,代碼:

public class MyIntentService extends IntentService{

  public MyIntentService() {
    super("MyIntentService");
  }

  @Override
  protected void onHandleIntent(Intent intent) {
    Log.d("MyIntentService", "MyIntentServiceThread id is " + Thread.currentThread().getId());
  }

  @Override
  public void onDestroy() {
    super.onDestroy();
    Log.d("MyIntentService", "onDestroy executed");
  }
}

  以上代碼做了幾件事:

  1、提供了一個無參的構造方法,並且調用了父類的有參構造函數(這個就不需要我說為什麼了吧);

  2、子類實現父類的onHandleIntent()抽象方法,這個方法好就好在,它是一個已經運行在子線程中的方法。也就是說,服務調用了它,那麼執行的邏輯就如同Thread子線程。

        onHandleIntent = Thread().start() + stopSelf()

  3、onHandleIntent()執行完後會銷毀服務?會selfStop()?接著往下看代碼。

----------------------------------------------------------------------------------------------------------------------------------------

  (2)在xml文件中,創建一個MyIntentService服務按鈕:

<Button
  android:id="@+id/start_intent_service"
  android:layout_width="match_parent"
  android:layout_height="wrap_content"
  android:text="@string/intent_service"/>

  (3)接下來,修改MainActivity中的代碼:

public class MainActivity extends Activity {
  @Override   
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);     setContentView(R.layout.activity_main);     Button startIntentService
= (Button) super.findViewById(R.id.start_intent_service);     startIntentService.setOnClickListener(new View.OnClickListener() {
      @Override       
public void onClick(View v) {         Log.d("MyIntentService", "MainActivity Thread id is " + Thread.currentThread().getId()); // 查看主線程的id         Intent intentService = new Intent(getBaseContext(), MyIntentService.class);         startService(intentService);       }     });   } }

  (4)最後,在AndroidMainfest中註冊服務:

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

【結果】:直接看一下代碼執行的效果。

  從打出的LOG可以看出:

  (1)MyIntentService和MainActivity所在進程的id不一樣的;

  (2)onHandleIntent()方法在執行完邏輯後確實銷毀了服務,效果等同於stopSelf()。

  從上面的分析可以看出onHandleIntent()方法確實相當的好用!


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

-Advertisement-
Play Games
更多相關文章
  • 很多在manifest中的屬性我們經常遺忘了它們,或者經常看到但又不是很明白它的作用。那麼在這裡我就拿了一些屬性簡單的解釋一下,防止以後碰到卻不知道其中的意思。不是很全,以後會斷斷續續的補充吧 一、android:installLocation="internalOnly"android:insta ...
  • 在Android開發過程中,有很多東西都是常常用到的,為了提高效率,將常用的方法做個記錄。 1.在網路編程中,如果還沒建立套接字就使用發送write,會出現異常,封裝後沒問題了(若發送byte[]型自己更改參數類型): public static boolean sendMsg(OutputStre ...
  • 與其他編程語言所不同的是,Swift 並不要求你為自定義類和結構去創建獨立的介面和實現文件。你所要做的是在一個單一文件中定義一個類或者結構體,系統將會自動生成面向其它代碼的外部介面。 註意:通常一個類的實例被稱為對象。然而在Swift 中,類和結構體的關係要比在其他語言中更加的密切,本章中所討論的大 ...
  • 第一種方法:如果使用導航第一個按鈕方法:[self.navigationController pushViewController:secondVC animated:YES];第二個按鈕方法:[self.navigationController popViewControllerAnimated: ...
  • copy與retain的區別: copy是創建一個新對象,retain是創建一個指針,引用對象計數加1。Copy屬性表示兩個對象內容相同,新的對象retain為1 ,與舊有對象的引用計數無關,舊有對象沒有變化。copy減少對象對上下文的依賴。 retain屬性表示兩個對象地址相同(建立一個指針,指針 ...
  • 傳值方式 1 初始化傳值:(順傳) 自定義初始化方法在 UI中,一般在一個界面推送另一個界面的時候,因此要是想用想到既然要用自定義初始化方法,至少要在該方法中去創建該對象.所以這種方式不適合回調. 自定義初始化方法,將需要傳遞的內容作為參數,如需要傳遞多個,則設置多個參數. 在合適的地方進行調用自定 ...
  • 傳值 1 順向傳值 順傳 屬性傳值 運行方式為 A à B 原理: 在B頁面的控制器中,創建需要的屬性,由於控制器是由 A 跳到 B 的,因此可以在 A 中拿到 B 中的屬性,直接賦值.(賦值在控制器跳轉的時候完成) 在 B 中定義一個屬性 在 A 中拿到 B 的非私有屬性 在 A 中直接對 A 拿 ...
  • title: Android學習路線總結,絕對乾貨 tags: Android學習路線,Android學習資料,怎麼學習android grammar_cjkRuby: true 一、前言 不知不覺自己已經做了幾年開發了,由記得剛出來工作的時候感覺自己能牛逼,現在回想起來感覺好無知。懂的越多的時候你 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...