我正在Kotlin
中开发一个应用程序,但是当涉及到我的特定片段ChartsFragment
时遇到了一些问题。基本上发生的是,当单击导航抽屉项时,我将从数据库中查询数据并将其传递到名为ChartsFragment
的片段以进行显示。我正在使用Graph View在片段上绘制数据。现在,一切正常,只是出于某种原因,我必须单击导航图标两次以查看该片段以加载更新的数据。单击导航图标一次,它将不会更新数据。
这是一个全局变量,用于存储数据库查询中的数据:
private var completed = mutableListOf<String>()
这是我从数据库查询数据时的代码:
/**
* For firebase database
* Read user progress from db
*/
private fun getProgress() {
val uid = FirebaseAuth.getInstance().getCurrentUser()?.uid
val rootRef = FirebaseDatabase.getInstance().getReference()
val progressRef = rootRef.child("Users").child(uid!!).child("Progress")
val valueEventListener = object : ValueEventListener {
override fun onDataChange(dataSnapshot:DataSnapshot) {
for(dSnapshot : DataSnapshot in dataSnapshot.getChildren()) {
for(ds : DataSnapshot in dSnapshot.getChildren()) {
val key = ds.getKey() as String
completed.add(key)
Log.d(TAG2, key)
}
}
}
override fun onCancelled(@NonNull databaseError : DatabaseError) {
Log.d(TAG2, databaseError.getMessage())
}
}
progressRef.addListenerForSingleValueEvent(valueEventListener)
}
这是我单击导航抽屉图标Charts
时的代码:
R.id.nav_chart -> {
val currentUser = mAuth!!.currentUser
if (currentUser != null) {
getProgress()
var basics = 0
var loops = 0
var functions = 0
completed.forEach {
if (it == "Data Type" || it == "Operation" || it == "Condition" || it == "Null Safety") {
basics++
} else if (it == "For" || it == "Break" || it == "Repeat" || it == "When" || it == "While") {
loops++
} else if (it == "Call" || it == "Main" || it == "Recursion" || it == "Single Expression") {
functions++
}
}
completed.clear()
val bundle = Bundle()
bundle.putDouble("basics",basics.toDouble())
bundle.putDouble("loops",loops.toDouble())
bundle.putDouble("functions",functions.toDouble())
chartFragment(bundle)
} else {
Toast.makeText(applicationContext, R.string.sign_in_to_track, Toast.LENGTH_SHORT).show()
}
}
这是我加载片段时的代码:
/**
* For loading fragments
*/
fun chartFragment(bundle: Bundle){
this.setTitle(R.string.navigation_drawer_chart)
val fragmentManager = supportFragmentManager
val transaction = fragmentManager.beginTransaction()
val fragment = ChartFragment()
fragment.setArguments(bundle)
transaction.replace(R.id.frameLayout1, fragment)
transaction.commit()
}
这是我的片段的onCreateView
:
override fun onCreateView(inflater: LayoutInflater, container: ViewGroup?,
savedInstanceState: Bundle?): View? {
val view = inflater.inflate(R.layout.fragment_chart, container, false)
val graph = view?.findViewById(R.id.graph) as GraphView
val basics = arguments!!.getDouble("basics")
val loops = arguments!!.getDouble("loops")
val functions = arguments!!.getDouble("functions")
val series = LineGraphSeries(arrayOf(
DataPoint(0.0, 0.0),
DataPoint(1.0, basics),
DataPoint(2.0, loops),
DataPoint(3.0, functions),
DataPoint(4.0, 0.0)
))
series.setColor(Color.GREEN)
series.setDrawDataPoints(true)
series.setAnimated(true)
graph.addSeries(series)
return view
}
我在想这是一些缓存问题吗?有没有一种方法可以强制单击导航抽屉中的项目,作为一个临时解决方案?
更新:我认为
AsyncTasks
可能会解决此问题,但是我不知道如何实现AsyncTasks
任务。似乎最可能的原因是getProgress()
可能太慢,程序正在执行下一行而不是等待getProgress()
完成。更新:第一次单击后得到的
completed
的大小始终是上一次单击的大小(不是db中的更新数据),第二次单击时,我将得到当前大小(当前为db)。因此,如果我的数据库有0个项目,那么在第一次单击时,病态会得到0,这是正确的,然后,如果我在应用程序中完成了一项活动,则数据库将更新为1个项目。因此,如果我现在单击图表图标,我应该得到1号,但我得到的是0号。第二次单击,我得到的是1号。更新:因此,如果我接下来去完成应用程序中的另一项活动,则我的数据库现在应该有2个项目。因此,如果我再次单击图表图标,我应该获得2号尺寸,但仍不能停留在以前的1号尺寸上。第二次点击,我将获得2号尺寸。
答案 0 :(得分:1)
似乎最可能的原因是getProgress()可能太慢,并且程序正在执行下一行而不是等待getProgress()完成。
是的,程序正在执行getProgress()
之后的下一行,而已完成被更新。但实际上getProgress()
已经完成。它的工作只是触发数据库调用。调用本身是异步执行的(即在另一个线程上)。一旦执行onDataChange()
,主线程将有权访问结果。只有在当前运行的功能(处理图标单击的功能)完成后才能发生这种情况。
由于您的逻辑取决于数据库调用的结果,因此应将评估新数据的行移至onDataChange()
。