Android 7.0 BroadcastReceiver接收pendingIntent 传递过来序列化数据

问题比较少见,只有你在跨进程传递数据的时候会碰到,如pendingIntent中

在7.0中通过pendingIntent的bundle传递的数据时,你会发现serializable和parcelable的数据拿不到

如果你只传了string,那是没问题的,但是如果你传了string和一个serializable你会发现,不光serializable拿不到,连string也拿不到了,黑人问好脸吧

原因参考:

https://commonsware.com/blog/2016/07/22/be-careful-where-you-use-custom-parcelables.html

解决方案:

https://stackoverflow.com/questions/18000093/how-to-marshall-and-unmarshall-a-parcelable-to-a-byte-array-with-help-of-parcel/18000094#18000094

他的解决方法很独特,将所有数据转为基本类型bytes再去传递,当然,这可以解决我们的问题

然后我就想既然byte能解决,只传string也没问题,那为什么不干脆直接传string嗯,所以另一个简单的修改方法就是把所有的对象全部转成jsonstring去传递,也可以解决问题

然后续继续寻找有没有更加合适的方案时发现google的issuetracker中有个大神发现了一个更简单的方法,直接把所有数据放到bundle里,然后将bundler作为参数传递即intent.putExtra("data",bundle)也可以解决问题

详情参考:

https://issuetracker.google.com/issues/37097877

以下是简单例子:

public class LongRunningService extends Service {

    private MeasurePlan measurePlan;

    @Nullable

    @Override

    public IBinder onBind(Intent intent) {

        return null;

    }

    @Override

    public int onStartCommand(Intent intent, int flags, int startId) {

      AlarmManager manager = (AlarmManager) getSystemService(ALARM_SERVICE);

      Intent intent = new Intent(this, AlarmReceiver.class);  //AlarmReceiver:继承BroadcastReceiver的类

      Bundle bundle = new Bundle();

      bundle.putSerializable("MeasurePlan", measurePlan); //measurePlan:要传递的实体类

      intent.putExtra("data",bundle);

      //参数:getBroadcast(Context context, int requestCode, Intent intent, int flags),flag是标记的意思,可以通过PendingIntent点调用不同标记

      PendingIntent pi = PendingIntent.getBroadcast(this, 0, i, 0);

      //ELAPSED_REALTIME_WAKEUP表示让定时任务的出发时间从系统开机算起,并且会唤醒CPU。

      manager.set(AlarmManager.ELAPSED_REALTIME_WAKEUP, triggerAtTime, pi);

  }

}

AlarmManager里面一些方法 :

1.set(int type,long startTime,PendingIntent pi);

该方法用于设置一次性闹钟,第一个参数表示闹钟类型,第二个参数表示闹钟执行时间,第三个参数表示闹钟响应动作。

2.setRepeating(int type,long triggerAtTime,long intervalTime,PendingIntent pi);

该方法用于设置重复闹钟,第一个参数表示闹钟类型,第二个参数表示闹钟首次执行时间,第三个参数表示闹钟两次执行的间隔时间, 第三个参数表示闹钟响应动作。

3.setInexactRepeating(int type,long startTime,long intervalTime,PendingIntent pi);

该方法也用于设置重复闹钟,与第二个方法相似,不过其两个闹钟执行的间隔时间不是固定的而已。

type:闹钟类型,有五个可选值

1.ELAPSED_REALTIME:以手机开机的时间为基准

2.ELAPSED_REALTIME_WAKEUP:以手机开机的时间为基准,并且可以在休眠时发出广播

3.RTC:以UTC标准时间为基准

4.RTC_WAKEUP:以UTC标准时间为基准,并且可以在休眠时发出广播。这种方式是最常见的形式。

5.POWER_OFF_WAKEUP:关机状态下也能进行提示

long startTime:

表示闹钟第一次执行的时间,可以自己设置,也可以使用系统当前时间,以毫秒为单位。本属性与第 

一个属性(type)密切相关,如果第一个参数对 应的闹钟使用的是相对时间(ELAPSED_REALTIME和ELAPSED_REALTIME_WAKEUP), 那么本属性就得使用相对时间(相对于 系统启动时间来 

说),比如当前时间就表示为:SystemClock.elapsedRealtime(); 如果第一个参数对应的闹钟使用的是绝对时间 (RTC、RTC_WAKEUP、POWER_OFF_WAKEUP),那么本属性就得使用绝对时间, 比如当前时间就表示 为:System.currentTimeMillis()。

long intervalTime: 

闹钟的间隔时间,也是毫秒为单位。

PendingIntent pi: 

这个就是我们可以执行的动作,可以去启动一个service,发送一个广播,启动一个activity,方法一看就明白了。 

最后一个小提示: 

Calendar.HOUR_OF_DAY (24小时) 

Calendar.HOUR (12小时)

完成之后在自定义类中获取数据:

public class AlarmReceiver extends BroadcastReceiver {

    private MeasurePlan measurePlan;

    private MedicinePlan medicinePlan;

    @Override

    public void onReceive(Context context, Intent intent) {

    Bundle data = intent.getBundleExtra("data");

    medicinePlan = (MedicinePlan)data.get("MedicinePlan");

  }

}

最后是清单文件AndroidManifest.xml的注册:

    android:name=".activitys.inform.remindservice.AlarmReceiver">

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

作者:EncounterTo

来源:CSDN

原文:https://blog.csdn.net/EncounterTo/article/details/79305324

版权声明:本文为博主原创文章,转载请附上博文链接!

你可能感兴趣的:(Android 7.0 BroadcastReceiver接收pendingIntent 传递过来序列化数据)