原因
recyceleview 当页面划出屏幕外后,默认会有两条进入缓存区,这些item的结构会被保存,数据被清除,方便其他新进入屏幕的数据复用item,超过两条外的item会进入缓存池被完全销毁重用。
如果我们的页面上有editText 或者倒计时CountDownTimer在不停的给item赋值的时候就会出现数据错乱的问题,解决的方式就是我们需要手动在adapter中存储数据,然后手动复制或者停止倒计时。
具体实现(以倒计时为例)
首先在adapter中定义一个map取存储你每次新建的倒计时对象,然后需要在你对应的item的实体类中新增一个leftTime字段用来存储你倒计时的最新时间(因为页面刷新时并不会重新获取数据),然后在倒计时开始和item回收时取消item原有的倒计时。
//在adapter中定义 maps的key建议使用position最为唯一标识
private val maps = HashMap<String, CountDownTimer>()
//在onBindViewHolder中开启倒计时
it.entity.check_outdate_time?.let {
maps[item.entity.id]?.cancel()
var timeLeft = 0L
item.entity.leftTime?.let { timeLeft = it } ?: run {
timeLeft = item.entity.check_outdate_time!!.toLong() * 1000
}
val timer = object : CountDownTimer(timeLeft, 1) {
override fun onTick(millisUntilFinished: Long) {
item.entity.leftTime = millisUntilFinished
//开始倒计时
val d = (millisUntilFinished / 1000) / 3600 / 24
val h = (millisUntilFinished / 1000) / 3600 % 24
val m = (millisUntilFinished / 1000) % 3600 / 60
val s = (millisUntilFinished / 1000) % 3600 % 60
// LogUtils.d("timer", "dsafdsafdsaafdsfdsafdsa")
holder.setText(
R.id.tv_counterTime,
"${d.toInt()}天${getNum(h.toInt())}时${
getNum(
m.toInt()
)
}分${
getNum(
s.toInt()
)
}秒"
)
}
override fun onFinish() {
maps.remove(item.entity.id)
holder.setText(
R.id.tv_counterTime, "报名截止"
)
}
}
maps.put(item.entity.id, timer)
timer.start()
}
//回收时
override fun onViewRecycled(holder: RecyclerView.ViewHolder) {
super.onViewRecycled(holder)
val position = holder.position
if (position != RecyclerView.NO_POSITION) {
if (items.size > position) {
val item = items[position]
maps[item.entity.id]?.cancel() // 在回收时取消倒计时
}
}
}
//在页面销毁或recycleview重置时时关闭map中的倒计时
fun cancelTimer() {
maps.values.forEach {
it.cancel()
}
}