- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我有一个带有一个选项卡的 ViewPager,它通过带有 LiveData 对象的 ViewModel 从 Room 检索数据。
当我切换到任何选项卡并返回显示项目列表的选项卡时,一切正常,但当我从后台返回时,我不知道为什么,但我收到的列表少了。
例如第一次进入“列表 fragment ”。
我从 5 个不同的实体检索数据并显示 5 个列表项
转到背景并返回“列表 fragment ”
4个列表有数据但1个为空
转到背景并返回“列表 fragment ”
3 个列表有数据但 2 个为空
... 直到完全为空,然后我切换选项卡并且列表填满了
感谢您的帮助。
View 模型
class TagsViewModel(application: Application) : AndroidViewModel(application) {
private var repository: TagsRepository = TagsRepository(application)
private var listText: LiveData<List<Text>> = repository.getAllText()
private var listUrl: LiveData<List<Url>> = repository.getAllUrl()
private var listEmail: LiveData<List<Email>> = repository.getAllEmail()
private var listPhone: LiveData<List<Phone>> = repository.getAllPhone()
private var listLauncher: LiveData<List<Launcher>> = repository.getAllLauncher()
fun getData(): List<BaseTag>{
var allData= mutableListOf<BaseTag>()
if (listText.value != null){
listText.value!!.forEach {
allData.add(it)
}
}
if (listUrl.value != null){
listUrl.value!!.forEach {
allData.add(it)
}
}
if (listEmail.value != null){
listEmail.value!!.forEach {
allData.add(it)
}
}
if (listPhone.value != null){
listPhone.value!!.forEach {
allData.add(it)
}
}
if (listLauncher.value != null){
listLauncher.value!!.forEach {
allData.add(it)
}
}
return allData
}
//-------TEXT--------
fun insertText(text: Text) {
repository.insertText(text)
}
fun deleteTextByID(id: Int) {
repository.deleteTextByID(id)
}
fun getAllText(): LiveData<List<Text>> {
return listText
}
fun deleteAllText() {
repository.deleteAllText()
}
repository getAllText() 返回 tagsDao.getTextList()
而 Dao 是:
@Query("SELECT * FROM Text")
fun getTextList(): LiveData<List<Text>>
编辑
我将 viewPager 更改为不再重新创建 fragment ,在 onViewCreated 中观察 LiveData,在 onPause 中清除列表并在 onResume 中再次设置数据并添加 viewLyfeCycleOwner
override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?,
savedInstanceState: Bundle?): View? {
v = inflater.inflate(R.layout.fragment_my_tags, container, false)
return v
}
override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
super.onViewCreated(view, savedInstanceState)
setupUI()
}
private fun setupUI(){
val recyclerView = v.findViewById<RecyclerView>(R.id.recycler_view)
recyclerView.layoutManager = LinearLayoutManager(context)
recyclerView.setHasFixedSize(true)
recyclerView.adapter = adapter
tagsViewModel = ViewModelProviders.of(this).get(TagsViewModel::class.java)
getTags()
(activity as MainActivity).mToolbar.onMenuItemClick {
when (it!!.itemId){
R.id.btnDeleteAll -> toast("sort")
R.id.btnOrderBy -> tagsViewModel.deleteAll()
}
}
}
private fun getTags(){
tagsViewModel.getAllText().observe(viewLifecycleOwner,
Observer<List<Text>> { t -> adapter.setTags(t!!) })
tagsViewModel.getAllEmail().observe(viewLifecycleOwner,
Observer<List<Email>> { t -> adapter.setTags(t!!) })
tagsViewModel.getAllUrl().observe(viewLifecycleOwner,
Observer<List<Url>> { t -> adapter.setTags(t!!) })
tagsViewModel.getAllPhone().observe(viewLifecycleOwner,
Observer<List<Phone>> { t -> adapter.setTags(t!!) })
tagsViewModel.getAllLauncher().observe(viewLifecycleOwner,
Observer<List<Launcher>> { t -> adapter.setTags(t!!) })
}
override fun onPause() {
adapter.clear()
super.onPause()
}
override fun onResume() {
adapter.setTags(tagsViewModel.getData())
super.onResume()
}
它的工作方式和以前一样,当我在 onResume 中从后台返回时,我的 viewModel 总是少返回一个完全随机的列表,有时是 textList,有时是 emailList ... 为什么它会丢失数据?我不知道为什么。
编辑2
class TagsAdapter : RecyclerView.Adapter<TagsAdapter.TagHolder>() {
private var baseList: MutableList<BaseTag> = ArrayList()
override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): TagHolder {
val itemView = LayoutInflater.from(parent.context)
.inflate(R.layout.tag_item, parent, false)
return TagHolder(itemView)
}
override fun onBindViewHolder(holder: TagHolder, position: Int) {
val tag = baseList[position]
holder.tvAlias.text = tag.content
holder.tvDate.text = tag.date?.let { AppUtils.toDate(it) }
holder.ivType.background = tag.type?.let { getDrawable(it) }
}
override fun getItemCount(): Int {
return baseList.size
}
fun setTags(list: List<BaseTag>) {
if (this.baseList.isEmpty()) {
this.baseList = list as MutableList<BaseTag>
} else {
this.baseList.addAll(list)
}
orderList()
notifyDataSetChanged()
}
private fun getDrawable(type: String): Drawable? {
when (type) {
AppConstants.TYPE_TEXT -> return ContextCompat.getDrawable(App.instance, R.drawable.ic_text)
AppConstants.TYPE_EMAIL -> return ContextCompat.getDrawable(App.instance, R.drawable.ic_write)
AppConstants.TYPE_URL -> return ContextCompat.getDrawable(App.instance, R.drawable.ic_clean)
AppConstants.TYPE_PHONE -> return ContextCompat.getDrawable(App.instance, R.drawable.ic_mobile)
AppConstants.TYPE_LAUNCHER -> return ContextCompat.getDrawable(App.instance, R.drawable.ic_code)
}
return null
}
private fun orderList(){
baseList.sortBy { it.date }
baseList.reverse()
}
fun clear(){
this.baseList.clear()
}
inner class TagHolder(itemView: View) : RecyclerView.ViewHolder(itemView) {
var tvAlias: TextView = itemView.findViewById(R.id.txtAlias)
var tvDate: TextView = itemView.findViewById(R.id.txtDate)
var ivType: ImageView = itemView.findViewById(R.id.ivType)
}
}
最佳答案
虽然我不确定 fragment 为何以这种方式运行——可能需要查看适配器实现——但很少有事情引起我的注意:
在
但在onResume
中观察onPause
中未清除
即使适配器在 onPause
状态下被“清除”,观察者也不会。这意味着每次 fragment 进入后台然后进入前台时,该 fragment 都会创建一组新的观察者,而以前的观察者仍然活着并观察相同的 LiveData
,因此由于这些重复使得适配器更难预测观察员。
使用this
观察LiveData
LifeCycle
对象进入
DESTROYED 时,
LiveData
观察者被取消订阅状态。这意味着为了避免观察者被重复,观察者应该在 CREATED
状态之前订阅,即 onCreate
方法是 Activity
和 fragment
。如果您不喜欢这种行为,您需要传入一个不同的 LifeCycleOwner
对象。
1。使用 getViewLifecycleOwner()
在 fragment.onCreate
中观察是非常不切实际的,因为 fragment 的 View 甚至还没有准备好。所以 Android 有一个名为 getViewLifecycleOwner()
的便捷方法,它可以在 onDestroyView()
阶段发出 DESTROYED
信号,以便 Fragment
可以在onCreateView()
或onViewCreated()
中观察。你可以像这样使用它:
override fun onCreateView(...): View {
...
getTags()
...
return view
}
private fun getTags(){
tagsViewModel.getAllText().observe(getViewLifecycleOwner(),
Observer<List<Text>> { t -> adapter.setTags(t!!) })
tagsViewModel.getAllEmail().observe(getViewLifecycleOwner(),
Observer<List<Email>> { t -> adapter.setTags(t!!) })
tagsViewModel.getAllUrl().observe(getViewLifecycleOwner(),
Observer<List<Url>> { t -> adapter.setTags(t!!) })
tagsViewModel.getAllPhone().observe(getViewLifecycleOwner(),
Observer<List<Phone>> { t -> adapter.setTags(t!!) })
tagsViewModel.getAllLauncher().observe(getViewLifecycleOwner(),
Observer<List<Launcher>> { t -> adapter.setTags(t!!) })
}
2。使用自定义 LifeCycleOwner
实现自定义 LifeCycleOwner
。我觉得这个article可以提供帮助。
ViewPager
中的
Fragment
生命周期
依赖于ViewPager
的实现,Fragment
只是切换不会经历onPause
-> onResume
状态选项卡。此外,标签之间的距离也很重要。例如,如果您切换到紧挨着它的选项卡,Fragment
可能不会“暂停”。尝试记录 fragment 状态以查看切换选项卡如何影响其生命周期。
关于android - LiveData Room 进入后台时丢失数据,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/56003736/
如果我 mov, eax 12345 和之后的 mov var, eax (假设 var 是一个 32 位的 int 等..等等)并输出 var 稍后它会正确输出。 与 ax 相同。 mov ax,
我有这个代码: for($nrt=0; $nrt"; if($sidesIndexes[$nrt]==$nrt) { echo "am I in??? ".$sidesInde
我正在阅读The Go Programming Language的8.5章,并陷入一些代码。下面的代码列表。 func main() { naturals := make(chan int)
我写了一个 MySQL 查询用于将数据导出到文本文件。 查询运行成功,但结果与我的预期不符。 我想在列之间没有间距的结果。 select sample_export_record1_2013.
在普通的 Excel 窗口中,我可以打开 VBE 并通过触摸键序列插入一个新模块:ALT+F11、ALTim 全部不使用鼠标。有没有办法打开 VBE 并导航到 本工作手册 不使用鼠标的代码区域? 最佳
我正在使用 axios 发出 http 请求。在 .then() 内部,我正在使用另一个 axios 调用。最后,我有第三个 then(),它应该在第二个 then 之后运行,但实际上并没有这样做。
我需要在 cocos2d 项目中播放视频..我的问题是:如何将 MPMoviePlayerController 放入我的 View 中,如下所示:? UIView *theView = [[CCDir
我正在学习 Angular。以下代码有效: .controller('abc', function ($scope, $http) { $http.get("/Handlers/Authenticat
目前我正在使用 WPF 学习 C#。我的主要方法是尽我所能使用 MVVM 模式,但现在我有点困惑。 在我所有 View 的应用程序中,我有一个 View 模型: private DruckviewVi
关于将 G 邮件提取到 Google 电子表格,我该如何添加 IF 以按主题驳回特定电子邮件?例如:电子邮件回复(主题中带有“RE:”)。我不希望这些电子邮件出现在我的电子表格中。 我尝试过使用 LO
我正在尝试使用 Spotify API 并进入数组。 const App = () => { const [isLoading, setIsLoading] = useState(true);
我有一个 html 模板,并且有一个条件为 --> 的代码 --> window.jQuery || document.write(""+"");
我正在开发一个 Android 应用程序,该应用程序会暴力破解从 int 创建的 MD5 和。 暴力破解部分工作正常。 (我可以sysout最终值并且它是正确的。) 我在将输出值发送到警报对话框时遇到
我正在创建一个界面,用户可以通过该界面生成多系列折线图,并控制绘制哪些数据集。 要绘制哪些数据集由复选框控制。页面加载时,默认数据集以图表形式呈现,并且 $('input:checkbox.data-
我尝试将有向无环图绘制为力布局。 但是我注意到,尽管为每个组元素灌输了进入/退出条件,弹出的节点/链接并没有从 DOM 中删除它们自己。 相反,弹出的节点/链接在力布局中卡住;这意味着没有调用进入/退
这里是新手。我不知道它是怎么发生的,但我正在处理一个程序,当我去调试并进入时,黄色箭头走到了我代码的最后并跳过了整个 block 。有快速解决方法吗? 最佳答案 按 F11,或单击工具栏上的“Step
我的 Action 栏 sherlock 中有一个列表。我想在用户点击该列表时得到。我不想知道用户何时点击某个项目,我已经知道 (onNavigationItemSelected)。 在我的 onCr
** 你好 **我如何从 ci 中的 mysql 日期获取 eurodate 来工作......无法弄清楚 - 请帮忙 想要获取日期 YY-mm- dd -> dd-mm-yy提前致谢 最佳答案 $t
我有以下脚本: #!/bin/bash ls -1 | while read d do [[ -f "$d" ]] && continue echo $d cd $d done
TL;DR - 跳转到最后一段 背景 我正在执行一些数据驱动测试,并将日志文件用作测试输出之一。它的工作原理是这样的- 读取文件夹中的第一个文件 处理第一行并转换为测试 运行测试 执行验证 1 ...
我是一名优秀的程序员,十分优秀!