開啟服務的時候,如果我們是通過bindService來綁定服務並且要向服務傳遞數據,可以直接在Intent中設置bundle來達到效果,但是如果是我們需要從服務中返回一些數據到Activity中的時候,實現起來就有各種各樣的方法,比如說使用回調,使用廣播等等,今天說的是使用回調的方法。
新建一個工程,並編寫一個服務:
1 public class MyService extends Service { 2 private boolean connecting = false; 3 private Callback callback; 4 5 @Nullable 6 @Override 7 public IBinder onBind(Intent intent) { 8 return new Binder(); 9 } 10 11 public class Binder extends android.os.Binder { 12 public MyService getService() { 13 return MyService.this; 14 } 15 } 16 17 @Override 18 public void onCreate() { 19 super.onCreate(); 20 connecting = true; 21 new Thread(new Runnable() { 22 23 @Override 24 public void run() { 25 int i = 0; 26 while (connecting == true) { 27 i++; 28 if (callback != null) { 29 callback.onDataChange(i + ""); 30 } 31 try { 32 Thread.sleep(1000); 33 } catch (InterruptedException e) { 34 e.printStackTrace(); 35 } 36 } 37 } 38 }).start(); 39 } 40 41 public void setCallback(Callback callback) { 42 this.callback = callback; 43 } 44 45 public static interface Callback { 46 void onDataChange(String data); 47 } 48 49 @Override 50 public void onDestroy() { 51 super.onDestroy(); 52 connecting = false; 53 } 54 }
在服務中的onCreate方法中,我們打開了一個線程來模擬服務的運行,並在線程每隔1s中給私有變量i賦值遞增,然后我們編寫了一個公有的接口Callback,並且定義了一個該接口的私有成員,並且在onCreate方法中調用了接口里面的函數onDataChange。接下來我們自定義了一個Binder的子類並在這個類中定義了函數返回當前的這個Service,這里的目的就是在Activity中可以訪問到這個Service的回調接口Callback並實現該接口的方法。
Activity代碼如下:
1 public class MainActivity extends AppCompatActivity implements View.OnClickListener, 2 ServiceConnection { 3 4 private TextView tvOut; 5 6 @Override 7 protected void onCreate(Bundle savedInstanceState) { 8 super.onCreate(savedInstanceState); 9 setContentView(R.layout.activity_main); 10 tvOut = (TextView) findViewById(R.id.tvOut); 11 findViewById(R.id.btnBindService).setOnClickListener(this); 12 } 13 14 @Override 15 public void onClick(View v) { 16 bindService(new Intent(this, MyService.class), this, BIND_AUTO_CREATE); 17 } 18 19 @Override 20 public void onServiceConnected(ComponentName name, IBinder service) { 21 MyService.Binder binder = (MyService.Binder) service; 22 MyService myService = binder.getService(); 23 myService.setCallback(new MyService.Callback() { 24 @Override 25 public void onDataChange(String data) { 26 Message msg = new Message(); 27 msg.obj = data; 28 handler.sendMessage(msg); 29 } 30 }); 31 } 32 33 @Override 34 public void onServiceDisconnected(ComponentName name) { 35 36 } 37 38 private Handler handler = new Handler() { 39 @Override 40 public void handleMessage(Message msg) { 41 super.handleMessage(msg); 42 tvOut.setText(msg.obj.toString()); 43 } 44 }; 45 }
因為服務綁定后,會從onBind方法中返回一個Binder對象,這個對象會在onServiceConnectde方法中獲取到,所以我們先從Binder對象中獲取到我們從服務傳遞過來的MyService對象,然后調用MyService對象的setCallback方法來設置我們需要的處理邏輯,這里是把i的值打印出來,因為服務中開啟了線程,所以這里也不能直接更新UI。
總結:回調機制是Java中的一個重要特性,在Android中使用到的地方很廣泛,例如我們給按鈕設定點擊事件等。這里的回調,其實是通過在發送端定義回調接口,並且調用接口的回調方法,然后在接收端實現該接口的方法。只要接口被調用了,就會回調接收端的被實現了的方法,這樣數據就能傳遞過來。
注意:開啟的服務是在主線程中運行的,如果在服務中開啟了線程,那么在子線程中就不能直接更新UI