91欧美超碰AV自拍|国产成年人性爱视频免费看|亚洲 日韩 欧美一厂二区入|人人看人人爽人人操aV|丝袜美腿视频一区二区在线看|人人操人人爽人人爱|婷婷五月天超碰|97色色欧美亚州A√|另类A√无码精品一级av|欧美特级日韩特级

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

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

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

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

FPGA之家 ? 來(lái)源: IOT物聯(lián)網(wǎng)小鎮(zhèn) ? 作者:道哥 ? 2021-04-30 13:43 ? 次閱讀
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

一、前言

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

從功能上來(lái)說(shuō),這是沒(méi)有問(wèn)題的,但是在一些時(shí)間關(guān)鍵場(chǎng)合,字符串的格式化效率會(huì)對(duì)整個(gè)系統(tǒng)產(chǎn)生顯著的影響。

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

天下武功,唯快不破!

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

二、最簡(jiǎn)單的格式化

#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é)束了,但是如果是一百萬(wàn)、一千萬(wàn)次呢?

三、測(cè)試1:手動(dòng)格式化數(shù)字

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

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

為了測(cè)試代碼執(zhí)行的耗時(shí),我們寫一個(gè)簡(jiǎn)單的函數(shù):獲取系統(tǒng)的時(shí)間戳,通過(guò)計(jì)算時(shí)間差值來(lái)看一下代碼的執(zhí)行速度。

// 獲取系統(tǒng)時(shí)間戳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: 格式化之后字符串存儲(chǔ)地址;// value: 待格式化的數(shù)字void Long2String(char *buff, long value){ long tmp; char tmpBuf[32] = { 0 }; // p 指向臨時(shí)數(shù)組的最后一個(gè)位置 char *p = &tmpBuf[sizeof(tmpBuf) - 1]; while (value != 0) { tmp = value / 10; // 把一個(gè)數(shù)字轉(zhuǎn)成 ASCII 碼,放到 p 指向的位置。 // 然后 p 往前移動(dòng)一個(gè)位置。 *--p = (char)('0' + (value - tmp * 10)); value = tmp; } // 把臨時(shí)數(shù)組中的每個(gè)字符,復(fù)制到 buff 中。 while (*p) *buff++ = *p++;}

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

3. 測(cè)試代碼

int main(){ printf("long size = %d, LONG_MAX = %ld ", sizeof(long), LONG_MAX); // 測(cè)試 1000 萬(wàn)次 int total = 1000 * 10000; char buff1[32] = { 0 }; char buff2[32] = { 0 }; // 測(cè)試 sprintf long long start1 = getSysTimestamp(); for (int i = 0; i < total; ++i) sprintf(buff1, "%ld", LONG_MAX); printf("sprintf ellapse: %lld us ", getSysTimestamp() - start1); // 測(cè)試 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é)果對(duì)比

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

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

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

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

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

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

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

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

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

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

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

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

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

int main(){ printf("long size = %d, LONG_MAX = %ld ", sizeof(long), LONG_MAX); // 測(cè)試 1000 萬(wàn) 次 const char *prefix = "ZhangSan has money: "; int total = 1000 * 10000; char buff1[32] = { 0 }; char buff2[32] = { 0 }; // 測(cè)試 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); // 測(cè)試 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é)果對(duì)比:

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

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

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

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

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

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 就是一個(gè) char* 型指針。

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

我們以剛才的示例 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)用時(shí)

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

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

Step2: 執(zhí)行 va_start

va_start(arg, num);

把上面這語(yǔ)句,帶入下面這宏定義:

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

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

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

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

val = va_arg(arg, int);

把上面這語(yǔ)句,帶入下面這宏定義:

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

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

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

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

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

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

Step4: 執(zhí)行 va_end

va_end(arg);

把上面這語(yǔ)句,帶入下面這宏定義:

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

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

arg = (char *)0;

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

六、總結(jié)

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

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

編輯:jq

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

    關(guān)注

    183

    文章

    7644

    瀏覽量

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

    關(guān)注

    3

    文章

    4417

    瀏覽量

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

    關(guān)注

    30

    文章

    4967

    瀏覽量

    73954

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

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

收藏 人收藏
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

    評(píng)論

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

    求助 LabVIEW 字符串比較

    1.輸入一個(gè)字符串,儲(chǔ)存起來(lái)。 2.再次輸入一個(gè)字符串,先和儲(chǔ)存起來(lái)的字符串比較,如果不同則存儲(chǔ)起來(lái),如果相同則不儲(chǔ)存。 3.再次輸入一個(gè)字符串,和所有儲(chǔ)存起來(lái)的
    發(fā)表于 03-02 17:24

    打開工程后工程中的字體沒(méi)有顯示,如字符串,數(shù)字等控件不能預(yù)覽顯示字體?

    打開工程后工程中的字體沒(méi)有顯示,如字符串,數(shù)字等控件不能預(yù)覽顯示字體?
    發(fā)表于 02-25 17:39

    華為鴻蒙系統(tǒng)應(yīng)用數(shù)字與度量衡的格式化指南

    本文旨在深入探討華為鴻蒙 HarmonyOS 系統(tǒng)在應(yīng)用國(guó)際中數(shù)字與度量衡格式化方面的技術(shù)細(xì)節(jié),基于實(shí)際開發(fā)實(shí)踐進(jìn)行總結(jié)。
    的頭像 發(fā)表于 01-24 16:55 ?1171次閱讀

    字符串控件與靜態(tài)字符串控件中預(yù)覽字符顯示亂碼,如何修改顯示正常?

    字符串控件與靜態(tài)字符串控件中預(yù)覽字符顯示亂碼,如何修改顯示正常?
    發(fā)表于 01-20 17:17

    字符串,數(shù)字控件如何控制背景顏色和前景字體顏色?

    字符串,數(shù)字控件如何控制背景顏色和前景字體顏色?
    發(fā)表于 01-20 15:12

    Linux下怎么讓中文字符串按照拼音排序?

    求教 Linux 下怎么讓中文字符串按照拼音排序?
    發(fā)表于 01-06 07:40

    字符串關(guān)聯(lián)數(shù)字變量如何使用?我們的地址都是16位數(shù)據(jù),可以使用16位數(shù)字變量顯示字符串嗎?

    字符串關(guān)聯(lián)數(shù)字變量如何使用?我們的地址都是16位數(shù)據(jù),可以使用16位數(shù)字變量顯示字符串嗎?
    發(fā)表于 12-15 08:24

    飛凌嵌入式ElfBoard-標(biāo)準(zhǔn)IO接口之格式化輸入

    格式化輸入用于從不同輸入源中獲取數(shù)據(jù)并根據(jù)格式化字符串format轉(zhuǎn)換為對(duì)應(yīng)的格式代碼并存儲(chǔ)在對(duì)應(yīng)的類型中。格式化輸入函數(shù):即按特定的
    發(fā)表于 11-12 08:35

    飛凌嵌入式ElfBoard-標(biāo)準(zhǔn)IO接口之格式化輸出

    ( const char *format, ... );3)參數(shù)format:表示C 字符串,包含了要打印的格式化數(shù)據(jù)。...:表示附加可變參數(shù),根據(jù)不同的 format 字符串,函數(shù)可能需要一系列的附加
    發(fā)表于 11-11 08:43

    求助,關(guān)于使用sprintf函數(shù)格式化浮點(diǎn)數(shù)的安全問(wèn)題求解

    (); return n; } 經(jīng)過(guò)測(cè)試,未關(guān)閉調(diào)度之前,線程被打斷再恢復(fù)之后,格式化的數(shù)據(jù)可能是一些亂碼,甚至程序跑飛了.如果使用這個(gè)字符串很容易死機(jī),主要是data abort異常.加入關(guān)閉調(diào)度之后,能夠得到
    發(fā)表于 10-09 08:22

    labview如何生成一個(gè)帶字符串返回的dll

    labview如何生成一個(gè)dll,如下圖,要求一個(gè)輸入,類型是字符串,返回類型也是字符串
    發(fā)表于 08-28 23:20

    在Python中字符串逆序有幾種方式,代碼是什么

    對(duì)于一個(gè)給定的字符串,逆序輸出,這個(gè)任務(wù)對(duì)于python來(lái)說(shuō)是一種很簡(jiǎn)單的操作,畢竟強(qiáng)大的列表和字符串處理的一些列函數(shù)足以應(yīng)付這些問(wèn)題 了,今天總結(jié)了一下python中對(duì)于字符串的逆序輸出的幾種常用
    的頭像 發(fā)表于 08-28 14:44 ?1082次閱讀

    harmony-utils之FormatUtil,格式化工具類

    harmony-utils之FormatUtil,格式化工具類
    的頭像 發(fā)表于 07-03 18:22 ?534次閱讀

    harmony-utils之StrUtil,字符串工具類

    harmony-utils之StrUtil,字符串工具類 harmony-utils 簡(jiǎn)介與說(shuō)明 [harmony-utils] 一款功能豐富且極易上手的HarmonyOS工具庫(kù),借助眾多實(shí)用工具類
    的頭像 發(fā)表于 07-03 11:32 ?632次閱讀

    通過(guò)FATFS文件系統(tǒng)讀寫SD卡創(chuàng)建文件可以了,加入MSC一直顯示一個(gè)沒(méi)有格式化的U盤盤符,也不能格式化,為什么?

    通過(guò)FATFS文件系統(tǒng)讀寫SD卡創(chuàng)建文件可以了,但加入MSC,一直顯示一個(gè)沒(méi)有格式化的U盤盤符,也不能格式化,這是什么問(wèn)題導(dǎo)致的
    發(fā)表于 03-12 07:20