提问者:小点点

碎片newInstance保存到捆绑包的数据过多-Android.os.TransactionToolargeException


我已经创建了碎片标准方式(Android工作室生成的骨架)。所以我有newInstance静态方法:

public static EventsMapFragment newInstance(List<Event> eventList, String accessToken, String tokenType, String cityId, boolean isMessage)
{
     Bundle args = new Bundle();
     args.putString("events", new Gson().toJson(eventList));
     EventsMapFragment fragment = new EventsMapFragment();

     ///...........
     fragment.setArguments(args);
     return fragment;
}

然后,在oncreateoverride中,我需要检索我的集合,非常标准的方式:

 @Override
 public void onCreate(Bundle savedInstanceState)
 {
    super.onCreate(savedInstanceState);
    if (getArguments() != null)
    {
        String json = getArguments().getString("events");
        Type type = new TypeToken<List<Event>>(){}.getType();
        mEventList = new Gson().fromJson(json, type);
   }

问题是eventlist集合非常大,因此我得到了异常:

java.lang.RuntimeException:Android.os.TransactionToolargeException:数据包大小592196字节

在我的应用程序中的其他地方(奇怪的是,json被正确地传递,但是系统不允许传递更多的东西到bundle,跟踪问题的确切位置并不容易)。

因为newInstance()方法是静态的,所以我无法创建retain fragment并将数据传递到它中(在这种情况下,我实际上不知道如何做到这一点)。

我的解决方案是使用weakReference来传递数据,类似于:

public class DataHolder
{
    private static final DataHolder holder = new DataHolder();
    Map<String, WeakReference<Object>> data = new HashMap<>();

    public static DataHolder getInstance() {return holder;}

    public void save(String id, Object object)
    {
        data.put(id, new WeakReference<>(object));
    }

    public Object retrieve(String id)
    {
        WeakReference<Object> objectWeakReference = data.get(id);
        return objectWeakReference.get();
    }
}

然后,要存储数据(而不是使用bundle):

DataHolder.getInstance().save("eventList", eventList);

并检索数据:

mEventList = (List<Event>) DataHolder.getInstance().retrieve("eventList");

但是,这在KitKat上不起作用(在5.1和更新版本上似乎可以很好地工作)。主要问题我无法在KitKat上调试,因为multidex,所以调试编译甚至无法在KitKat上运行(发行版有minifyEnabled=true并且没有multidex)。但在KitKat上,似乎是这样的:

mEventList = (List<Event>) DataHolder.getInstance().retrieve("eventList");

始终为空。

你有什么想法吗?我目前的解决方案是检查构建版本并将json传递到KitKat上的bundle(它在KitKat上工作,问题是7.0及更新版本),并在更新的Android版本上使用WeakReference。

但实际上我对此并不满意。

[编辑]所以我的最后一次尝试是从父级activity获取事件集合。在主activity,我有以下代码(一个应用程序抽屉菜单点击):

if (menuId == R.id.nav_events_map)
{
  eventsToDraw =  Lists.newArrayList(Collections2.filter(events,x->!Utils.isGeojsonEmpty(x.getGeojson())));
  //....................
  if(eventsToDraw.size() >0)
  {
    mFragment = EventsMapFragment.newInstance();
    ReplaceFragment(mFragment);
  }
  else
  {
   mFragment = NoDataFragment.newInstance(getString(R.string.message_no_events));
   ReplaceFragment(mFragment);
 }
}

EventStoDraw是私有字段,我只是创建了getter:

public List<Event> getEventsToDraw()
{
  return eventsToDraw;
}

然后,在我的地图片段内部:

@Override
 public View onCreateView(@NonNull LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState)
 {
  //Map initialization stuff
  //....................

   mEventList = ((MainActivity)getActivity()).getEventsToDraw();

  //Draw on map stuff etc
 }

问题是我总是得到meventlistnull。经过一些调试,我注意到getEventStoDraw()被调用为early,因此它不会返回我的集合,它返回null,因为还没有可用的数据。


共1个答案

匿名用户

它在KitKat上起作用

不,不是。1MB的IPC事务限制已经存在了很多年。但是,该限制只适用于您的流程的所有同时IPC事务,因此失败将是概率性的。例如,592196不到1MB,因此,为您崩溃的东西有时可能会通过OK,因为碰巧在那一刻与IPC发生的其他事情很少。

所以我有newInstance静态方法

不要传递事件集合。传递片段获取事件集合所需的最起码的信息。使用流程级POJO缓存来最大限度地减少冗余I/O,但如果需要,支持再次访问数据库/网络/任何内容,因此您可以处理配置更改(流程缓存仍然存在)和流程终止(但用户返回任务的速度相当快,因此仍然使用实例stat)。