缘故原由
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()
- }
- }
复制代码 免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。 |