Android面接3——Service

13120 ワード

本文はAndroid Serviceの2種類の起動方式の詳細(総括版)を参考に書いて、ここで感謝します!
1.サービスの2つの起動方法
1.startService()完全ライフサイクル関数は次のとおりです.
 onCreate() -> onStartCommand -> onDestroy() 

onStartService()メソッドを呼び出すと、*stopService()またはstopSelf()を呼び出すまで、サービスは停止して破棄されます.
  • onCreate()
  • Serviceは作成されず、startService()を呼び出すとメソッド
  • が呼び出されます.
  • Serviceはすでに実行中であり、startService()の呼び出し時にメソッド
  • は呼び出されません.
  • onStartCommand()startService()メソッドを複数回実行すると、現在のメソッドも複数回呼び出されます.このメソッドでは、入力されたパラメータに基づいて実際の操作を行います.
  • onDestroy()は、サービスが破棄されたときにこの方法を実行します.

  • startService()インスタンス
  • サービスインスタンス
  • を作成する
    package com.example.servicetest;
    
    import android.app.Service;
    import android.content.Intent;
    import android.os.IBinder;
    import android.util.Log;
    
    public class ServiceByStart extends Service {
        public ServiceByStart() {
        }
    
        @Override
        public void onCreate() {
            super.onCreate();
            Log.i("karl", "onCreate - Thread ID = "+ Thread.currentThread().getId());
        }
    
        @Override
        public int onStartCommand(Intent intent, int flags, int startId) {
            Log.i("karl", "onStartCommand -startId = "+ startId +", Thread ID = "+ Thread.currentThread().getId());
            return super.onStartCommand(intent, flags, startId);
        }
    
        @Override
        public IBinder onBind(Intent intent) {
            Log.i("karl", "onBind Thread ID = "+ Thread.currentThread().getId());
            return null;
        }
    
        @Override
        public void onDestroy() {
            super.onDestroy();
            Log.i("karl", "onDestroy Thread ID = "+ Thread.currentThread().getId());
        }
    }
    
  • ActivityでService
  • を起動
    package com.example.servicetest;
    
    import android.content.Intent;
    import android.support.v7.app.AppCompatActivity;
    import android.os.Bundle;
    import android.util.Log;
    
    public class MainActivity extends AppCompatActivity {
    
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_main);
            Log.i("karl", "Activity Thread ID = "+ Thread.currentThread().getId());
            Log.i("karl", "before service started");
            Intent intent = new Intent(this, ServiceByStart.class);
            startService(intent);
            Log.i("karl", "after service started");
    
            stopService(intent);
            Log.i("karl", "after service stopped");
        }
    }
    
    

    2.bindService()の完全なライフサイクル関数は次のとおりです.
    onCreate() -> onBind() -> onUnbind() -> onDestroy()
    
  • bindService()で起動されたサービスと呼び出し元の関係はclient-server関係です.サービスは1つしかありませんが、サービスにバインドされているクライアントはいくつかあります.ここでclientとは、あるActivityなどのコンポーネントを指します.
  • クライアントは、IBinderインタフェースを介してサービスインスタンスを取得することができ、クライアントでサービス内のカスタムメソッドを呼び出す必要があり、柔軟なインタラクションを実現します.
  • bindService()方式で開始されたサービスは、そのライフサイクルがバインドされたclientと密接に関連しています.クライアントが破棄されると、対応するServiceとのバインドが自動的に解除され、クライアントはunbindService()メソッドを呼び出して明示的にバインドを解除することもできます.クライアントがサービスにバインドされていない場合、サービスは破棄されます.

  • bindServiceコードインスタンス
    質問:Seviceを作成し、2つのActivity MainActivityとTestActivityを作成し、MainActivityからTestActivityにジャンプします.
  • bind方式のサービスを作成するbindService()方式をサービスにサポートさせるには、
  • という作業が必要です.
  • SeviceのonBind()メソッドでIBinderタイプのインスタンス
  • を返す
  • onBind()メソッドで返されるIBinderインスタンスでは、現在のサービスインスタンスを返すことができます.最も簡単な方法は、サービスにBinderのカスタム内部クラスを作成し、getService()のような方法を加えてサービスインスタンス
  • を返すことです.
    package com.example.servicetest;
    
    import android.app.Service;
    import android.content.Intent;
    import android.os.Binder;
    import android.os.IBinder;
    import android.util.Log;
    
    import java.util.Random;
    
    public class ServiceByBind extends Service {
        public ServiceByBind() {
        }
    
        // client    Binder       Service   
        public class MyBinder extends Binder {
            public ServiceByBind getService() {
                return ServiceByBind.this;
            }
        }
    
        //    Binder   ,   client   server      
        private Binder binder = new MyBinder();
    
        private final Random random = new Random();
    
        @Override
        public void onCreate() {
            super.onCreate();
            Log.i("karl", "onCreate - Thread ID = "+ Thread.currentThread().getId());
        }
    
        @Override
        public int onStartCommand(Intent intent, int flags, int startId) {
            Log.i("karl", "onStartCommand -startId = "+ startId +", Thread ID = "+ Thread.currentThread().getId());
            return super.onStartCommand(intent, flags, startId);
        }
    
        @Override
        public IBinder onBind(Intent intent) {
            Log.i("karl", "onBind Thread ID = "+ Thread.currentThread().getId());
            return binder;
        }
    
        @Override
        public boolean onUnbind(Intent intent) {
            Log.i("karl", "onUnbind from : "+ intent.getStringExtra("from"));
            return false;
        }
    
        @Override
        public void onDestroy() {
            super.onDestroy();
            Log.i("karl", "onDestroy Thread ID = "+ Thread.currentThread().getId());
        }
    
        /**
         *      ,     
         * @return         
         */
        public int getRandomInt() {
            return random.nextInt();
        }
    
    }
    
  • クライアント側がクライアント側を呼び出すには、
  • という作業が必要です.
  • ServiceConnectionインスタンスを作成し、onServiceConnected()メソッドとonServiceDisconnected()メソッド
  • を書き換えます.
  • onServiceConnected()メソッドを実行すると、IBinderインスタンスによってServiceインスタンスが取得され、client側とserver側の接続
  • が実現する.
  • onServiceDisconnected()メソッドを実行すると、client側とserver側は接続を切断する
  • .
    MainActivity.JAvaは以下の通りです.
    package com.example.servicetest;
    
    import android.content.ComponentName;
    import android.content.Intent;
    import android.content.ServiceConnection;
    import android.os.IBinder;
    import android.support.v7.app.AppCompatActivity;
    import android.os.Bundle;
    import android.util.Log;
    import android.view.View;
    import android.widget.Button;
    
    public class MainActivity extends AppCompatActivity implements View .OnClickListener{
    
        private Button bind, unbind, start, finish;
        private ServiceByBind byBind = null;
        private boolean isBind = false;
    
        private ServiceConnection connection = new ServiceConnection() {
            @Override
            public void onServiceConnected(ComponentName name, IBinder service) {
                isBind = true;
                ServiceByBind.MyBinder myBinder = (ServiceByBind.MyBinder) service;
                byBind = myBinder.getService();
                Log.i("karl", "main - service bound ComponentName:"+ name.getShortClassName());
                int num = byBind.getRandomInt();
                Log.i("karl", "main - random int = "+ num);
            }
    
            @Override
            public void onServiceDisconnected(ComponentName name) {
                isBind = false;
                Log.i("karl", "main service unbound ComponentName:"+ name.getShortClassName());
            }
        };
    
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_main);
            Log.i("karl", "main - onCreate() Thread ID : "+ Thread.currentThread().getId());
    
    /*
    
            Log.i("karl", "Activity Thread ID = "+ Thread.currentThread().getId());
            Log.i("karl", "before service started");
            Intent intent = new Intent(this, ServiceByStart.class);
            startService(intent);
            Log.i("karl", "after service started");
    
            stopService(intent);
            Log.i("karl", "after service stopped");
    */
            initView();
    
        }
    
        private void initView() {
            bind = findViewById(R.id.bind);
            unbind = findViewById(R.id.unbind);
            start = findViewById(R.id.startActivity);
            finish = findViewById(R.id.finish);
        }
    
        @Override
        protected void onResume() {
            super.onResume();
            bind.setOnClickListener(this);
            unbind.setOnClickListener(this);
            start.setOnClickListener(this);
            finish.setOnClickListener(this);
        }
    
        @Override
        public void onClick(View v) {
            switch (v.getId()) {
                case R.id.bind:
                    Intent intent = new Intent(this, ServiceByBind.class);
                    intent.putExtra("from", "main");
                    Log.i("karl", "=============================");
                    Log.i("karl", "main bindService");
                    bindService(intent, connection, BIND_AUTO_CREATE);
                    break;
                case R.id.unbind:
                    if(isBind) {
                        Log.i("karl", "main unbindService");
                        unbindService(connection);
                    }
                    break;
                case R.id.startActivity:
                    Intent intent1 = new Intent(this, TestActivity.class);
                    Log.i("karl", "start activity test");
                    startActivity(intent1);
                    break;
                case R.id.finish:
                    Log.i("karl", "main finished");
                    finish();
                    break;
            }
        }
    
        @Override
        protected void onDestroy() {
            super.onDestroy();
            Log.i("karl", "main destroyed");
        }
    }
    

    TestActivity.JAvaは以下の通りです.
    package com.example.servicetest;
    
    import android.content.ComponentName;
    import android.content.Intent;
    import android.content.ServiceConnection;
    import android.os.IBinder;
    import android.support.v7.app.AppCompatActivity;
    import android.os.Bundle;
    import android.util.Log;
    import android.view.View;
    import android.widget.Button;
    
    public class TestActivity extends AppCompatActivity implements View .OnClickListener{
    
        private ServiceByBind byBind = null;
        private boolean isBind = false;
        private Button bind, unbind, finish;
    
    
        private ServiceConnection connection = new ServiceConnection() {
            @Override
            public void onServiceConnected(ComponentName name, IBinder service) {
                isBind = true;
                ServiceByBind.MyBinder myBinder = (ServiceByBind.MyBinder) service;
                byBind = myBinder.getService();
                Log.i("karl", "test - service bound ComponentName:"+ name.getShortClassName());
                int num = byBind.getRandomInt();
                Log.i("karl", "test - random int = "+ num);
            }
    
            @Override
            public void onServiceDisconnected(ComponentName name) {
                isBind = false;
                Log.i("karl", "test service unbound ComponentName:"+ name.getShortClassName());
            }
        };
    
    
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_test);
    
            initView();
        }
    
        private void initView() {
            bind = findViewById(R.id.bind);
            unbind = findViewById(R.id.unbind);
            finish = findViewById(R.id.finish);
        }
    
        @Override
        protected void onResume() {
            super.onResume();
            bind.setOnClickListener(this);
            unbind.setOnClickListener(this);
            finish.setOnClickListener(this);
        }
    
        @Override
        public void onClick(View v) {
            switch (v.getId()) {
                case R.id.bind:
                    Intent intent = new Intent(this, ServiceByBind.class);
                    intent.putExtra("from", "test");
                    Log.i("karl", "=============================");
                    Log.i("karl", "test bindService");
                    bindService(intent, connection, BIND_AUTO_CREATE);
                    break;
                case R.id.unbind:
                    if(isBind) {
                        Log.i("karl", "test unbindService");
                        unbindService(connection);
                    }
                    break;
                case R.id.finish:
                    Log.i("karl", "test finished");
                    finish();
                    break;
            }
        }
    
        @Override
        protected void onDestroy() {
            super.onDestroy();
            Log.i("karl", "test onDestroy");
        }
    }
    

    2.サービスが破棄されないことを保証する方法
  • onStartCommand()メソッドの戻り値を調整する
  • START_NOT_STICKYは、サービスが実行するプロセスがAndroidシステムによって強制的に殺された後、サービスを再作成しないことを示しています.
  • START_STICKYは、サービスが実行されるプロセスがAndroidシステムによって強制的に破棄されると、サービスが再構築されることを示していますが、onStartCommand()メソッドのintentパラメータ値はnullです.
  • START_REDELIEVER_INTENTは、サービスが実行するプロセスがAndroidシステムによって強制的に破棄されると、サービスが再構築され、現在のサービスが破棄される前にintentパラメータのパラメータ値が保存されることを示します.
  • サービスの優先度をAndroidManifestに上げる.xmlファイルのラベルではandroid:priority=「1000」という属性で最高優先度を設定できます.1000は最高値で、数字が小さいほど優先度が低く、ブロードキャストにも適用されます.
  • サービス・プロセスの優先度を上げるには、startForeground()を使用してサービスをフロント・ステータスに配置します.これにより、メモリが低い場合、殺される確率が低くなります.
  • onDestroy()メソッドでサービスを再起動します.サービスがonDestroy()に移行すると、カスタムブロードキャストが送信され、ブロードキャストが受信されるとサービスが再起動されます.