溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

C++中Boost.Chrono時間庫的使用方法

發布時間:2020-09-25 11:36:31 來源:腳本之家 閱讀:528 作者:taozj 欄目:編程語言

前言

大家應該都有所體會,時鐘這個東西在程序中扮演者重要的角色,在系統編程的時候睡眠、帶超時的等待、帶超時的條件變量、帶超時的鎖都會用到,但是往往對特定系統依賴性很大,感覺即使不考慮系統的跨平臺性,如果能使用一個穩定的接口,同時如果能夠方便的對時刻、時段等進行相關的操作和運算,將是再好不過的了。

在boost庫中和時間相關的庫有Boost.DateTime和Boost.Chrono,前者專注于時間時刻以及本地化相關的內容,而后者主要是時刻、時長和時間的計算等內容。當然,C++11標準已經支持std::chrono了,但是為了兼容老編譯系統現在很多C++庫和程序都使用boost.chrono作為時間類庫(還有的原因就是std::chrono沒有收錄boost.chrono的所有功能,比如統計CPU使用時間、自定義時間輸出格式等),不過比較可惜的是即便使用boost::chrono作為權宜之計,也需要boost-1.47版本之上才行,而現在比較舊的發行版需要升級boost庫才可以使用。想想現在RHEL-6.x仍然被大規模的部署,而且RedHat要為這貨提供長達十年的技術支持,真不知道啥時候才能順順利利的享受C++11……

Boost.Chrono的時間類型分為duration和time_point,也就是時長和時刻兩類,很多概念和接口都是圍繞這兩個維度去定義和實現的。

一、Clock

clock是Boost.Chrono中的重要概念,而且這些clock都包含一個now()的成員函數,用于返回當前的time_point。Boost.Chrono包含的clock類型有:

  (1) chrono::system_clock 代表系統時間,比如電腦上顯示的當前時間,其特點是這個時間可以被用戶手動設置更新,所以這個時鐘是可以和外部時鐘源同步的。這個時鐘還有一個to_time_t()成員函數,用于返回自1970.1.1開始到某個時間點所經過的秒數,數據類型是std::time_t。這種時鐘通常用來轉換成日歷時間使用。

  (2) chrono::steady_clock 其特點是時間是單調增長的,后一個時刻訪問得到的時間點肯定比之前時刻得到的時間點要晚,即使我們手動將系統時間向前調整了也不會改變這個時鐘穩步向前推行累計,其也被稱為monotonic time,該時鐘是均勻增長且不能被調整,其特性對于很多不允許時間錯亂的系統是十分重要的。chrono::steady_clock通常是基于系統啟動時間來計時的,而且常常用來進行耗時、等待等工作使用。

  (3) chrono::high_resolution_clock 依賴于系統實現,通常是上面兩種時鐘的某個宏定義,取決于哪個時鐘源更為的精確,所以其輸出也決定于取決于上面哪個clock來實現的。

  (4) chrono::process_real_cpu_clock 表示自進程啟動以來使用的CPU時間,而這個數據也可以通過使用std::clock()來獲得。chrono::process_user_cpu_clock、boost::chrono::process_system_cpu_clock表示自進程啟動以來,在用戶態、內核態所花費的時間,而所有的這些事件可以通過chrono::process_cpu_clock來獲得,他返回上面所有時間組成的一個tuple結構。

  (5) chrono::thread_clock 返回基于線程統計的花費時間,而且不區分用戶態、內核態的時間。

二、time_point

time_point代表時間點,其等價于某個時刻(clock)+duration的結果,同時兩個time_point做減法也可以得到一個duration。time_point常見的描述為:3分鐘之后、2038年1月1日10:32:23、定時器啟動后的20ms……

上面的clock都有一個now()成員函數,其返回的就是chrono::time_point類型。這個類型使用一個模板來實現的,所以其實際類型極度依賴于所選擇的時鐘源。

Boost.Chrono有一個chrono::time_point_cast轉換函數,可以顯式從高粒度向低粒度對time_point進行轉換。

chrono::process_real_cpu_clock::time_point p = chrono::process_real_cpu_clock::now();
std::cout << chrono::time_point_cast<minutes>(p) << '\n';

三、duration

關于時長,chrono::duration也是一個模板類型,其第一個模板參數表明存儲所用的數據類型(int、long、double等),第二個模板參數表示ratio(比如24、60、1000等)。為了方便用戶的使用,Boost.Chrono提供了duration常用的六種時間類型,且他們都用一個足夠大的整數進行內部保存,其計量值可以用count()成員函數得到:

chrono::nanoseconds、chrono::milliseconds、chrono::microseconds、chrono::seconds、chrono::minutes、chrono::hours,而且為這些類型都重載了計算操作符:+、-、<等,方便時間的計算和比較。文檔說傳統的Boost.DateTime是用繼承實現的,相比而言Boost.Chrono更加的簡潔高效,而且和前者保持了一致的接口。

chrono::steady_clock::time_point start = chrono::steady_clock::now();
...
chrono::duration<double> sec = chrono::steady_clock::now() - start;
std::cout << "we took " << sec.count() << " seconds\n";
auto go = chrono::steady_clock::now() + chrono::nanoseconds(500); 
while (chrono::steady_clock::now() < go) 
... ;

上面的六種time_point類型表示的維度不一,粗粒度的時長肯定能用細粒度的類型表示,反之則可能丟失精度,所以需要使用chrono::duration_cast()函數做顯式的轉換。

可能上面六種類型的時間不咋的,但重點是現在boost::chrono被廣為使用在boost的其他庫里面,比如我們看一個條件變量的帶超時等待的原型:

template< typename Clock, typename Duration >
cv_status wait_until( std::unique_lock< mutex > & lk,
      std::chrono::time_point< Clock, Duration > const& abs_time);
      
template< typename Rep, typename Period >
cv_status wait_for( std::unique_lock< mutex > & lk,
     std::chrono::duration< Rep, Period > const& rel_time);

這就意味著我們可以直接將chrono::minutes{2}這樣的duration對象丟給這個函數就好了,創建任意精度的時長都很方便,而不用像以前一樣關注函數接口有人用seconds、有人用milliseconds、有人用timeval了。

現在boost庫和標準庫中,基于時間段超時的函數都具有for后綴,而基于時間點超時的變量具有until后綴,比如this_thread::sleep_for()this_thread::sleep_until()類似的還有:wait、try_lock、unique_lock用于條件變量、mutex互斥、unique_lock操作。

四、自定義格式的時間輸出

通過time_fmt()可以對時刻進行格式化輸出,使用的時候需要包含頭文件。

time_fmt(boost::chrono::timezone::local, "%H:%M:%S");
time_fmt(boost::chrono::timezone::utc, "%H:%M:%S");

參考

Chapter 37. Boost.Chrono

Chapter 8. Boost.Chrono 2.0.5

總結

以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作能帶來一定的幫助,如果有疑問大家可以留言交流,謝謝大家對億速云的支持。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

亚洲午夜精品一区二区_中文无码日韩欧免_久久香蕉精品视频_欧美主播一区二区三区美女