Activity之间使用intent传递大量数据带来问题总结

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Activity之间使用intent传递大量数据带来问题总结相关的知识,希望对你有一定的参考价值。


Activity之间使用Parcel传递大量数据产生的问题。

Activity之间通过intent传递大量数据,导致新Activity无法启动。
Activity之间数据传递方式总结參考 这 里


比較经常使用的是直接利用intent传递,比方使用bundle。例如以下:
Intent intent =new Intent(ActivityA.this,ActivityB.class); 
Bundle bundle =new Bundle();
bundle.putParcelableArrayList("data", dataList);
intent.putExtras(bundle);
startActivity(intent);
问题:当传递数据量过大,比方list的size过大,会导致B无法启动。现象即启动失败。activityB的oncreate()都不会运行。

分析:
官方文档提到TransactionTooLargeException异常,“The Binder transaction failed because it was too large.”即数据传输过大异常。
而且提到这样一句话:“Parcel objects stored in the Binder transaction buffe”,这表明实际上底层parcel对象在不同activity直接传递过程中保存在一个叫做“ Binder transaction buffe”的地方,既然是缓冲区,肯定有限制大小。
官方文档还提到
The Binder transaction buffer has a limited fixed size, currently 1Mb, which is shared by all transactions in progress for the process. Consequently this exception can be thrown when there are many transactions in progress even when most of the individual transactions are of moderate size.
即缓冲区最大1MB,而且这是该进程中全部正在进行中的传输对象所公用的。至于都有哪些传输对象、详细怎么分配,这个还不太清楚。能够肯定的是Activity之间使用Parcel数据传输是有限制大小的。那么在传输大小可能非常大的情况下就要做点处理了。

-------
另外,
该博客Yet another post on Serializable vs Parcelable在对照Serializable和Parcel时提到下面两点数据,仅供參考了解。实际使用各有不同
1. 使用Serializable和parcel传输同样对象,都转换为byte[]后,parcel大概是serializable的20倍了。
2. 可是官方建议使用Parcel,原因是说速度是serializable的将近10倍。

Serializable: it‘s error prone and horribly slow. So in general: stay away from Serializable if possible. 
Parcelable: If you want to pass complex user-defined objects, take a look at the Parcelable interface. It‘s harder to implement, but it has considerable speed gains compared to Serializable.

可是有时候出现该问题时竟然不报错(我遇到的就没抛异常)。甚至没有特殊的log(adb logcat -v threadtime -s ActivityManager 、adb logcat -b events)
并没有像文章或者官方文档中提到的“ throwing a TransactionTooLargeException (or just logging E: !!! FAILED BINDER TRANSACTION !!! on pre 15 API levels)

比較奇葩。
---------

针对parcel传递数据限制大小,自个儿做了个简单实验:
机型:Galaxy Nexus
系统:4.1.2  sdk16
过程:ActivityA,ActivityB。DataBean(每一个对象大概200byte),A启动B并使用Parcel对象传递list<dataBean>。

当list大小为900个时。无法启动B。即数据传输大概在200*900 < 200k
所以依照官方解释,对于详细某一次Activity间传输的大小限制是不确定的。根据使用环境而定。

解决方法:
括弧,赶紧下面解决方式都不是非常理想啊,,,
一. 限制传递数据量
二. 改变传输数据方式(參见Activity之间传递数据的方式
1. 静态static
2. 单例
3. Application
4. 持久化


參考:




















以上是关于Activity之间使用intent传递大量数据带来问题总结的主要内容,如果未能解决你的问题,请参考以下文章

【Android 开发】 : Activity之间传递数据的几种方式

基于Intent实现Activity与Activity之间的数据传递,实现二个Activity的跳转功能

android 中如何使用intent通过传递变量传递数据

Android -- 每日一问:两个 Activity 之间如何传递参数?

android activity之间传递对象 对象为啥要序列化

用intent怎么传递long型数据