網(wǎng)站首頁 編程語言 正文
本文主要探討C/C++中無符號整數(shù)超過范圍后的計算問題。
問題提出
nrf52832 的 SDK 中是沒有時間戳獲取的函數(shù)的,為了統(tǒng)計性能耗時,也為了向一些庫提供時間戳(毫秒級別),需要自己利用定時器實現(xiàn)獲取毫秒的接口。
nrf52832 是 32 位的,按毫秒計算,大概49天就會達到最大值,如何處理毫秒數(shù)值溢出后的情況,其實我是不懂的。看了些帖子,說在單獨處理溢出反轉的情形,但總覺得這樣不太好,因此集中了一點時間,了解學習了無符號數(shù)的溢出(或說進位),并寫了點代碼測試。
設計思路
為了方便調(diào)試,本文用 32 位虛擬機 Linux 進行測試。用get_time
獲取系統(tǒng)的秒數(shù)值,其值用g_ms
表示,開一線程time_handler
每秒累計一次時間數(shù)值,開另一線程myfunc_sleep
統(tǒng)計耗時。
需要注意的是,上面所述僅是模擬演示,旨在說明本質(zhì)問題,并非實際使用的。
工程代碼
首先簡單測試無符號數(shù)的相加,函數(shù)如下:
// 測試無符號溢出后的差值 delta void delta_test(int delta) { mytime_t start = 0xfffffffe; /* 以ms為10為例,ent得到的結果為8 8 - start = 10 因此,即使溢出后,差值也是不變的 在延時函數(shù)中,即使時間戳溢出,也是無問題的。 */ mytime_t end = start + delta; mytime_t mydelta = end - start; ? printf("end: %u start: %u delta: %u mydelta: %u\n", end, start, delta, mydelta); for (int i = 0; i < delta; i++) { printf("%u %d\n", start, start); start++; } }
測試代碼:
delta_test(10);
其打印結果如下:
end: 8 start: 4294967294 delta: 10 mydelta: 10
4294967294 -2
4294967295 -1
0 0
1 1
2 2
3 3
4 4
5 5
6 6
7 7
起始數(shù)值為0xfffffffe
,即4294967294
,使用有符號打印,其值為-2
。當超過0xffffffff
則從0
開始計數(shù)。如果僅從結果看,可得到:8 - 4294967294 = 10
,與傳遞的參數(shù)一致。
下面給出多線程測試代碼:
代碼中定義的時間戳變量為mytime_t
,實際上是無符號類型unsigned
。另外也做了測試,在32位機器上使用uint8_t類型也可以得到正確值,但負數(shù)就無法打印出來了。
測試
當g_start
的值為0xfffffffe
時,測試結果如下:
test of unsigned overflow.. sizeof: 4 after sleep 3 s 1 - 4294967294 = 3 after sleep 3 s 4 - 1 = 3 after sleep 3 s 7 - 4 = 3 after sleep 3 s 10 - 7 = 3 after sleep 3 s 13 - 10 = 3
當g_start
的值為0xfffffff0
時,測試結果如下:
第一次測試: test of unsigned overflow.. sizeof: 4 after sleep 2 s 4294967282 - 4294967281 = 1 after sleep 2 s 4294967284 - 4294967282 = 2 after sleep 2 s 4294967286 - 4294967284 = 2 after sleep 2 s 4294967288 - 4294967286 = 2 after sleep 2 s 4294967290 - 4294967288 = 2 after sleep 2 s 4294967292 - 4294967290 = 2 after sleep 2 s 4294967294 - 4294967292 = 2 after sleep 2 s 0 - 4294967294 = 2 after sleep 2 s 2 - 0 = 2 after sleep 2 s 4 - 2 = 2 after sleep 2 s 6 - 4 = 2 after sleep 2 s 8 - 6 = 2 ? 第二次測試: test of unsigned overflow.. sizeof: 4 after sleep 2 s 4294967282 - 4294967280 = 2 after sleep 2 s 4294967284 - 4294967282 = 2 after sleep 2 s 4294967286 - 4294967284 = 2 after sleep 2 s 4294967288 - 4294967286 = 2 after sleep 2 s 4294967290 - 4294967288 = 2 after sleep 2 s 4294967292 - 4294967290 = 2 after sleep 2 s 4294967294 - 4294967292 = 2 after sleep 2 s 0 - 4294967294 = 2 after sleep 2 s 2 - 0 = 2 after sleep 2 s 4 - 2 = 2 after sleep 2 s 6 - 4 = 2 after sleep 2 s 8 - 6 = 2
從結果上看,基本符合要求,即延時2秒,統(tǒng)計的耗時是2。——不管是否有溢出。
擴展知識
計算機中數(shù)值存儲的是2的補碼(2’s complement)。正數(shù)的補碼是其本身,負數(shù)的補碼是原碼基礎上取反碼,末位加1。
mpu6050 芯片的陀螺儀和加速度數(shù)值,是16位有符號數(shù)值,就是用2的補碼形式存儲的。
小結
對于計時、延時類的函數(shù),記錄時間戳的變量為無符號數(shù)。類型為unsinged,不能加范圍限制,這是指平臺最大者,如32位系統(tǒng),使用的是32位無符號數(shù),64位的系統(tǒng)則是64位無符號數(shù)。當變量數(shù)值溢出后,其值歸0,但計時函數(shù)是正常的,不需要額外處理溢出情況。
原文鏈接:https://juejin.cn/post/7106302326763159565
相關推薦
- 2023-07-15 前后端交互,前端參數(shù)格式form-data
- 2022-07-03 pandas創(chuàng)建series的三種方法小結_python
- 2022-06-15 C#實現(xiàn)冒泡排序和插入排序算法_C#教程
- 2022-10-19 react+antd實現(xiàn)動態(tài)編輯表格數(shù)據(jù)_React
- 2022-12-11 C++?Boost?Tokenizer使用詳細講解_C 語言
- 2022-07-10 ENOENT讀取文件報錯(fs)
- 2021-12-01 C語言system函數(shù)使用方法詳解_C 語言
- 2023-03-03 AJAX亂碼與異步同步以及封裝jQuery庫實現(xiàn)步驟詳解_AJAX相關
- 最近更新
-
- window11 系統(tǒng)安裝 yarn
- 超詳細win安裝深度學習環(huán)境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區(qū)別,Jav
- spring @retryable不生效的一種
- Spring Security之認證信息的處理
- Spring Security之認證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權
- redisson分布式鎖中waittime的設
- maven:解決release錯誤:Artif
- restTemplate使用總結
- Spring Security之安全異常處理
- MybatisPlus優(yōu)雅實現(xiàn)加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務發(fā)現(xiàn)-Nac
- Spring Security之基于HttpR
- Redis 底層數(shù)據(jù)結構-簡單動態(tài)字符串(SD
- arthas操作spring被代理目標對象命令
- Spring中的單例模式應用詳解
- 聊聊消息隊列,發(fā)送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支