Markdown版本笔记 我的GitHub首页 我的博客 我的微信 我的邮箱
MyAndroidBlogs baiqiantao baiqiantao bqt20094 baiqiantao@sina.com

倒计时 总结 Timer Handler RxJava


目录

利用系统API的几种实现方式

使用 CountDownTimer 实现 - 最简洁【推荐】

CountDownTimer 简介

文档

Schedule安排、清单 a countdown until a time in the future, with regular规律的 notifications on intervals间隔 along the way过程.

在文本字段中显示一个30秒倒计时的示例:

@BindView(R.id.send) Button send;//发送验证码
new CountDownTimer(60000, 1000) {
@Override
public void onTick(long millisUntilFinished) {
send.setText(millisUntilFinished / 1000 + "S");
}
@Override
public void onFinish() {
send.setEnabled(true);
send.setText("重新发送");
}
}.start();

The calls to onTick(long) are synchronized同步 to this object so that one call to onTick(long) won't ever occur before the previous callback is complete.

This is only relevant相应、相关 when the implementation of onTick(long) takes an amount of一定数量的 time to execute执行 that is significant重大 compared to the countdown interval间隔.

API数量非常少,但各个都极其有用

构造方法

CountDownTimer(long millisInFuture, long countDownInterval)
  • millisInFuture: The number of millis in the future from the call to start() until the countdown is done and onFinish() is called.
  • countDownInterval: The interval along the way to receive onTick(long) callbacks.

开启和结束方法

final void cancel()
final CountDownTimer start()

抽象(回调)方法

abstract void onFinish():Callback fired when the time is up.
abstract void onTick(long millisUntilFinished):Callback fired on regular interval. millisUntilFinished: The amount of time until finished.

使用案例

@BindView(R.id.send) Button send;//发送验证码
private CountDownTimer timer;//使用CountDownTimer

@OnClick({R.id.send, R.id.next})
public void onClickIv(View v) {
switch (v.getId()) {
case R.id.send:
setTimer();//实际是要在点击之后判断如果符合倒计时条件才调用 setTimer 方法,要在失败后调用 destoryTimer
break;
}
}

@Override
protected void onDestroy() {
super.onDestroy();
destoryTimer();
}

private void setTimer() {
send.setEnabled(false);
timer = new CountDownTimer(60 * 1000, 1000) {
@Override
public void onTick(long millisUntilFinished) {
int time = (int) (millisUntilFinished / 1000);
send.setText(time + "s");
} @Override
public void onFinish() {
destoryTimer();
}
};
timer.start();
}

private void destoryTimer() {
send.setEnabled(true);
send.setText("获取验证码");
if (timer != null) {
timer.cancel();
timer = null;
}
}

使用 RxJava 实现 - 方便强大【推荐】

可以使用 intervalRange 很方便的实现这个功能,也可以使用 repeat、repeatUntil、repeatWhen 间接实现类似功能。

@BindView(R.id.send) Button send;//发送验证码
private Disposable disposable; @OnClick({R.id.send, R.id.next})
public void onClickIv(View v) {
switch (v.getId()) {
case R.id.send:
startCountDown();
break;
}
}

@Override
protected void onDestroy() {
super.onDestroy();
if (disposable != null && !disposable.isDisposed()) {
disposable.dispose();
}
} private void startCountDown() {
send.setEnabled(false);
disposable = Observable.intervalRange(0, 10, 0, 1, TimeUnit.SECONDS) //起始值,发送总数量,初始延迟,固定延迟
.subscribeOn(Schedulers.io())
.observeOn(AndroidSchedulers.mainThread())
.subscribe(time -> send.setText((10 - time) + "s"),
Throwable::printStackTrace,
() -> {
send.setEnabled(true);
send.setText("获取验证码");
}
);
}

使用 Timer + Handler 实现 - 麻烦【不推荐】

Timer + 普通 Handler - 麻烦

@BindView(R.id.send) Button send;//发送验证码
private int time = 60;//倒计时时间
private Timer timer;

@OnClick({R.id.send, R.id.next})
public void onClickIv(View v) {
switch (v.getId()) {
case R.id.send:
setTimer();//实际是要在点击之后判断如果符合倒计时条件才调用 setTimer 方法,要在失败后调用 destoryTimer
break;
}
}

@Override
protected void onDestroy() {
super.onDestroy();
destoryTimer();
}

@SuppressLint("HandlerLeak")
private Handler handler = new Handler() {
@Override
public void handleMessage(android.os.Message msg) {
switch (msg.what) {
case 1:
send.setText(time + "s");
break;
case 2:
destoryTimer();
break;
}
}
};

//定时器
private void setTimer() {
send.setEnabled(false);//不可点击
timer = new Timer();
TimerTask task = new TimerTask() {
@Override
public void run() {
time--;
if (time > 0) {
handler.sendEmptyMessage(1);
} else {
handler.sendEmptyMessage(2);
}
}
};
timer.schedule(task, 0, 1000);//每隔一秒钟执行一次
}

private void destoryTimer() {
time = 60;//重新倒计时
send.setEnabled(true);//重新可点击
send.setText("重新发送");//重设文字
if (timer != null) {
timer.cancel();
timer = null;
}
if (handler!=null) {
handler.removeCallbacksAndMessages(null);
}
}

Timer + 静态 Handler - 更麻烦

相比示例一,是将Handler定义为了静态内部类,以防止内存泄漏

@BindView(R.id.send) Button send;//发送验证码
private int time = 60;//倒计时时间
private Timer timer;

@OnClick({R.id.send, R.id.next})
public void onClickIv(View v) {
switch (v.getId()) {
case R.id.send:
setTimer();//实际是要在点击之后判断如果符合倒计时条件才调用 setTimer 方法,要在失败后调用 destoryTimer
break;
}
}

@Override
protected void onDestroy() {
super.onDestroy();
destoryTimer();
}

private Handler handler = new MyHandler(this);
private static class MyHandler extends Handler {
private SoftReference<ForgetPasswordActivity> mSoftReference; MyHandler(ForgetPasswordActivity activity) {
mSoftReference = new SoftReference<>(activity);
} @Override
public void handleMessage(Message msg) {
super.handleMessage(msg);
ForgetPasswordActivity activity = mSoftReference.get();
if (activity != null) {
switch (msg.what) {
case 1:
activity.send.setText(activity.time + "s");
break;
case 2:
activity.destoryTimer();
break;
}
}
}
}

//定时器
private void setTimer() {
send.setEnabled(false);//不可点击
timer = new Timer();
TimerTask task = new TimerTask() {
@Override
public void run() {
time--;
if (time > 0) {
handler.sendEmptyMessage(1);
} else {
handler.sendEmptyMessage(2);
}
}
};
timer.schedule(task, 0, 1000);//每隔一秒钟执行一次
}

private void destoryTimer() {
time = 60;//重新倒计时
send.setEnabled(true);//重新可点击
send.setText("重新发送");//重设文字
if (timer != null) {
timer.cancel();
timer = null;
}
if (handler!=null) {
handler.removeCallbacksAndMessages(null);
}
}

Timer + runOnUiThread - 也麻烦

可以不用Handler而用其他更精简的API:

@BindView(R.id.send) Button send;//发送验证码
private int time = 60;//倒计时时间
private Timer timer;

@OnClick({R.id.send, R.id.next})
public void onClickIv(View v) {
switch (v.getId()) {
case R.id.send:
setTimer();//实际是要在点击之后判断如果符合倒计时条件才调用 setTimer 方法,要在失败后调用 destoryTimer
break;
}
}

@Override
protected void onDestroy() {
super.onDestroy();
destoryTimer();
}

private void setTimer() {
send.setEnabled(false);
TimerTask task = new TimerTask() {
@Override
public void run() {
runOnUiThread(() -> {
time--;
if (time > 0) {
send.setText(time + "s");
} else {
destoryTimer();
}
});
}
};
timer = new Timer();
timer.schedule(task, 0, 1000);//每隔一秒钟执行一次
}

private void destoryTimer() {
time = 60;
send.setEnabled(true);
send.setText("获取验证码");
if (timer != null) {
timer.cancel();
timer = null;
}
}

使用纯 Handler 实现 - 特麻烦【强烈不建议】

倒计时通过用 Handler 发送 Delayed 消息来实现。核心代码为:

handler.sendMessageDelayed(handler.obtainMessage(1), 1000);

final Handler handler = new Handler() {
public void handleMessage(Message msg) {
switch (msg.what) {
case 1:
time--;
if (time > 0) {
send.setText(time + "S");
handler.sendMessageDelayed(handler.obtainMessage(1), 1000);//循环发送
} else {
send.setEnabled(true);
send.setText("重新发送");
}
}
}
};

倒计时通过用 Handler 发送 Delayed 的 Runnable 来实现,和上面原理是完全一样的。核心代码为:

Handler handler = new Handler();
handler.postDelayed(runnable, 1000);

Runnable runnable = new Runnable() {
@Override
public void run() {
time--;
if (time > 0) {
send.setText(time + "S");
handler.postDelayed(this, 1000);
} else {
send.setEnabled(true);
send.setText("重新发送");
}
}
};

开源框架 CountdownView 简介

GitHub上星星最多的倒计时控件:CountdownView

CountdownView:Android倒计时控件,使用Canvas绘制,支持多种样式

compile 'com.github.iwgang:countdownview:2.1.3'

引用类名

cn.iwgang.countdownview.CountdownView

基本使用

CountdownView mCountdownView = (CountdownView)findViewById(R.id.countdownView);
mCountdownView.start(995550000); // 毫秒 // 或者自己编写倒计时逻辑,然后调用 updateShow 来更新UI
for (int time=0; time<1000; time++) {
mCountdownView.updateShow(time);
}

其他用法

  • 动态设置自定义属性:.dynamicShow(DynamicConfig)
  • 倒计时结束后的回调:.setOnCountdownEndListener(OnCountdownEndListener);
  • 指定间隔时间的回调:.setOnCountdownIntervalListener(long, OnCountdownIntervalListener);

在 RecyclerView 中实现倒计时

更改数据源方式 - 简单但不可靠

这种方案在数据量特别小(即List的size()特别小),且刷新item及计算倒计时耗费的时间特别短时适用,否则,将会产生巨大的时间延迟。

//定时器,用于刷新 GridView 的数据源
private void setQryTimer() {
cancelQryTimer(); //取消之前的定时器
qryTimer = new Timer(); //重新设置定时器
qryTimer.schedule(new TimerTask() {
@Override
public void run() {
runOnUiThread(() -> {
if (fixRpList != null && fixRpList.size() > 0) {
for (FixRpBean item : fixRpList) {
if (item.diff_time >= 0) item.diff_time = item.diff_time - 1000L; //更改数据源
}
if (fixRpDialog != null) fixRpDialog.upDate(fixRpList); //刷新页面
}
});
}
}, 0, 1000); //以固定的周期刷新
} public void upDate(List<FixRpBean> redPacketList) {
list.clear(); //情况旧的数据
list.addAll(redPacketList); //设置新的数据(如果列表的数据和源数据是同一个集合,也可以直接更新)
mRecyclerView.getAdapter().notifyDataSetChanged();//建议使用RecyclerView的局部刷新功能
}

让 System 帮我们倒计时 - 推荐

核心思想为:利用System.currentTimeMillis()帮我们计算倒计时,并且在onViewAttachedToWindow时重新开始倒计时,在onViewDetachedFromWindow时关闭倒计时。

public class RecyclerViewActivity extends Activity {
private List<ItemInfo> mDataList; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_recyclerview); initData(); RecyclerView recyclerView = (RecyclerView) findViewById(R.id.rv);
recyclerView.setAdapter(new MyAdapter(this, mDataList));
recyclerView.setLayoutManager(new LinearLayoutManager(this));
recyclerView.addItemDecoration(new DividerItemDecoration(this, DividerItemDecoration.VERTICAL));
recyclerView.setItemAnimator(new DefaultItemAnimator());
} private void initData() {
mDataList = new ArrayList<>();
for (int i = 1; i < 20; i++) {
mDataList.add(new ItemInfo(i * 20 * 1000));
}
// 校对倒计时
long curTime = System.currentTimeMillis();
for (ItemInfo itemInfo : mDataList) {
itemInfo.endTime = curTime + itemInfo.countdown;
}
} static class MyAdapter extends RecyclerView.Adapter<MyViewHolder> {
private Context mContext;
private List<ItemInfo> mDatas; public MyAdapter(Context context, List<ItemInfo> datas) {
this.mContext = context;
this.mDatas = datas;
} @Override
public MyViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
return new MyViewHolder(LayoutInflater.from(mContext).inflate(R.layout.list_item, parent, false));
} @Override
public void onBindViewHolder(MyViewHolder holder, int position) {
holder.bindData(mDatas.get(holder.getAdapterPosition()));
} @Override
public int getItemCount() {
return mDatas.size();
} //******************************************** 关键代码 ↓↓ **********************************
@Override
public void onViewAttachedToWindow(MyViewHolder holder) {
super.onViewAttachedToWindow(holder);//父类中为空代码
holder.refreshTime(mDatas.get(holder.getAdapterPosition()).endTime - System.currentTimeMillis());
} @Override
public void onViewDetachedFromWindow(MyViewHolder holder) {
super.onViewDetachedFromWindow(holder);
holder.countdownView.stop();
}
//******************************************** 关键代码 ↑↑ **********************************
} static class MyViewHolder extends RecyclerView.ViewHolder {
public CountdownView countdownView; public MyViewHolder(View itemView) {
super(itemView);
countdownView = (CountdownView) itemView.findViewById(R.id.countdownView);
} public void bindData(ItemInfo itemInfo) {
refreshTime(itemInfo.endTime - System.currentTimeMillis());
} public void refreshTime(long leftTime) {
if (leftTime > 0) {
countdownView.start(leftTime);
} else {
countdownView.stop();//停止计时器,mCustomCountDownTimer.stop();
countdownView.allShowZero();//所有计时清零,即mCountdown.setTimes(0, 0, 0, 0, 0);
}
}
} static class ItemInfo {
public long countdown;
/*
根据服务器返回的countdown换算成手机对应的开奖时间 (毫秒)
[正常情况最好由服务器返回countdown字段,然后客户端再校对成该手机对应的时间,不然误差很大]
*/
public long endTime; public ItemInfo(long countdown) {
this.countdown = countdown;
}
}
}

自己维护倒计时 - 既麻烦又低效

自己维护倒计时,再调用 countdownView.updateShow 来刷新显示

并且根据需要在 onResume 时开启倒计时,在 onPause 及 onDestroy 时关闭倒计时。

//自己维护倒计时,再调用 countdownView.updateShow 来刷新显示
public class RecyclerViewActivity2 extends AppCompatActivity {
private MyAdapter mMyAdapter;
private List<ItemInfo> mDataList; @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_recyclerview); initData(); RecyclerView recyclerView = (RecyclerView) findViewById(R.id.rv);
mMyAdapter = new RecyclerViewActivity2.MyAdapter(this, mDataList);
recyclerView.setAdapter(mMyAdapter);
recyclerView.setLayoutManager(new LinearLayoutManager(this));
recyclerView.addItemDecoration(new DividerItemDecoration(this, DividerItemDecoration.VERTICAL));
recyclerView.setItemAnimator(new DefaultItemAnimator());
} private void initData() {
mDataList = new ArrayList<>();
for (int i = 1; i < 20; i++) {
mDataList.add(new ItemInfo(1000 + i, "RecyclerView_测试标题_" + i, i * 20 * 1000));
}
// 校对倒计时
long curTime = System.currentTimeMillis();
for (ItemInfo itemInfo : mDataList) {
itemInfo.setEndTime(curTime + itemInfo.getCountdown());
}
} @Override
protected void onResume() {
super.onResume();
if (null != mMyAdapter)
} @Override
protected void onPause() {
super.onPause();
if (null != mMyAdapter)
} @Override
public void onDestroy() {
super.onDestroy();
if (null != mMyAdapter)
} static class MyAdapter extends RecyclerView.Adapter<MyViewHolder> {
private Context mContext;
private List<ItemInfo> mDatas;
private final SparseArray<MyViewHolder> mCountdownVHList;
private Handler mHandler = new Handler();
private Timer mTimer;
private boolean isCancel = true; public MyAdapter(Context context, List<ItemInfo> datas) {
this.mContext = context;
this.mDatas = datas;
mCountdownVHList = new SparseArray<>();
startRefreshTime(); //开启倒计时
} public void startRefreshTime() {
if (!isCancel) return;
if (null != mTimer) mTimer.cancel(); isCancel = false;
mTimer = new Timer();
mTimer.schedule(new TimerTask() {
@Override
public void run() {
mHandler.post(mRefreshTimeRunnable);
}
}, 0, 10);
} public void cancelRefreshTime() {
isCancel = true;
if (null != mTimer) {
mTimer.cancel();
}
mHandler.removeCallbacks(mRefreshTimeRunnable);
} @Override
public MyViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
return new MyViewHolder(LayoutInflater.from(mContext).inflate(R.layout.list_item, parent, false));
} @Override
public void onBindViewHolder(MyViewHolder holder, int position) {
ItemInfo curItemInfo = mDatas.get(position);
holder.bindData(curItemInfo); // 处理倒计时
if (curItemInfo.getCountdown() > 0) {
synchronized (mCountdownVHList) {
mCountdownVHList.put(curItemInfo.getId(), holder); //开启倒计时
}
}
} @Override
public int getItemCount() {
return mDatas.size();
} @Override
public void onViewRecycled(MyViewHolder holder) {
super.onViewRecycled(holder); ItemInfo curAnnounceGoodsInfo = holder.getBean();
if (null != curAnnounceGoodsInfo && curAnnounceGoodsInfo.getCountdown() > 0) {
mCountdownVHList.remove(curAnnounceGoodsInfo.getId()); //移除
}
} private Runnable mRefreshTimeRunnable = new Runnable() {
@Override
public void run() {
if (mCountdownVHList.size() == 0) return;
synchronized (mCountdownVHList) {
long currentTime = System.currentTimeMillis();
int key;
for (int i = 0; i < mCountdownVHList.size(); i++) {
key = mCountdownVHList.keyAt(i);
MyViewHolder curMyViewHolder = mCountdownVHList.get(key);
if (currentTime >= curMyViewHolder.getBean().getEndTime()) {
curMyViewHolder.getBean().setCountdown(0);// 倒计时结束
mCountdownVHList.remove(key);
notifyDataSetChanged();
} else {
curMyViewHolder.refreshTime(currentTime); //刷新时间
}
}
}
}
};
} static class MyViewHolder extends RecyclerView.ViewHolder {
private TextView mTvTitle;
private CountdownView mCvCountdownView;
private ItemInfo mItemInfo; public MyViewHolder(View itemView) {
super(itemView);
mTvTitle = (TextView) itemView.findViewById(R.id.tv_title);
mCvCountdownView = (CountdownView) itemView.findViewById(R.id.cv_countdownView);
} public void bindData(ItemInfo itemInfo) {
mItemInfo = itemInfo;
if (itemInfo.getCountdown() > 0) {
refreshTime(System.currentTimeMillis());
} else {
mCvCountdownView.allShowZero();
}
mTvTitle.setText(itemInfo.getTitle());
} public void refreshTime(long curTimeMillis) {
if (null == mItemInfo || mItemInfo.getCountdown() <= 0) return;
mCvCountdownView.updateShow(mItemInfo.getEndTime() - curTimeMillis);
} public ItemInfo getBean() {
return mItemInfo;
}
} static class ItemInfo {
private int id;
private String title;
private long countdown;
/*
根据服务器返回的countdown换算成手机对应的开奖时间 (毫秒)
[正常情况最好由服务器返回countdown字段,然后客户端再校对成该手机对应的时间,不然误差很大]
*/
private long endTime; public ItemInfo(int id, String title, long countdown) {
this.id = id;
this.title = title;
this.countdown = countdown;
}
//get、set方法...
} }

2017-6-12

倒计时 总结 Timer Handler CountDownTimer RxJava MD的更多相关文章

  1. 倒计时实现方案总结 Timer Handler

    利用Timer实现倒计时 @BindView(R.id.send) Button send;//发送验证码 private int time = 60;//倒计时 private Timer time ...

  2. Android中三种计时器Timer、CountDownTimer、handler.postDelayed的使用

    在android开发中,我们常常需要用到计时器,倒计时多少秒后再执行相应的功能,下面我就分别来讲讲这三种常用的计时的方法. 一.CountDownTimer 该类是个抽象类,如果要使用这个类中的方法, ...

  3. 115、定时器(TimerTask+Timer+Handler)

    public class TimerUtils { public static Activity act; public static List<MaiDianModels> listMa ...

  4. 线程 Timer TimerTask 计时器 定时任务 MD

    Markdown版本笔记 我的GitHub首页 我的博客 我的微信 我的邮箱 MyAndroidBlogs baiqiantao baiqiantao bqt20094 baiqiantao@sina ...

  5. Android基础之——CountDownTimer类,轻松实现倒计时功能

    在发现这个类之前,一直是用的handler,子线程发消息,UI线程进行倒计时的显示工作.前几天在做一个倒计时显示的时候发现了这个类,用起来非常方便 翻看了下源代码.内部已经帮我们实现了handler的 ...

  6. 拓展 Android 原生 CountDownTimer 倒计时

    拓展 Android 原生 CountDownTimer 倒计时 [TOC] CountDownTimer 在系统的CountDownTimer上进行的修改,主要是拓展了功能,当然也保留了系统默认的模 ...

  7. [Android Pro] CountDownTimer倒计时

    定时执行在一段时候后停止的倒计时,在倒计时执行过程中会在固定间隔时间得到通知(译者:触发onTick方法),下面的例子显示在一个文本框中显示一个30s倒计时: new CountdownTimer(3 ...

  8. Android短轮询解决方案——CountDownTimer+Handler

    转载请注明原文地址:http://www.cnblogs.com/ygj0930/p/7657194.html 一:应用场景 在诸如自动售卖机之类的扫码支付场景中,客户端在获得支付二维码或者发出支付请 ...

  9. android中倒计时控件CountDownTimer分析

    android中倒计时控件CountDownTimer分析 1 示例代码 new CountDownTimer(10000, 1000) { public void onTick(long milli ...

随机推荐

  1. linux 下安装jdk环境安装

    一.创建jdk安装目录mkdir /usr/local/java 二.将jdk解压到安装目录中,直接到java目录中,如果不是处理下不要有子目录 tar -zxvf jdk-8u91-linux-x6 ...

  2. Extjs设置或获取cookie

    设置cookie var myCookie = Ext.util.Cookie.set(‘YourCookieName’,'YourValue’); 读取cookie Ext.util.Cookie. ...

  3. 俄罗斯方块 Tetris

    今天,为大家带来一个用Qt C++ (Windows环境下)做的一个简易俄罗斯方块小游戏 思路和模块介绍都在注释里面,其次就是一些项目中遇到的问题以及解决方案,在后面部分说明. 一.效果 测试图样 Q ...

  4. MySQL 关于索引那点事

    索引 其实数据库中的数据是按页存放的其实索引也是按页存放的所以本质上索引也占硬盘空间(以最小的消耗,换取最大的利益) 索引是一种有效组合数据的方式!为快速查找到指定记录做铺垫 目的就是快速或者某个记录 ...

  5. nginx安装第三方模块

    原已经安装好的nginx,现在需要添加一个未被编译安装的模块 举例说明:安装第三方的ngx_cache_purge模块(用于清除指定URL的缓存) nginx的模块是需要重新编译nginx,而不是像a ...

  6. 用Win32编写发送消息至Notepad++的程序

    这次利用Win32编程写一个发送"Win32 Assembly,My First SendMessage Program !" 每个程序要发送消息至另一个程序的时候,通常使用Sen ...

  7. (转)park1.0.0生态圈一览

    转自博客:http://www.tuicool.com/articles/FVBJBjN Spark1.0.0生态圈一览 Spark生态圈,也就是BDAS(伯克利数据分析栈),是伯克利APMLab实验 ...

  8. CentOS7安装GNOME可视化界面和如何配置IP地址

    本人在虚拟机安装 CentOS7 1,检查一下我们已经安装的软件以及可以安装的软件,用命令 yum grouplist 2,然后安装我们需要的图形界面软件,GNOME(GNOME Desktop) 这 ...

  9. 离线下载Xcode的文档

    https://developer.apple.com/library/downloads/docset-index.dvtdownloadableindex 找到里面的文档下载地址 例如iOS 8. ...

  10. RTSP交互过程

    步骤一: 发送:OPTIONS rtsp://127.0.0.1/172.30.31.225:8000:HIK-DS8000HC:0:1:admin:hs123456:av_stream RTSP/1 ...