Android開發學習之路-回調實現Service向activity傳遞數據


開啟服務的時候,如果我們是通過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

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM