173 lines
5.1 KiB
C++
173 lines
5.1 KiB
C++
/*
|
||
* Copyright (c) 2016 The ZLToolKit project authors. All Rights Reserved.
|
||
*
|
||
* This file is part of ZLToolKit(https://github.com/ZLMediaKit/ZLToolKit).
|
||
*
|
||
* Use of this source code is governed by MIT license that can be found in the
|
||
* LICENSE file in the root of the source tree. All contributing project authors
|
||
* may be found in the AUTHORS file in the root of the source tree.
|
||
*/
|
||
|
||
#ifndef UTIL_TIMETICKER_H_
|
||
#define UTIL_TIMETICKER_H_
|
||
|
||
#include <cassert>
|
||
#include "logger.h"
|
||
|
||
namespace toolkit {
|
||
|
||
class Ticker {
|
||
public:
|
||
/**
|
||
* 此对象可以用于代码执行时间统计,以可以用于一般计时
|
||
* @param min_ms 开启码执行时间统计时,如果代码执行耗时超过该参数,则打印警告日志
|
||
* @param ctx 日志上下文捕获,用于捕获当前日志代码所在位置
|
||
* @param print_log 是否打印代码执行时间
|
||
* This object can be used for code execution time statistics, and can be used for general timing
|
||
* @param min_ms When the code execution time statistics is enabled, if the code execution time exceeds this parameter, a warning log is printed
|
||
* @param ctx Log context capture, used to capture the current log code location
|
||
* @param print_log Whether to print the code execution time
|
||
|
||
* [AUTO-TRANSLATED:4436cf19]
|
||
*/
|
||
Ticker(uint64_t min_ms = 0,
|
||
LogContextCapture ctx = LogContextCapture(Logger::Instance(), LWarn, __FILE__, "", __LINE__),
|
||
bool print_log = false) : _ctx(std::move(ctx)) {
|
||
if (!print_log) {
|
||
_ctx.clear();
|
||
}
|
||
_created = _begin = getCurrentMillisecond();
|
||
_min_ms = min_ms;
|
||
}
|
||
|
||
~Ticker() {
|
||
uint64_t tm = createdTime();
|
||
if (tm > _min_ms) {
|
||
_ctx << "take time: " << tm << "ms" << ", thread may be overloaded";
|
||
} else {
|
||
_ctx.clear();
|
||
}
|
||
}
|
||
|
||
/**
|
||
* 获取上次resetTime后至今的时间,单位毫秒
|
||
* Get the time from the last resetTime to now, in milliseconds
|
||
|
||
* [AUTO-TRANSLATED:739ad90a]
|
||
*/
|
||
uint64_t elapsedTime() const {
|
||
return getCurrentMillisecond() - _begin;
|
||
}
|
||
|
||
/**
|
||
* 获取从创建至今的时间,单位毫秒
|
||
* Get the time from creation to now, in milliseconds
|
||
|
||
* [AUTO-TRANSLATED:83a189e2]
|
||
*/
|
||
uint64_t createdTime() const {
|
||
return getCurrentMillisecond() - _created;
|
||
}
|
||
|
||
/**
|
||
* 重置计时器
|
||
* Reset the timer
|
||
|
||
* [AUTO-TRANSLATED:2500c6f1]
|
||
*/
|
||
void resetTime() {
|
||
_begin = getCurrentMillisecond();
|
||
}
|
||
|
||
private:
|
||
uint64_t _min_ms;
|
||
uint64_t _begin;
|
||
uint64_t _created;
|
||
LogContextCapture _ctx;
|
||
};
|
||
|
||
class SmoothTicker {
|
||
public:
|
||
/**
|
||
* 此对象用于生成平滑的时间戳
|
||
* @param reset_ms 时间戳重置间隔,没间隔reset_ms毫秒, 生成的时间戳会同步一次系统时间戳
|
||
* This object is used to generate smooth timestamps
|
||
* @param reset_ms Timestamp reset interval, every reset_ms milliseconds, the generated timestamp will be synchronized with the system timestamp
|
||
|
||
* [AUTO-TRANSLATED:0ff567e7]
|
||
*/
|
||
SmoothTicker(uint64_t reset_ms = 10000) {
|
||
_reset_ms = reset_ms;
|
||
_ticker.resetTime();
|
||
}
|
||
|
||
~SmoothTicker() {}
|
||
|
||
/**
|
||
* 返回平滑的时间戳,防止由于网络抖动导致时间戳不平滑
|
||
* Return a smooth timestamp, to prevent the timestamp from being unsmooth due to network jitter
|
||
|
||
* [AUTO-TRANSLATED:26f78ae3]
|
||
*/
|
||
uint64_t elapsedTime() {
|
||
auto now_time = _ticker.elapsedTime();
|
||
if (_first_time == 0) {
|
||
if (now_time < _last_time) {
|
||
auto last_time = _last_time - _time_inc;
|
||
double elapse_time = (now_time - last_time);
|
||
_time_inc += (elapse_time / ++_pkt_count) / 3;
|
||
auto ret_time = last_time + _time_inc;
|
||
_last_time = (uint64_t) ret_time;
|
||
return (uint64_t) ret_time;
|
||
}
|
||
_first_time = now_time;
|
||
_last_time = now_time;
|
||
_pkt_count = 0;
|
||
_time_inc = 0;
|
||
return now_time;
|
||
}
|
||
|
||
auto elapse_time = (now_time - _first_time);
|
||
_time_inc += elapse_time / ++_pkt_count;
|
||
auto ret_time = _first_time + _time_inc;
|
||
if (elapse_time > _reset_ms) {
|
||
_first_time = 0;
|
||
}
|
||
_last_time = (uint64_t) ret_time;
|
||
return (uint64_t) ret_time;
|
||
}
|
||
|
||
/**
|
||
* 时间戳重置为0开始
|
||
* Reset the timestamp to start from 0
|
||
|
||
* [AUTO-TRANSLATED:ca42c3d1]
|
||
*/
|
||
void resetTime() {
|
||
_first_time = 0;
|
||
_pkt_count = 0;
|
||
_ticker.resetTime();
|
||
}
|
||
|
||
private:
|
||
double _time_inc = 0;
|
||
uint64_t _first_time = 0;
|
||
uint64_t _last_time = 0;
|
||
uint64_t _pkt_count = 0;
|
||
uint64_t _reset_ms;
|
||
Ticker _ticker;
|
||
};
|
||
|
||
#if !defined(NDEBUG)
|
||
#define TimeTicker() Ticker __ticker(5,WarnL,true)
|
||
#define TimeTicker1(tm) Ticker __ticker1(tm,WarnL,true)
|
||
#define TimeTicker2(tm, log) Ticker __ticker2(tm,log,true)
|
||
#else
|
||
#define TimeTicker()
|
||
#define TimeTicker1(tm)
|
||
#define TimeTicker2(tm,log)
|
||
#endif
|
||
|
||
} /* namespace toolkit */
|
||
#endif /* UTIL_TIMETICKER_H_ */
|