Android实现抢购倒计时功能
龙旋
共 10296字,需浏览 21分钟
· 2021-03-25
一、效果图
二、思路
算多少秒,秒数取余60,(满足分后剩下的秒数)
算多少分,秒数除60,再取余60 (总分数满足小时后剩下的分数)
算多少时,秒数除60,除60,再取余24 (总小时满足天后剩下的小时)
算多少天,秒数除60,除60,除24 等到的整数就是天数
三、实现步骤:
我们这里的时间格式为后台返回,格式为:
2021-12-24 00:00:00
1、时间转换的工具类
//将年-月-天 时:分:秒转化为毫秒格式
public static long residueTimeout(String endDate, String newDate) throws ParseException {
SimpleDateFormat df = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
Date d1 = df.parse(endDate);
Date d2 = df.parse(newDate);
long diff = d1.getTime() - d2.getTime();
return diff;
}
/*
* 将毫秒转换成时间戳
*/
public static String stampToDate(Long s) {
String res;
SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
Date date = new Date(s);
res = simpleDateFormat.format(date);
return res;
}
2、时间倒计时工具类
package com.sjl.keeplive.huawei;
import android.os.CountDownTimer;
/**
* 倒计时工具类
*/
public class CountDownTimerUtils {
/**
* 倒计时结束的回调接口
*/
public interface FinishDelegate {
void onFinish();
}
/**
* 定期回调的接口
*/
public interface TickDelegate {
void onTick(long pMillisUntilFinished);
}
private final static long ONE_SECOND = 1000;
/**
* 总倒计时时间
*/
private long mMillisInFuture = 0;
/**
* 定期回调的时间 必须大于0 否则会出现ANR
*/
private long mCountDownInterval;
/**
* 倒计时结束的回调
*/
private FinishDelegate mFinishDelegate;
/**
* 定期回调
*/
private TickDelegate mTickDelegate;
private MyCountDownTimer mCountDownTimer;
/**
* 获取 CountDownTimerUtils
*
* @return CountDownTimerUtils
*/
public static CountDownTimerUtils getCountDownTimer() {
return new CountDownTimerUtils();
}
/**
* 设置定期回调的时间 调用{@link #setTickDelegate(TickDelegate)}
*
* @param pCountDownInterval 定期回调的时间 必须大于0
* @return CountDownTimerUtils
*/
public CountDownTimerUtils setCountDownInterval(long pCountDownInterval) {
this.mCountDownInterval = pCountDownInterval;
return this;
}
/**
* 设置倒计时结束的回调
*
* @param pFinishDelegate 倒计时结束的回调接口
* @return CountDownTimerUtils
*/
public CountDownTimerUtils setFinishDelegate(FinishDelegate pFinishDelegate) {
this.mFinishDelegate = pFinishDelegate;
return this;
}
/**
* 设置总倒计时时间
*
* @param pMillisInFuture 总倒计时时间
* @return CountDownTimerUtils
*/
public CountDownTimerUtils setMillisInFuture(long pMillisInFuture) {
this.mMillisInFuture = pMillisInFuture;
return this;
}
/**
* 设置定期回调
*
* @param pTickDelegate 定期回调接口
* @return CountDownTimerUtils
*/
public CountDownTimerUtils setTickDelegate(TickDelegate pTickDelegate) {
this.mTickDelegate = pTickDelegate;
return this;
}
public void create() {
if (mCountDownTimer != null) {
mCountDownTimer.cancel();
mCountDownTimer = null;
}
if (mCountDownInterval <= 0) {
mCountDownInterval = mMillisInFuture + ONE_SECOND;
}
mCountDownTimer = new MyCountDownTimer(mMillisInFuture, mCountDownInterval);
mCountDownTimer.setTickDelegate(mTickDelegate);
mCountDownTimer.setFinishDelegate(mFinishDelegate);
}
/**
* 开始倒计时
*/
public void start() {
if (mCountDownTimer == null) {
create();
}
mCountDownTimer.start();
}
/**
* 取消倒计时
*/
public void cancel() {
if (mCountDownTimer != null) {
mCountDownTimer.cancel();
}
}
private static class MyCountDownTimer extends CountDownTimer {
private FinishDelegate mFinishDelegate;
private TickDelegate mTickDelegate;
/**
* @param millisInFuture The number of millis in the future from the call
* to {@link #start()} until the countdown is done and {@link #onFinish()}
* is called.
* @param countDownInterval The interval along the way to receive
* {@link #onTick(long)} callbacks.
*/
public MyCountDownTimer(long millisInFuture, long countDownInterval) {
super(millisInFuture, countDownInterval);
}
@Override
public void onTick(long millisUntilFinished) {
if (mTickDelegate != null) {
mTickDelegate.onTick(millisUntilFinished);
}
}
@Override
public void onFinish() {
if (mFinishDelegate != null) {
mFinishDelegate.onFinish();
}
}
void setFinishDelegate(FinishDelegate pFinishDelegate) {
this.mFinishDelegate = pFinishDelegate;
}
void setTickDelegate(TickDelegate pTickDelegate) {
this.mTickDelegate = pTickDelegate;
}
}
}
3、布局文件
<LinearLayout
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center"
android:orientation="horizontal">
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="抢购倒计时:" />
<TextView
android:id="@+id/text_day"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:background="#ff0000"
android:padding="5dp"
android:text="00" />
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text=" 天 " />
<TextView
android:id="@+id/text_hour"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:background="#ff0000"
android:padding="5dp"
android:text="00" />
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text=" 时 " />
<TextView
android:id="@+id/text_minute"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:background="#ff0000"
android:padding="5dp"
android:text="00" />
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text=" 分 " />
<TextView
android:id="@+id/text_second"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:background="#ff0000"
android:padding="5dp"
android:text="00" />
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text=" 秒 " />
</LinearLayout>
4、倒计时显示处理
public static void liveDescCountTime(long ms, TextView tvDays, TextView tvHour, TextView tvMinutes, TextView tvSeconds) {
long totalSeconds = ms / 1000;
long seconds = totalSeconds % 60;
long minutes = totalSeconds / 60 % 60;
long hours = totalSeconds / 60 / 60 % 24;
long days = totalSeconds / 60 / 60 / 24;
String dayStr = "";
if (days > 0) {
if (days > 9) {
dayStr += days + "";
} else if (days > 0) {
dayStr += "0" + days + "";
} else {
dayStr += "00";
}
} else {
dayStr = "00";
}
tvDays.setText(dayStr);
String hourStr = "";
if (hours > 0) {
if (hours > 9) {
hourStr += hours + "";
} else if (hours > 0) {
hourStr += "0" + hours + "";
} else {
hourStr += "00";
}
} else {
hourStr = "00";
}
tvHour.setText(hourStr);
String minutesStr = "";
if (minutes > 0) {
if (minutes > 9) {
minutesStr += minutes + "";
} else if (minutes > 0) {
minutesStr += "0" + minutes + "";
} else {
minutesStr += "00";
}
} else {
minutesStr = "00";
}
tvMinutes.setText(minutesStr);
String secondStr = "";
if (minutes > 0) {
if (seconds > 9) {
secondStr += seconds;
} else if (seconds > 0) {
secondStr += "0" + seconds;
} else {
secondStr += "00";
}
} else {
secondStr = "00";
}
tvSeconds.setText(secondStr);
}
5、开始倒计时
final TextView text_day = findViewById(R.id.text_day);
final TextView text_hour = findViewById(R.id.text_hour);
final TextView text_minute = findViewById(R.id.text_minute);
final TextView text_second = findViewById(R.id.text_second);
long residueTime = 0;
//获取当前时间
String stampToDate = stampToDate(System.currentTimeMillis());
try {
//2021-12-24 00:00:00为模拟倒计时间数据
residueTime = residueTimeout("2021-12-24 00:00:00", stampToDate);
} catch (ParseException e) {
e.printStackTrace();
}
//倒计时
CountDownTimerUtils.getCountDownTimer()
.setMillisInFuture(residueTime)
.setCountDownInterval(1000)
.setTickDelegate(new CountDownTimerUtils.TickDelegate() {
@Override
public void onTick(long pMillisUntilFinished) {
liveDescCountTime(pMillisUntilFinished, text_day, text_hour, text_minute, text_second);
}
})
.setFinishDelegate(new CountDownTimerUtils.FinishDelegate() {
@Override
public void onFinish() {
//倒计时完成后处理
}
}).start();
代码都已经放在上面啦
到这里就结束啦
评论
15种时间序列预测方法总结(包含多种方法代码实现)
向AI转型的程序员都关注了这个号👇👇👇在这篇文章中,我们将深入探讨时间序列预测的基本概念和方法。我们将首先介绍单元预测和多元预测的概念,然后详细介绍各种深度学习和传统机器学习方法如何应用于时间序列预测,包括循环神经网络(RNN)、一维卷积神经网络(1D-CNN)、Transformer、自回归模型(
机器学习AI算法工程
0
Go 1.22 的新增功能系列之二:reflect.TypeFor
Go 1.22 的第一个候选版本已经发布,这意味着最终版本即将发布,现在是我在博客中介绍我在这个周期中所做工作的时候了。像往常一样,我的贡献很小,但它们是我的,所以我将从幕后的角度来谈谈它们。首先是reflect.TypeFor。这是整个函数:// TypeFor returns the [Type
GoCN
0
SpringBoot 实现图片防盗链功能
程序员的成长之路互联网/程序员/技术/资料共享 关注阅读本文大概需要 4 分钟。来自:blog.csdn.net/weixin_46157208/article/details/138051737前言出于安全考虑,我们需要后端返回的图片只允许在某个网站内展示,不想被爬虫拿到图片地
程序员的成长之路
0
一站式解决方案:基于 Arthas 实现服务发现和权限控制
来源:juejin.cn/post/7281849496983994383👉 欢迎加入小哈的星球 ,你将获得: 专属的项目实战 / Java 学习路线 / 一对一提问 / 学习打卡 / 赠书福利全栈前后端分离博客项目 2.0 版本完结啦, 演示链接
小哈学Java
0
用 Shader 实现旗帜飘扬动画效果
我觉得对于刚入门 3D 编程的朋友来说,如果能够完成代码创建模型数据->创建材质->编写Shader动画这一系列,想必会有满满的成就感。今天就用 Cocos Creator 的 utils.MeshUtils.createMesh 接口,带大家感受一下这个流程。这个流程不仅可以用于新手学
COCOS
2
Go 1.22 的新增功能系列之一:cmp.Or
截至撰写本文时,Go 1.22 已经发布几个月了。早就该结束我为 1.22 所做的工作的系列了。抱歉耽搁了这么久,我最近忙于生活事务。如果您错过了我关于reflect.TypeFor(https://blog.carlana.net/post/2024/golang-reflect-type-for
GoCN
1
21.3K star!推荐一款可视化自动化测试/爬虫/数据采集神器!功能免费且强大!
【温馨提示】由于公众号更改了推送规则,不再按照时间顺序排列,如果不想错过测试开发技术精心准备的的干货文章,请将测试开发技术设为“星标☆”,看完文章在文尾处点亮“在看”!大家好,我是狂师!在大数据时代,信息的获取与分析变得尤为重要。对于开发者、数据分析师乃至非技术人员来说,能够高效地采集网络数据并进行
测试开发技术
4
知乎高问:程序员有必要知道为什么做某个功能吗?
将Python客栈设为“星标⭐”第一时间收到最新资讯前言知乎上有一个提问:程序员有必要知道为什么做某个功能吗?↓↓↓今天,我们就这个话题一起来做个讨论。不知道程序员的你,在接到产品经理提的一个需求后,是习惯马上动手开始撸代码呢?还是会先暂停一下,认真思考一会如下一些问题,比如这个需求产生的背景是什么
Python客栈
0