前言

转眼间距离上次写博客已是过了一个年轮,期间发生了不少事;经历了离职、找工作,新公司的第一版项目上线。现在总算是有时间可以将遇到的问题梳理下了,后期有时间也会分享更多的东西~~

场景

今天分享的问题是当在列表里面显示倒计时,这时候滑动列表会出现时间显示不正常的问题。首先关于倒计时我们需要注意的问题有以下几方面:

在RecyclerView中ViewHolder的复用导致的时间乱跳的问题。

滑动列表时倒计时会重置的问题。

在退出页面后定时器的资源释放问题,这里我使用的是用系统自带的CountDownTimer

ps:这里我们讨论的是对倒计时要求不是很严格的场景,对于用户手动修改系统时间这种操作没法预计;对于淘宝秒杀这种业务场景建议是实时不断请求后台拿取正确时间,对应的接口尽量设计简单,响应数据更快。

接下来通过代码具体了解:

代码

// 适配器
public class MyAdapter extends RecyclerView.Adapter<MyAdapter.ViewHolder> {
  //服务器返回数据
  private List<TimeBean> mDatas;
  //退出activity时关闭所有定时器,避免造成资源泄漏。
  private SparseArray<CountDownTimer> countDownMap;

  //记录每次刷新时的时间
  private long tempTime;

  public MyAdapter(Context context, List<TimeBean> datas) {
    mDatas = datas;
    countDownMap = new SparseArray<>();
  }

  @Override
  public ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
    View view = LayoutInflater.from(parent.getContext()).inflate(R.layout.list_item_common, parent, false);
    return new ViewHolder(view);
  }

  @Override
  public void onBindViewHolder(final ViewHolder holder, int position) {
    final TimeBean data = mDatas.get(position);
    //记录时间点
    long timeStamp = System.currentTimeMillis() - tempTime;

    long time = data.getLeftTime() - timeStamp;

    //将前一个缓存清除
    if (holder.countDownTimer != null) {
      holder.countDownTimer.cancel();
    }
    if (time > 0) { //判断倒计时是否结束
      holder.countDownTimer = new CountDownTimer(time, 1000) {
        public void onTick(long millisUntilFinished) {
          holder.timeTv.setText(getMinuteSecond(millisUntilFinished));
        }
        public void onFinish() {
          //倒计时结束
          holder.timeTv.setText("00:00");
        }
      }.start();

      countDownMap.put(holder.timeTv.hashCode(), holder.countDownTimer);
    } else {
      holder.timeTv.setText("00:00");
    }
  }

  @Override
  public int getItemCount() {
    if (mDatas != null && !mDatas.isEmpty()) {
      return mDatas.size();
    }
    return 0;
  }

  public class ViewHolder extends RecyclerView.ViewHolder {
    public TextView timeTv;
    public CountDownTimer countDownTimer;

    public ViewHolder(View itemView) {
      super(itemView);
      timeTv = (TextView) itemView.findViewById(R.id.tv_time);
    }
  }

  public void setGetTime(long tempTime) {
    this.tempTime = tempTime;
  }

  /**
   * 将毫秒数换算成 00:00 形式
   */
  public static String getMinuteSecond(long time) {
    int ss = 1000;
    int mi = ss * 60;
    int hh = mi * 60;
    int dd = hh * 24;

    long day = time / dd;
    long hour = (time - day * dd) / hh;
    long minute = (time - day * dd - hour * hh) / mi;
    long second = (time - day * dd - hour * hh - minute * mi) / ss;

    String strMinute = minute < 10 ? "0"   minute : ""   minute;
    String strSecond = second < 10 ? "0"   second : ""   second;
    return strMinute   ":"   strSecond;
  }

  /**
   * 清空资源
   */
  public void cancelAllTimers() {
    if (countDownMap == null) {
      return;
    }
    for (int i = 0,length = countDownMap.size(); i < length; i  ) {
      CountDownTimer cdt = countDownMap.get(countDownMap.keyAt(i));
      if (cdt != null) {
        cdt.cancel();
      }
    }
  }
}

以上算是整个问题的核心代码了;其中SparseArray<CountDownTimer> 用来保存列表里面的定时器,用于退出页面时回收定时器。SparseArray是安卓特有的数据结构,建议多使用;data.getLeftTime() 是服务器返回的需要倒计时的时间,毫秒为单位。

问题一:ViewHolder的复用导致的数据错乱

if (holder.countDownTimer != null) {
    holder.countDownTimer.cancel();
  }

每次设置倒计时之前重置下倒计时即可解决。

问题二:滑动列表时倒计时会重置的问题

这个问题是由于解决问题一而导致的,因为列表滑动时离开屏幕的会被复用,这个时候我们会重新设置定时器,之前我是在倒计时里面记录倒计时剩余的时间然后重新设值,但是还是会有问题;这里借用了系统时间来解决,也就是tempTime 这个值。

首先在服务器请求成功后回调里面设置这个值,如:

  private MyAdapter adapter;

  @Override
  public void onHttpRequestSuccess(String url, HttpContext httpContext)   {
    if (服务器返回数据) {
      adapter.setGetTime(System.currentTimeMillis());
  }

相当于每次做刷新操作时获取的都是系统当时的时间戳。

然后在adapter里面计算

long timeStamp = System.currentTimeMillis() - tempTime;

long time = data.getLeftTime() - timeStamp;

其中tempTime就是我们保存的系统当前时间戳,然后每次滑动列表时都会调用onBindViewHolder,所以timeStamp就是记录的距离上次刷新经过了多少秒,然后用服务器返回的需要倒计时的时间减去经过的秒数就是还剩下的倒计时秒数。最后给定时器设置上就好了。

问题三:资源的释放

在当前的activity中调用以下方法。

@Override
protected void onDestroy() {
  super.onDestroy();
  if (adapter != null) {
    adapter.cancelAllTimers();
  }
}

好了,今天的分享就到这了,因为代码比较简单,布局都是一个Textview,所以没有贴出来,需要代码的可以留言~~

补充知识:Android 自定义倒计时,支持listview多item一起倒计时

项目中用到的两种倒计时,一种是用CountDownTimer,但是这种方式在listview中就不是那么好用了,当listview 里面多个item都需要倒计时,就不可以用这种了,我这里想到用Thread 加handler来一起实现。如果大家还有好的倒计时方法,可以留言一起讨论哦,由于代码都是在项目中的,我就截取几段代码。

第一种 CountDownTimer:

主要自定义一个类继承CountDownTimer,在启动的时候调用start(),倒计时完毕调用canel()方法。

time = new TimeCount(remainingTime, 1000);//构造CountDownTimer对象
time.start();//开始计时

class TimeCount extends CountDownTimer {
    public TimeCount(long millisInFuture, long countDownInterval) {
      super(millisInFuture, countDownInterval);
    }

    @Override
    public void onFinish() {//计时完毕时触发
      if (isDead) {
        remainingTime = 90000;
        ColorStateList colorStateList = getResources().getColorStateList(R.color.button_send_code_text2_selector);
        getCode.setTextColor(colorStateList);
        getCode.setText(R.string.register_tip7);
        getCode.setEnabled(true);
      }
    }

    @Override
    public void onTick(long millisUntilFinished) {//计时过程显示
      if (isDead) {
        getCode.setEnabled(false);
        getCode.setTextColor(getResources().getColor(R.color.grey5));
        remainingTime = millisUntilFinished;
        getCode.setText(millisUntilFinished / 1000   "秒后重发");
      }
    }
  }

第二种 Thread 加handler

创建一个新的线程,每秒中减一次时间,然后在handler中每秒中刷新一次界面就可以看到倒计时的效果。

 private Thread thread;

  //条目倒计时
  public void start() {
    thread = new Thread() {
      public void run() {
        while (true) {
          try {
            if (list != null) {
              for (InvestProjectVo item : list) {

                if(item.remainOpenTime == 0){
                  item.status = 0;
                }

                if(item.remainOpenTime > 0){
                  item.remainOpenTime = item.remainOpenTime - 1;
                }
              }
            }
            sleep(1000);
          } catch (InterruptedException e) {
            e.printStackTrace();
          }
        }
      }
    };
    thread.start();
  }

在adapter的getview()方法中,判断倒计时时间是否大于0,如果大于零可以继续显示倒计时时间

          if (vo.remainOpenTime != 0 && vo.remainOpenTime > 0) {

            viewCache.showProjectFullIcon.setVisibility(View.GONE);
            viewCache.projectProgress.setVisibility(View.GONE);
            viewCache.showTimer.setVisibility(View.VISIBLE);

            long tempTime = vo.remainOpenTime;
            long day = tempTime / 60 / 60 / 24;
            long hours = (tempTime - day * 24 * 60 * 60) / 60 / 60;
            long minutes = (tempTime - day * 24 * 60 * 60 - hours * 60 * 60) / 60;
            long seconds = (tempTime - day * 24 * 60 * 60 - hours * 60 * 60 - minutes * 60);
            if (minutes > 0) {
              viewCache.timer.setText(minutes   "分"   seconds   "秒");
            } else {
              viewCache.timer.setText(seconds   "秒");
            }
          }else{
            viewCache.showProjectFullIcon.setVisibility(View.GONE);
            viewCache.projectProgress.setVisibility(View.VISIBLE);
            viewCache.showTimer.setVisibility(View.GONE);
          }

在handler中每秒钟刷新一次界面

mHandler.sendEmptyMessageDelayed(2586221,1000);

adapter.notifyDataSetChanged();
//每隔1毫秒更新一次界面,如果只需要精确到秒的倒计时此处改成1000即可
mHandler.sendEmptyMessageDelayed(2586221,1000);

以上这篇解决Android-RecyclerView列表倒计时错乱问题就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持Devmax。

解决Android-RecyclerView列表倒计时错乱问题的更多相关文章

  1. html5 canvas合成海报所遇问题及解决方案总结

    这篇文章主要介绍了html5 canvas合成海报所遇问题及解决方案总结,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧

  2. Html5 video标签视频的最佳实践

    这篇文章主要介绍了Html5 video标签视频的最佳实践,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

  3. HTML5在微信内置浏览器下右上角菜单的调整字体导致页面显示错乱的问题

    HTML5在微信内置浏览器下,在右上角菜单的调整字体导致页面显示错乱的问题,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧

  4. ios – containerURLForSecurityApplicationGroupIdentifier:在iPhone和Watch模拟器上给出不同的结果

    我使用默认的XCode模板创建了一个WatchKit应用程序.我向iOSTarget,WatchkitAppTarget和WatchkitAppExtensionTarget添加了应用程序组权利.(这是应用程序组名称:group.com.lombax.fiveminutes)然后,我尝试使用iOSApp和WatchKitExtension访问共享文件夹URL:延期:iOS应用:但是,测试NSURL

  5. Ionic – Splash Screen适用于iOS,但不适用于Android

    我有一个离子应用程序,其中使用CLI命令离子资源生成的启动画面和图标iOS版本与正在渲染的启动画面完美配合,但在Android版本中,只有在加载应用程序时才会显示白屏.我检查了config.xml文件,所有路径看起来都是正确的,生成的图像出现在相应的文件夹中.(我使用了splash.psd模板来生成它们.我错过了什么?这是config.xml文件供参考,我觉得我在这里做错了–解决方法在config.xml中添加以下键:它对我有用!

  6. ios – 无法启动iPhone模拟器

    /Library/Developer/CoreSimulator/Devices/530A44CB-5978-4926-9E91-E9DBD5BFB105/data/Containers/Bundle/Application/07612A5C-659D-4C04-ACD3-D211D2830E17/ProductName.app/ProductName然后,如果您在Xcode构建设置中选择标准体系结构并再次构建和运行,则会产生以下结果:dyld:lazysymbolbindingFailed:Symbol

  7. Xamarin iOS图像在Grid内部重叠

    heyo,所以在Xamarin我有一个使用并在其中包含一对,所有这些都包含在内.这在Xamarin.Android中看起来完全没问题,但是在Xamarin.iOS中,图像与标签重叠.我不确定它的区别是什么–为什么它在Xamarin.Android中看起来不错但在iOS中它的全部都不稳定?

  8. 在iOS上向后播放HTML5视频

    我试图在iPad上反向播放HTML5视频.HTML5元素包括一个名为playbackRate的属性,它允许以更快或更慢的速率或相反的方式播放视频.根据Apple’sdocumentation,iOS不支持此属性.通过每秒多次设置currentTime属性,可以反复播放,而无需使用playbackRate.这种方法适用于桌面Safari,但似乎在iOS设备上的搜索限制为每秒1次更新–在我的情况下太慢了.有没有办法在iOS设备上向后播放HTML5视频?解决方法iOS6Safari现在支持playbackRat

  9. 使用 Swift 语言编写 Android 应用入门

    Swift标准库可以编译安卓armv7的内核,这使得可以在安卓移动设备上执行Swift语句代码。做梦,虽然Swift编译器可以胜任在安卓设备上编译Swift代码并运行。这需要的不仅仅是用Swift标准库编写一个APP,更多的是你需要一些框架来搭建你的应用用户界面,以上这些Swift标准库不能提供。简单来说,构建在安卓设备上使用的Swiftstdlib需要libiconv和libicu。通过命令行执行以下命令:gitclonegit@github.com:SwiftAndroid/libiconv-libi

  10. Android – 调用GONE然后VISIBLE使视图显示在错误的位置

    我有两个视图,A和B,视图A在视图B上方.当我以编程方式将视图A设置为GONE时,它将消失,并且它正下方的视图将转到视图A的位置.但是,当我再次将相同的视图设置为VISIBLE时,它会在视图B上显示.我不希望这样.我希望视图B回到原来的位置,这是我认为会发生的事情.我怎样才能做到这一点?编辑–代码}这里是XML:解决方法您可以尝试将两个视图放在RelativeLayout中并相对于彼此设置它们的位置.

随机推荐

  1. Flutter 网络请求框架封装详解

    这篇文章主要介绍了Flutter 网络请求框架封装详解,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧

  2. Android单选按钮RadioButton的使用详解

    今天小编就为大家分享一篇关于Android单选按钮RadioButton的使用详解,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧

  3. 解决android studio 打包发现generate signed apk 消失不见问题

    这篇文章主要介绍了解决android studio 打包发现generate signed apk 消失不见问题,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧

  4. Android 实现自定义圆形listview功能的实例代码

    这篇文章主要介绍了Android 实现自定义圆形listview功能的实例代码,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下

  5. 详解Android studio 动态fragment的用法

    这篇文章主要介绍了Android studio 动态fragment的用法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下

  6. Android用RecyclerView实现图标拖拽排序以及增删管理

    这篇文章主要介绍了Android用RecyclerView实现图标拖拽排序以及增删管理的方法,帮助大家更好的理解和学习使用Android,感兴趣的朋友可以了解下

  7. Android notifyDataSetChanged() 动态更新ListView案例详解

    这篇文章主要介绍了Android notifyDataSetChanged() 动态更新ListView案例详解,本篇文章通过简要的案例,讲解了该项技术的了解与使用,以下就是详细内容,需要的朋友可以参考下

  8. Android自定义View实现弹幕效果

    这篇文章主要为大家详细介绍了Android自定义View实现弹幕效果,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

  9. Android自定义View实现跟随手指移动

    这篇文章主要为大家详细介绍了Android自定义View实现跟随手指移动,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

  10. Android实现多点触摸操作

    这篇文章主要介绍了Android实现多点触摸操作,实现图片的放大、缩小和旋转等处理,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

返回
顶部