android特賣列表倒計(jì)時(shí)卡頓問題的解決方法
在Android的開發(fā)中,我們經(jīng)常遇見倒計(jì)時(shí)的操作,通常使用Timer和Handler共同操作來完成。當(dāng)然也可以使用Android系統(tǒng)控件CountDownTimer,這里我們封裝成一個(gè)控件,也方便大家的使用。
首先上一張效果圖吧:
說一下造成卡頓的原因,由于滑動(dòng)的時(shí)候,adapter的getView頻繁的創(chuàng)建和銷毀,就會(huì)出現(xiàn)卡頓和數(shù)據(jù)錯(cuò)位問題,那么我們每一個(gè)item的倒計(jì)時(shí)就需要單獨(dú)維護(hù),這里我用的Handler與timer及TimerTask結(jié)合的方法,我們知道TimerTask運(yùn)行在自己子線程,然后通過Timer的schedule()方法實(shí)現(xiàn)倒計(jì)時(shí)功能,最后通過Hander實(shí)現(xiàn)View的刷新,其核心代碼如下:
public class CountDownView extends LinearLayout { @BindView(R.id.tv_day) TextView tvDay; @BindView(R.id.tv_hour) TextView tvHour; @BindView(R.id.tv_minute) TextView tvMinute; @BindView(R.id.tv_second) TextView tvSecond; @BindView(R.id.day) TextView day; @BindView(R.id.hour) TextView hour; @BindView(R.id.minute) TextView minute; private Context context; private int viewBg;//倒計(jì)時(shí)的背景 private int cellBg;//每個(gè)倒計(jì)時(shí)的背景 private int cellTextColor;//文字顏色 private int textColor;//外部:等顏色 private int textSize = 14;//外部文字大小 private int cellTextSize = 12;//cell文字大小 private TimerTask timerTask = null; private Timer timer = new Timer(); private Handler handler = new Handler() { public void handleMessage(Message msg) { countDown(); } }; public CountDownView(Context context, AttributeSet attrs) { this(context, attrs, 0); this.context = context; } public CountDownView(Context context, AttributeSet attrs, int defStyleAttr) { super(context, attrs, defStyleAttr); this.context = context; initAttrs(attrs, defStyleAttr); initView(context); } private void initAttrs(AttributeSet attrs, int defStyle) { TypedArray typedArray = getContext().obtainStyledAttributes(attrs, R.styleable.CountDownView, defStyle,0); viewBg = typedArray.getColor(R.styleable.CountDownView_viewBg, Color.parseColor("#FFFFFF")); cellBg = typedArray.getColor(R.styleable.CountDownView_cellBg, Color.parseColor("#F4F4F4")); cellTextColor = typedArray.getColor(R.styleable.CountDownView_cellTextColor, Color.parseColor("#646464")); textColor = typedArray.getColor(R.styleable.CountDownView_TextColor, Color.parseColor("#B3B3B3")); textSize = (int) typedArray.getDimension(R.styleable.CountDownView_TextSize, UIUtils.dp2px(getContext(), 14)); cellTextSize = (int) typedArray.getDimension(R.styleable.CountDownView_cellTextSize, UIUtils.dp2px(getContext(), 12)); typedArray.recycle(); } private void initView(Context context) { LayoutInflater inflater = (LayoutInflater) context .getSystemService(Context.LAYOUT_INFLATER_SERVICE); View view = inflater.inflate(R.layout.layout_countdown_layout, this); ButterKnife.bind(view); initProperty(); } private void initProperty() { tvDay.setBackgroundColor(cellBg); tvHour.setBackgroundColor(cellBg); tvMinute.setBackgroundColor(cellBg); tvSecond.setBackgroundColor(cellBg); tvDay.setTextColor(cellTextColor); tvHour.setTextColor(cellTextColor); tvMinute.setTextColor(cellTextColor); tvSecond.setTextColor(cellTextColor); day.setTextColor(textColor); hour.setTextColor(textColor); minute.setTextColor(textColor); } public void setLeftTime(long leftTime) { if (leftTime <= 0) return; long time = leftTime / 1000; long day = time / (3600 * 24); long hours = (time - day * 3600 * 24) / 3600; long minutes = (time - day * 3600 * 24 - hours * 3600) / 60; long seconds = time - day * 3600 * 24 - hours * 3600 - minutes * 60; setTextTime(time); } public void start() { if (timerTask == null) { timerTask = new TimerTask() { @Override public void run() { handler.sendEmptyMessage(0); } }; timer.schedule(timerTask, 1000, 1000); // timer.schedule(new TimerTask() { // @Override // public void run() { // handler.sendEmptyMessage(0); // } // }, 0, 1000); } } public void stop() { if (timer != null) { timer.cancel(); timer = null; } } //保證天,時(shí),分,秒都兩位顯示,不足的補(bǔ)0 private void setTextTime(long time) { String[] s = TimeUtils.formatTimer(time); tvDay.setText(s[0]); tvHour.setText(s[1]); tvMinute.setText(s[2]); tvSecond.setText(s[3]); } private void countDown() { if (isCarry4Unit(tvSecond)) { if (isCarry4Unit(tvMinute)) { if (isCarry4Unit(tvHour)) { if (isCarry4Unit(tvDay)) { stop(); } } } } } private boolean isCarry4Unit(TextView tv) { int time = Integer.valueOf(tv.getText().toString()); time = time - 1; if (time < 0) { time = 59; tv.setText(time + ""); return true; } else if (time < 10) { tv.setText("0" + time); return false; } else { tv.setText(time + ""); return false; } } }
另一種寫法:
public class CountDownTimerView extends LinearLayout { private TextView hourView, minuteView, secondView; private LimitTimer mTimer; private Handler handler; public static final int START_LIMIT_TIME_MSG = 0x0111; public static final int END_LIMIT_TIME_MSG = 0x0112; private long endTime, leftTime; private LimitTimeListener listener=null; public CountDownTimerView(Context context) { super(context); initView(context); } public CountDownTimerView(Context context, AttributeSet attrs) { super(context, attrs); initView(context); } private void initView(Context context) { setOrientation(HORIZONTAL); setGravity(Gravity.CENTER_VERTICAL); inflate(context, R.layout.layout_countdown_layout, this); hourView = (TextView) findViewById(R.id.tv_hour); minuteView = (TextView) findViewById(R.id.tv_minute); secondView = (TextView) findViewById(R.id.tv_second); } public long getLeftTime() { return leftTime; } //設(shè)置剩余時(shí)間 public void initLeftTime(long endTime) { endTime=endTime*1000; if (null != mTimer) { mTimer.cancel(); mTimer = null; } this.endTime = endTime; mTimer = new LimitTimer(endTime, 1000); mTimer.start(); if (handler != null) { handler.sendEmptyMessage(START_LIMIT_TIME_MSG); } } /** * 如果該控件使用在碎片中,返回時(shí),則最好還是要stop */ public void stopTimeCount() { if (null != mTimer) { mTimer.cancel(); mTimer = null; } } public Handler getHandler() { return handler; } public void setHandler(Handler handler) { this.handler = handler; } private class LimitTimer extends CountDownTimer { public LimitTimer(long millisInFuture, long countDownInterval) { super(0 != leftTime ? leftTime : endTime, countDownInterval); } @Override public void onTick(long millisUntilFinished) { leftTime = millisUntilFinished; long totalSecond = millisUntilFinished / 1000; int second = (int) (totalSecond % 60); int minute = (int) ((totalSecond / 60) % 60); int hour = (int) ((totalSecond / 3600) % 24); int day = (int) (totalSecond / (3600 * 24)); formatView(hourView, hour); formatView(minuteView, minute); formatView(secondView, second); } @Override public void onFinish() { String zero = "00"; hourView.setText(zero); minuteView.setText(zero); secondView.setText(zero); if (null != handler) { handler.sendEmptyMessage(END_LIMIT_TIME_MSG); } if (listener!=null){ listener.onTimeOver(true); } } private void formatView(TextView view, int time) { DecimalFormat df = new DecimalFormat("#00"); view.setText(df.format(time)); } } //倒計(jì)時(shí)結(jié)束監(jiān)聽 public void setOnLimitTimeListener(LimitTimeListener listener) { this.listener = listener; } public interface LimitTimeListener { void onTimeOver(boolean flag); } @Override protected void onDetachedFromWindow() { super.onDetachedFromWindow(); if (handler != null) { handler.removeMessages(START_LIMIT_TIME_MSG); } } }
涉及到的布局文件
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="wrap_content" android:layout_height="wrap_content" android:orientation="horizontal"> <TextView android:id="@+id/tv_hour" style="@style/style_countdown" /> <TextView style="@style/style_wrap_content" android:layout_marginLeft="3dp" android:layout_marginRight="3dp" android:layout_gravity="center_vertical" android:text="時(shí)" android:textColor="@color/color_646464" /> <TextView android:id="@+id/tv_minute" style="@style/style_countdown" /> <TextView style="@style/style_wrap_content" android:layout_marginLeft="3dp" android:layout_marginRight="3dp" android:layout_gravity="center_vertical" android:text="分" android:textColor="@color/color_646464" /> <TextView android:id="@+id/tv_second" style="@style/style_countdown" /> <TextView style="@style/style_wrap_content" android:layout_marginLeft="3dp" android:layout_marginRight="3dp" android:layout_gravity="center_vertical" android:text="秒" android:textColor="@color/color_646464" /> </LinearLayout>
附上源碼地址:點(diǎn)擊打開鏈接
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- Android倒計(jì)時(shí)控件 Splash界面5秒自動(dòng)跳轉(zhuǎn)
- Android倒計(jì)時(shí)的開始與停止 剩余時(shí)分秒的展示
- Android仿Keep運(yùn)動(dòng)休息倒計(jì)時(shí)圓形控件
- android實(shí)現(xiàn)條目倒計(jì)時(shí)功能
- Android利用RecyclerView實(shí)現(xiàn)列表倒計(jì)時(shí)效果
- android自定義倒計(jì)時(shí)控件示例
- android實(shí)現(xiàn)倒計(jì)時(shí)功能代碼
- Android實(shí)現(xiàn)計(jì)時(shí)與倒計(jì)時(shí)的常用方法小結(jié)
- Android實(shí)現(xiàn)加載廣告圖片和倒計(jì)時(shí)的開屏布局
- Android自定義圓環(huán)倒計(jì)時(shí)控件
相關(guān)文章
Android中通過view方式獲取當(dāng)前Activity的屏幕截圖實(shí)現(xiàn)方法
這篇文章主要介紹了Android中通過view方式獲取當(dāng)前Activity的屏幕截圖實(shí)現(xiàn)方法,本文方法相對(duì)簡(jiǎn)單,容易理解,需要的朋友可以參考下2014-09-09Android轉(zhuǎn)場(chǎng)動(dòng)畫深入分析探究
對(duì)于一個(gè)動(dòng)畫而言,它是由多個(gè)分鏡頭組成的,而轉(zhuǎn)場(chǎng)就是分鏡之間銜接方式。轉(zhuǎn)場(chǎng)的主要目的,就是為了讓鏡頭與鏡頭之間過渡的更加自然,讓動(dòng)畫更加連貫,例如兩個(gè)頁(yè)面切換之間的銜接動(dòng)畫2022-10-10僅4行代碼實(shí)現(xiàn)Android快速文件下載
僅4行代碼實(shí)現(xiàn)Android快速文件下載,這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)快速文件下載功能的方法,感興趣的小伙伴們可以參考一下2016-07-07android studio 清單配置文件androidmainfest.xml詳細(xì)解讀
AndroidManifest官方解釋是應(yīng)用清單,每個(gè)應(yīng)用的根目錄中都必須包含一個(gè),并且文件名必須一模一樣,這個(gè)文件中包含了APP的配置信息,系統(tǒng)需要根據(jù)里面的內(nèi)容運(yùn)行APP的代碼,顯示界面,這篇文章介紹了android studio 清單配置文件androidmainfest.xml解讀,需要的朋友可以參考下2024-04-04Android 中Popwindow彈出菜單的兩種方法實(shí)例
這篇文章主要介紹了Android 中Popwindow彈出菜單的兩種方法實(shí)例的相關(guān)資料,這里提供了兩種實(shí)現(xiàn)的方法,并附有實(shí)例代碼,需要的朋友可以參考下2017-03-03Android實(shí)現(xiàn)滑塊拼圖驗(yàn)證碼功能
這篇文章主要介紹了Android實(shí)現(xiàn)滑塊拼圖驗(yàn)證碼功能,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-02-02解決android有的手機(jī)拍照后上傳圖片被旋轉(zhuǎn)的問題
這篇文章主要介紹了解決android有的手機(jī)拍照后上傳圖片被旋轉(zhuǎn)的問題的相關(guān)資料,需要的朋友可以參考下2016-09-09