在Android的导航体系结构组件中传递复杂的对象作为参数

时间:2018-12-05 09:12:03

标签: android kotlin navigation android-architecture-components android-livedata

我一直在寻找Android architechture组件,最近一直在寻找Navigation组件。

我试图将一个对象作为参数传递,以便下一个片段可以检索该数据,但是要这样做,我需要做以下两件事之一:

  1. 通过Bundle传递它,这将使我实现对该对象的Parcelable接口。
  2. 使用我尝试过的“ Safeargs”插件,它似乎在后台利用了Bundles,并且仍然需要实现Parcelable接口。

关于这些选项的事情是,我读到Parcelable利用了反射功能,并且在时间上会变得非常昂贵


我也曾尝试构建一个“ SharedMasterDetailsViewModel”,但是由于在新创建的Fragment上未执行可观察到的回调,因此运气不佳。 (我认为LiveData在创建片段之前会执行回调)

以下是一些有关我如何尝试实现此目标的代码

SharedSessionViewModel

class SessionSharedViewModel : ViewModel() {


    var sharedSession: LiveData<Session> = MutableLiveData()
        private set

    fun setSession(data: Session) {
        val casted = this.sharedSession as MutableLiveData<Session>
        casted.postValue(data)
    }
}

MasterFragment

override fun onItemClicked(item: Session) {
    sessionSharedViewModel.setSession(item) // Item is a complex object of mine
    this@HomeFragment.findNavController().navigate(R.id.sessionDetailsFragment)
}

DetailsFragment

class SessionDetailsFragment : Fragment() {

    companion object {
        fun newInstance() = SessionDetailsFragment()
    }

    private lateinit var sharedViewModel: SessionSharedViewModel

    override fun onCreateView(
        inflater: LayoutInflater, container: ViewGroup?,
        savedInstanceState: Bundle?
    ): View? {
        return inflater.inflate(R.layout.session_details_fragment, container, false)
    }

    override fun onActivityCreated(savedInstanceState: Bundle?) {
        Log.d("SESSIONDETAILS","created!")

        super.onActivityCreated(savedInstanceState)
        sharedViewModel = ViewModelProviders.of(this).get(SessionSharedViewModel::class.java)
        sharedViewModel.sharedSession.observe({this.lifecycle},{ handleUI(it!!)})
    }

    fun handleUI(sharedSession: Session) {
        Toast.makeText(activity, "This is inside new activity: ${sharedSession.title()}", Toast.LENGTH_SHORT)
    }
}

我最后的希望是将我的对象序列化为一个JSON字符串,并在我的Detail片段的onCreateActivity lyfecycle钩子上重新解析该对象,但是我觉得那不是正确的解决方案。

在最坏的情况下,我只是传递对象的ID并从网络中重新获取它,但由于我已经有了想要显示的信息,因此我希望将其作为参数传递。

1 个答案:

答案 0 :(得分:2)

TL; DR

你不能。

实际解释

第一件事;以下语句

关于这些选项的事情是,我读到Parcelable利用了反射功能,并且在时间上会变得非常昂贵。

是谎言

自从实现Parcelable以来,您只提供了有关如何使用基本原始类型进行序列化和反序列化的方法:IE:writeBytesreadBytesreadInt,{{ 1}}。

writeInt不使用反射。 Parcelable做到了!

虽然确实如此,但您被迫使用Serializable喷射大脑开发了一个非常有用的注释,从而免除了必须编写名为Parcelable的可打包实现的痛苦。

样品用量:

@Parcelize

现在,您可以传递类的实例而无需编写@Parcelize data class User(val username: String, val password: String) : Parcelable 实现的一行。

更多信息here