How to use PendingIntent to communicate from a Service to a client/Activity?

前端 未结 3 1867
迷失自我
迷失自我 2021-01-30 02:41

I have been reading the following text on the Android Developers Site, specifically under the Framework Topics -> Services -> Starting a Service.

There it states the fol

相关标签:
3条回答
  • 2021-01-30 03:07

    In order to perform communication between service and activity. You can also use Binder as mentioned in Official Android Example http://developer.android.com/reference/android/app/Service.html#LocalServiceSample

    For detail explanation see this answers https://stackoverflow.com/a/36983011/4754141

    0 讨论(0)
  • 2021-01-30 03:20

    As written here

    Communication between service and Activity can be done using PendingIntent.For that we can use createPendingResult().createPendingResult() creates a new PendingIntent object which you can hand to service to use and to send result data back to your activity inside onActivityResult(int, int, Intent) callback.Since a PendingIntent is Parcelable , and can therefore be put into an Intent extra,your activity can pass this PendingIntent to the service.The service, in turn, can call send() method on the PendingIntent to notify the activity via onActivityResult of an event.

    Activity

    public class PendingIntentActivity extends AppCompatActivity
    {
    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);
    
    PendingIntent pendingResult = createPendingResult(
    100, new Intent(), 0);
    Intent intent = new Intent(getApplicationContext(), PendingIntentService.class);
    intent.putExtra("pendingIntent", pendingResult);
    startService(intent);
    
    }
    
    @Override
    protected void onActivityResult(int requestCode, int resultCode, Intent data) {
    if (requestCode == 100 && resultCode==200) {
    Toast.makeText(this,data.getStringExtra("name"),Toast.LENGTH_LONG).show();
    }
    super.onActivityResult(requestCode, resultCode, data);
    }
    }
    

    Service

    public class PendingIntentService extends Service {
    
        private static final String[] items= { "lorem", "ipsum", "dolor",
                "sit", "amet", "consectetuer", "adipiscing", "elit", "morbi",
                "vel", "ligula", "vitae", "arcu", "aliquet", "mollis", "etiam",
                "vel", "erat", "placerat", "ante", "porttitor", "sodales",
                "pellentesque", "augue", "purus" };
        private PendingIntent data;
    
        @Override
        public void onCreate() {
            super.onCreate();
        }
    
        @Override
        public int onStartCommand(Intent intent, int flags, int startId) {
    
            data = intent.getParcelableExtra("pendingIntent");
    
            new LoadWordsThread().start();
            return START_NOT_STICKY;
        }
    
        @Override
        public IBinder onBind(Intent intent) {
            return null;
        }
    
        @Override
        public void onDestroy() {
            super.onDestroy();
        }
    
        class LoadWordsThread extends Thread {
            @Override
            public void run() {
                for (String item : items) {
                    if (!isInterrupted()) {
    
                        Intent result = new Intent();
                        result.putExtra("name", item);
                        try {
                            data.send(PendingIntentService.this,200,result);
                        } catch (PendingIntent.CanceledException e) {
    
                            e.printStackTrace();
                        }
                        SystemClock.sleep(400);
    
                    }
                }
            }
        }
    }
    
    0 讨论(0)
  • 2021-01-30 03:24

    Question was asked few months ago, but in case anyone is still looking for answer I hope I can help.

    In the example below we have local service, responsible for performing some time-consuming operations. Activity makes the requests to the service, but does not bind to it - just sends the intent with request. Additionally, Activity includes the information of BroadcastReceiver that should be called back when service is done with the requested task. The information is passed by PendingIntent. The service handles the task in background thread and when task is finished, service broadcasts the BroadcastReceiver with an answer.

    1. Create BroadcastReceiver subclass:

    public class DataBroadcastReceiver extends BroadcastReceiver {
       static Logger log = LoggerFactory.getLogger(DataRequestService.class);   
       @Override
       public void onReceive(Context context, Intent intent) {
          log.info(" onReceive");
       }
    }
    

    This broadcast receiver will be notified from service, when task is done.

    2. Create Service

    public class DataRequestService extends Service {
    
       private final class ServiceHandler extends Handler {
          public ServiceHandler(Looper looper) {
             super(looper);
          }
    
          @Override
          public void handleMessage(Message msg) {
             log.info("handleMessage");
             //... performing some time-consuming operation         
             Bundle bundle = msg.getData();
             PendingIntent receiver = bundle.getParcelable("receiver");
             // Perform the operation associated with PendingIntent
             try {            
                //you can attach data from the operation in the intent.
                Intent intent = new Intent();
                Bundle b = new Bundle();
                //b.putString("key", value);
                intent.putExtras(b);
                receiver.send(getApplicationContext(), status, intent);
             } catch (CanceledException e) {         
             e.printStackTrace();
             }         
          }
       }
       
       @Override
       public void onStart(Intent intent, int startId) {
          Bundle bundle = intent.getExtras();
          Message msg = mServiceHandler.obtainMessage();
          msg.setData(bundle);
          mServiceHandler.sendMessage(msg);
       }
    

    Well, the most important part is in handleMessage() method. Service simply makes the broadcasts operation for delivering results to Broadcast Receiver.

    3. You also need to register your broadcast receiver and service in Manifest.xml

    <manifest xmlns:android="http://schemas.android.com/apk/res/android"
        package="com.ramps.servicetest"
        android:versionCode="1"
        android:versionName="1.0" >
       ....
           <service android:name=".service.DataRequestService" android:exported="false"/>
           <receiver android:name=".service.DataBroadcastReceiver"></receiver>
        </application>
    </manifest><br>
    

    4. And finally, make request to your service from Activity:

    Intent serviceIntent = new Intent(context, DataRequestService.class);   
       @Override
       public void onClick(View v) {
          //this is the intent that will be broadcasted by service.
          Intent broadcastReceiverIntent = new Intent(context, DataBroadcastReceiver.class);      
          //create pending intent for broadcasting the DataBroadcastReceiver
          PendingIntent pi = PendingIntent.getBroadcast(context, 0, broadcastReceiverIntent, 0);      
          Bundle bundle = new Bundle();            
          bundle.putParcelable("receiver", pi);
          //we want to start our service (for handling our time-consuming operation)
          Intent serviceIntent = new Intent(context, DataRequestService.class);
          serviceIntent.putExtras(bundle);
          context.startService(serviceIntent);
       }
    



    5. Delivering response to original client/activity.

    You can have abstract activity from which all your activities will be extending. This abstrct activity can automatically register/deregister itself as a response listener in broadcast receiver. Not many options here actually, but it is important that if you keep static references to your activity then you must remove the refernece when activity is destroyed.

    Regards,
    Ramps

    0 讨论(0)
提交回复
热议问题