在线观看www成人影院-在线观看www日本免费网站-在线观看www视频-在线观看操-欧美18在线-欧美1级

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

剖析提升字符串格式化效率的小技巧

FPGA之家 ? 來源: IOT物聯(lián)網(wǎng)小鎮(zhèn) ? 作者:道哥 ? 2021-04-30 13:43 ? 次閱讀

一、前言

嵌入式項目開發(fā)中,字符串格式化是很常見的操作,我們一般都會使用 C 庫中的 sprintf 系列函數(shù)來完成格式化。

從功能上來說,這是沒有問題的,但是在一些時間關(guān)鍵場合,字符串的格式化效率會對整個系統(tǒng)產(chǎn)生顯著的影響。

例如:在一個日志系統(tǒng)中,吞吐率是一個重要的性能指標(biāo)。每個功能模塊都產(chǎn)生了大量的日志信息,日志系統(tǒng)需要把時間戳添加到每條日志的頭部,此時字符串的格式化效率就比較關(guān)鍵了。

天下武功,唯快不破!

這篇文章就專門來聊一聊把數(shù)字格式化成字符串,可以有什么更好的方法。也許技術(shù)含量不高,但是很實(shí)用!

二、最簡單的格式化

#include #include #include #include

int main(){ char buff[32] = { 0 }; sprintf(buff, "%ld", LONG_MAX); printf("buff = %s ", buff);}

其中,LONG_MAX 表示 long 型數(shù)值的最大值。代碼在眨眼功夫之間就執(zhí)行結(jié)束了,但是如果是一百萬、一千萬次呢?

三、測試1:手動格式化數(shù)字

1. 獲取系統(tǒng)時間戳函數(shù)

我的測試環(huán)境是:在 Win10 中通過 VirtualBox,安裝了 Ubuntu16.04 虛擬機(jī),使用系統(tǒng)自帶的 gcc 編譯器。

為了測試代碼執(zhí)行的耗時,我們寫一個簡單的函數(shù):獲取系統(tǒng)的時間戳,通過計算時間差值來看一下代碼的執(zhí)行速度。

// 獲取系統(tǒng)時間戳long long getSysTimestamp(){ struct timeval tv; gettimeofday(&tv, 0); long long ts = (long long)tv.tv_sec * 1000000 + tv.tv_usec; return ts; }

2. 實(shí)現(xiàn)格式化數(shù)字的函數(shù)

// buff: 格式化之后字符串存儲地址;// value: 待格式化的數(shù)字void Long2String(char *buff, long value){ long tmp; char tmpBuf[32] = { 0 }; // p 指向臨時數(shù)組的最后一個位置 char *p = &tmpBuf[sizeof(tmpBuf) - 1]; while (value != 0) { tmp = value / 10; // 把一個數(shù)字轉(zhuǎn)成 ASCII 碼,放到 p 指向的位置。 // 然后 p 往前移動一個位置。 *--p = (char)('0' + (value - tmp * 10)); value = tmp; } // 把臨時數(shù)組中的每個字符,復(fù)制到 buff 中。 while (*p) *buff++ = *p++;}

這個函數(shù)的過程很簡單,從數(shù)字的后面開始,把每一個數(shù)字轉(zhuǎn)成 ASCII 碼,放到一個臨時數(shù)組中(也是從后往前放),最后統(tǒng)一復(fù)制到形參指針 buff 指向的空間。

3. 測試代碼

int main(){ printf("long size = %d, LONG_MAX = %ld ", sizeof(long), LONG_MAX); // 測試 1000 萬次 int total = 1000 * 10000; char buff1[32] = { 0 }; char buff2[32] = { 0 }; // 測試 sprintf long long start1 = getSysTimestamp(); for (int i = 0; i < total; ++i) sprintf(buff1, "%ld", LONG_MAX); printf("sprintf ellapse: %lld us ", getSysTimestamp() - start1); // 測試 Long2String long long start2 = getSysTimestamp(); for (int i = 0; i < total; ++i) Long2String(buff2, LONG_MAX); printf("Long2String ellapse: %lld us ", getSysTimestamp() - start2); return 0;}

4. 執(zhí)行結(jié)果對比

long size = 4, LONG_MAX = 2147483647sprintf ellapse: 1675761 us Long2String ellapse: 527728 us

也就是說:把一個 long 型數(shù)字格式化成字符串:

使用 sprintf 庫函數(shù),耗時 1675761 us;

使用自己寫的 Long2String 函數(shù),耗時 527728 us;

大概是 3 倍左右的差距。當(dāng)然,在你的電腦上可能會得到不同的結(jié)果,這與系統(tǒng)的負(fù)載等有關(guān)系,可以多測試幾次。

四、測試2:混合格式化字符串和數(shù)字

看起來使用自己寫的 Long2String 函數(shù)執(zhí)行速度更快一些,但是它有一個弊端,就是只能格式化數(shù)字。

如果我們需要把字符串和數(shù)字一起格式化成一個字符串,應(yīng)該如何處理?

如果使用 sprintf 庫函數(shù),那非常方便:

sprintf(buff, "%s%d", "hello", 123456);

如果繼續(xù)使用 Long2String 函數(shù),那么就要分步來格式化,例如:

// 拆成 2 個步驟sprintf(buff, "%s", "hello");Long2String(buff + strlen(buff), 123456);

以上兩種方式都能達(dá)到目的,那執(zhí)行效率如何呢?繼續(xù)測試:

int main(){ printf("long size = %d, LONG_MAX = %ld ", sizeof(long), LONG_MAX); // 測試 1000 萬 次 const char *prefix = "ZhangSan has money: "; int total = 1000 * 10000; char buff1[32] = { 0 }; char buff2[32] = { 0 }; // 測試 sprintf long long start1 = getSysTimestamp(); for (int i = 0; i < total; ++i) sprintf(buff1, "%s%ld", prefix, LONG_MAX); printf("sprintf ellapse: %lld us ", getSysTimestamp() - start1); // 測試 Long2String long long start2 = getSysTimestamp(); for (int i = 0; i < total; ++i) { sprintf(buff2, "%s", prefix); Long2String(buff2 + strlen(prefix), LONG_MAX); } printf("Long2String ellapse: %lld us ", getSysTimestamp() - start2); return 0;}

執(zhí)行結(jié)果對比:

long size = 4, LONG_MAX = 2147483647sprintf ellapse: 2477686 us Long2String ellapse: 816119 us

執(zhí)行速度仍然是 3 倍左右的差距。就是說,即使拆分成多個步驟來執(zhí)行,使用 Long2String 函數(shù)也會更快一些!

五、sprintf 的實(shí)現(xiàn)機(jī)制

sprintf 函數(shù)家族中,存在著一系列的函數(shù),其底層是通過可變參數(shù)來實(shí)現(xiàn)的。之前寫過一篇文章一個printf(結(jié)構(gòu)體指針)引發(fā)的血案,其中的第四部分,使用圖片詳細(xì)描述了可變參數(shù)的實(shí)現(xiàn)原理,摘抄如下。

1. 可變參數(shù)的幾個宏定義

typedef char * va_list; #define va_start _crt_va_start#define va_arg _crt_va_arg #define va_end _crt_va_end #define _crt_va_start(ap,v) ( ap = (va_list)_ADDRESSOF(v) + _INTSIZEOF(v) ) #define _crt_va_arg(ap,t) ( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) ) #define _crt_va_end(ap) ( ap = (va_list)0 )

注意:va_list 就是一個 char* 型指針。

2. 可變參數(shù)的處理過程

我們以剛才的示例 my_printf_int 函數(shù)為例,重新貼一下:

void my_printf_int(int num, ...) // step1{ int i, val; va_list arg; va_start(arg, num); // step2 for(i = 0; i < num; i++) { val = va_arg(arg, int); // step3 printf("%d ", val); } va_end(arg); // step4 printf(" ");} int main(){ int a = 1, b = 2, c = 3; my_printf_int(3, a, b, c);}

Step1: 函數(shù)調(diào)用時

C語言中函數(shù)調(diào)用時,參數(shù)是從右到左、逐個壓入到棧中的,因此在進(jìn)入 my_printf_int 的函數(shù)體中時,棧中的布局如下:

e34592a8-a95f-11eb-9728-12bb97331649.png

Step2: 執(zhí)行 va_start

va_start(arg, num);

把上面這語句,帶入下面這宏定義:

#define_crt_va_start(ap,v)(ap=(va_list)_ADDRESSOF(v)+_INTSIZEOF(v))
宏擴(kuò)展之后得到:

arg = (char *)num + sizeof(num);

結(jié)合下面的圖來分析一下:首先通過 _ADDRESSOF 得到 num 的地址 0x01020300,然后強(qiáng)轉(zhuǎn)成 char* 類型,再然后加上 num 占據(jù)的字節(jié)數(shù)(4個字節(jié)),得到地址 0x01020304,最后把這個地址賦值給 arg,因此 arg 這個指針就指向了棧中數(shù)字 1 的那個地址,也就是第一個參數(shù),如下圖所示:

e36fc1d6-a95f-11eb-9728-12bb97331649.png
Step3: 執(zhí)行 va_arg

val = va_arg(arg, int);

把上面這語句,帶入下面這宏定義:

#define_crt_va_arg(ap,t)(*(t*)((ap+=_INTSIZEOF(t))-_INTSIZEOF(t)))
宏擴(kuò)展之后得到:

val = ( *(int *)((arg += _INTSIZEOF(int)) - _INTSIZEOF(int)) )

結(jié)合下面的圖來分析一下:先把 arg 自增 int 型數(shù)據(jù)的大小(4個字節(jié)),使得 arg = 0x01020308;然后再把這個地址(0x01020308)減去4個字節(jié),得到的地址(0x01020304)里的這個值,強(qiáng)轉(zhuǎn)成 int 型,賦值給 val,如下圖所示:

e3be21f0-a95f-11eb-9728-12bb97331649.png

簡單理解,其實(shí)也就是:得到當(dāng)前 arg 指向的 int 數(shù)據(jù),然后把 arg 指向位于高地址處的下一個參數(shù)位置。

va_arg 可以反復(fù)調(diào)用,直到獲取棧中所有的函數(shù)傳入的參數(shù)。

Step4: 執(zhí)行 va_end

va_end(arg);

把上面這語句,帶入下面這宏定義:

#define _crt_va_end(ap) ( ap = (va_list)0 )

宏擴(kuò)展之后得到:

arg = (char *)0;

這就好理解了,直接把指針 arg 設(shè)置為空。因?yàn)闂V械乃袆討B(tài)參數(shù)被提取后,arg 的值為 0x01020310(最后一個參數(shù)的上一個地址),如果不設(shè)置為 NULL 的話,下面使用的話就得到未知的結(jié)果,為了防止誤操作,需要設(shè)置為NULL。

六、總結(jié)

這篇文章描述的格式化方法靈活性不太好,也許存在一定的局限性。但是在一些關(guān)鍵場景下,能明顯提高執(zhí)行效率。

如果文中演示代碼有什么問題,或者你有更好的方法,歡迎分享給大家!

編輯:jq

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • C語言
    +關(guān)注

    關(guān)注

    180

    文章

    7629

    瀏覽量

    140143
  • 函數(shù)
    +關(guān)注

    關(guān)注

    3

    文章

    4367

    瀏覽量

    64076
  • 代碼
    +關(guān)注

    關(guān)注

    30

    文章

    4886

    瀏覽量

    70206

原文標(biāo)題:天下武功,唯快不破:提升字符串格式化效率的小技巧

文章出處:【微信號:zhuyandz,微信公眾號:FPGA之家】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。

收藏 人收藏

    評論

    相關(guān)推薦
    熱點(diǎn)推薦

    字符串在數(shù)據(jù)庫中的存儲方式

    固定長度的空間,適合存儲長度變化不大的字符串。 可變長度字符串 :如VARCHAR類型,它根據(jù)字符串的實(shí)際長度動態(tài)分配空間,適合存儲長度變化較大的字符串。 文本類型 :如TEXT類型,
    的頭像 發(fā)表于 01-07 15:41 ?663次閱讀

    字符串在編程中的應(yīng)用實(shí)例

    字符串在編程中有著廣泛的應(yīng)用,它們被用于表示文本數(shù)據(jù)、處理用戶輸入、構(gòu)建動態(tài)內(nèi)容等。以下是一些字符串在編程中的應(yīng)用實(shí)例: 1. 用戶輸入與輸出 用戶輸入 :程序通常需要從用戶那里獲取輸入,這些輸入通
    的頭像 發(fā)表于 01-07 15:33 ?573次閱讀

    字符串字符數(shù)組的區(qū)別

    在編程語言中,字符串字符數(shù)組是兩種基本的數(shù)據(jù)結(jié)構(gòu),它們都用于存儲和處理文本數(shù)據(jù)。盡管它們在功能上有一定的重疊,但在內(nèi)部表示、操作方式和使用場景上存在顯著差異。 1. 內(nèi)部表示 字符串 字符串
    的頭像 發(fā)表于 01-07 15:29 ?929次閱讀

    字符串反轉(zhuǎn)的實(shí)現(xiàn)方式

    在編程中,字符串反轉(zhuǎn)是一個基礎(chǔ)而重要的操作,它涉及到將一個字符串中的字符順序顛倒過來。這個操作在多種編程語言中都有不同的實(shí)現(xiàn)方式,本文將探討幾種常見的字符串反轉(zhuǎn)方法。 1. 遞歸方法
    的頭像 發(fā)表于 01-07 15:27 ?665次閱讀

    字符串處理方法 字符串轉(zhuǎn)數(shù)字的實(shí)現(xiàn)

    在編程中,將字符串轉(zhuǎn)換為數(shù)字是一個常見的需求。不同的編程語言有不同的方法來實(shí)現(xiàn)這一功能。以下是一些常見編程語言中的字符串轉(zhuǎn)數(shù)字的實(shí)現(xiàn)方法: Python 在Python中,可以使用內(nèi)置的 int
    的頭像 發(fā)表于 01-07 15:26 ?698次閱讀

    FB16格式化功能介紹#硬核拆解 #佑華硬盤拷貝機(jī)#格式化#自動執(zhí)行

    格式化
    華佳興科技
    發(fā)布于 :2024年11月21日 16:04:30

    base64字符串轉(zhuǎn)換為二進(jìn)制文件

    Base64是一種編碼方法,用于將二進(jìn)制數(shù)據(jù)轉(zhuǎn)換為ASCII字符串。這種編碼通常用于在不支持二進(jìn)制數(shù)據(jù)的系統(tǒng)中傳輸數(shù)據(jù),例如電子郵件或網(wǎng)頁。將Base64字符串轉(zhuǎn)換為二進(jìn)制文件的過程相對簡單,但需要
    的頭像 發(fā)表于 11-10 10:55 ?2610次閱讀

    Labview 字符串 不常用功能:快捷才當(dāng)將字符串設(shè)置為反斜杠(\'\\\')代碼顯示,用于儀器通信

    (ASCIIFF,相當(dāng)于**\\\\08**) \\\\n 換行(ASCIILF,相當(dāng)于**\\\\0A**)。格式化寫入文件函數(shù)自動將此代碼轉(zhuǎn)換為獨(dú)立于平臺的行結(jié)束字符。 \\\\r 回車
    發(fā)表于 09-06 14:01

    MATLAB(5)--字符串處理

    字符串表示 在MATLAB中,字符串是用單引號括起來的字符序列,是把一個字符串當(dāng)做一個行向量,這個行向量中,每個元素對應(yīng)一個字符。 若
    發(fā)表于 09-06 10:22

    labview字符串數(shù)組轉(zhuǎn)化為數(shù)值數(shù)組

    在LabVIEW中,將字符串數(shù)組轉(zhuǎn)換為數(shù)值數(shù)組是一項常見的任務(wù),尤其是在處理數(shù)據(jù)采集、信號處理或用戶輸入時。 1. 理解LabVIEW的數(shù)據(jù)類型 在開始之前,了解LabVIEW中的數(shù)據(jù)類型是非
    的頭像 發(fā)表于 09-04 17:47 ?4675次閱讀

    labview字符串如何轉(zhuǎn)換為16進(jìn)制字符串

    在LabVIEW中,將字符串轉(zhuǎn)換為16進(jìn)制字符串是一個常見的需求,尤其是在處理數(shù)據(jù)通信和硬件接口時。LabVIEW提供了多種方法來實(shí)現(xiàn)這一轉(zhuǎn)換,包括使用內(nèi)置函數(shù)、編寫VI(Virtual
    的頭像 發(fā)表于 09-04 15:54 ?4521次閱讀

    labview中如何實(shí)現(xiàn)字符串換行

    。 使用 Append String 函數(shù) :在字符串末尾添加換行符。 使用 Format Into String 函數(shù) :格式化
    的頭像 發(fā)表于 09-04 15:47 ?3312次閱讀

    labview中如何實(shí)現(xiàn)字符串選擇輸出

    在LabVIEW中實(shí)現(xiàn)字符串選擇輸出是一項常見的任務(wù),它涉及到字符串處理、條件判斷和用戶界面設(shè)計等多個方面。由于LabVIEW是一種圖形編程語言,其編程方式與傳統(tǒng)的文本編程語言有所不同,因此實(shí)現(xiàn)
    的頭像 發(fā)表于 09-04 15:44 ?1946次閱讀

    labview中常用的字符串函數(shù)有哪些?

    在LabVIEW中,常用的字符串函數(shù)廣泛覆蓋了對字符串的各種操作,包括但不限于格式化、搜索、替換、連接、計算長度等。以下是一些常用的字符串函數(shù)及其簡要說明:
    的頭像 發(fā)表于 09-04 15:43 ?1605次閱讀

    labview字符串的四種表示各有什么特點(diǎn)

    。在LabVIEW中,字符串是一種基本的數(shù)據(jù)類型,用于表示文本信息。字符串在LabVIEW中有多種表示方式,每種方式都有其特定的應(yīng)用場景和特點(diǎn)。以下是對LabVIEW中四種字符串表示方式的分析: 1.
    的頭像 發(fā)表于 09-04 15:40 ?1196次閱讀
    主站蜘蛛池模板: 欧美色视频日本片高清在线观看 | 粗又长好猛好爽 | 两性色午夜视频免费网 | 男人的天堂在线免费视频 | 逼逼视频网站 | 免费一区二区三区 | 年轻护士女三级 | 免费被视频网站在线观看 | 中国一级黄色毛片 | 中文网丁香综合网 | 欧美在线小视频 | 婷婷丁香激情 | 五月综合激情久久婷婷 | 久久综合色婷婷 | 天天天天天天操 | 老湿影院免费体验区 | 久久久久久久蜜桃 | 一级视频在线观看 | 婷婷激情四射网 | 亚洲аv电影天堂网 | 男人不识本网站上遍色站也枉然 | 亚洲 欧美 日韩 综合 | аⅴ资源中文在线天堂 | 亚洲成人网在线播放 | 欧美精品福利 | h网站在线免费观看 | 亚洲精品网站日本xxxxxxx | 亚洲无色 | 色综合色综合色综合网址 | 永久黄网站色视频免费观看 | 亚洲国产精品丝袜在线观看 | 一级特黄aa大片 | 免费日韩网站 | 成人av电影在线 | 国产色爽女小说免费看 | 欧美午夜影视 | 欧美在线视频一区二区三区 | 人人洗澡人人洗澡人人 | 91亚色视频在线观看 | 99久久99久久精品免费看子伦 | 亚洲精品午夜久久aaa级久久久 |