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

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

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

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

如何把數(shù)字格式化成字符串

FPGA之家 ? 來(lái)源:FPGA之家 ? 作者:FPGA之家 ? 2022-08-04 10:35 ? 次閱讀

一、前言

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

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

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

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

六、總結(jié)

一、前言

嵌入式項(xiàng)目開(kāi)發(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ù)字的后面開(kāi)始,把每一個(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í),棧中的布局如下:

89992b18-138d-11ed-ba43-dac502259ad0.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ù),如下圖所示:

89b483f4-138d-11ed-ba43-dac502259ad0.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,如下圖所示:

89c5583c-138d-11ed-ba43-dac502259ad0.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í)行效率。

審核編輯:彭靜

聲明:本文內(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)投訴
  • 格式化
    +關(guān)注

    關(guān)注

    2

    文章

    39

    瀏覽量

    9152
  • 字符串
    +關(guān)注

    關(guān)注

    1

    文章

    585

    瀏覽量

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

    關(guān)注

    3

    文章

    4346

    瀏覽量

    63015
收藏 人收藏

    評(píng)論

    相關(guān)推薦

    C++字符串string

    string是C++編程語(yǔ)言中的字符串。在C++中字符串處理可以使用c語(yǔ)言字符串形式char *,也可以使用string類格式
    的頭像 發(fā)表于 07-10 00:26 ?1397次閱讀
    C++<b class='flag-5'>字符串</b>string

    LABVIEW中的printf函數(shù)---格式化寫入字符串函數(shù)

    `我們經(jīng)常會(huì)遇到格式化字符串輸出的問(wèn)題,參見(jiàn)下面的程序框圖。在C語(yǔ)言中,類似功能的函數(shù)是printf函數(shù),該函數(shù)幾乎是C語(yǔ)言經(jīng)典課程的第一節(jié)內(nèi)容,可見(jiàn)其用途之廣泛。C語(yǔ)言中實(shí)現(xiàn)上面程序框圖的功能非常
    發(fā)表于 11-15 10:46

    Labview格式化寫入字符串函數(shù)

    Labview格式化寫入字符串函數(shù)
    發(fā)表于 12-26 12:42

    格式化寫入字符串函數(shù) 求助

    格式化寫入字符串函數(shù)格式字符串:%6f\n%s\n%d\n%s這個(gè)怎么理解?
    發(fā)表于 07-13 13:23

    C語(yǔ)言技巧 sprintf()函數(shù):將格式化的數(shù)據(jù)寫入字符串

    [table][tr][td]頭文件#include 功能用于將格式化的數(shù)據(jù)寫入字符串。原型int sprintf(char *str, char * format [, argument
    發(fā)表于 04-01 11:26

    2.6 python字符串格式化

    2.6 python字符串格式化格式化輸出,主要有三種方式使用 % 進(jìn)行格式化使用 format 函數(shù)進(jìn)行格式化使用 f-string 進(jìn)行
    發(fā)表于 02-21 16:28

    AVR入門:怎么樣使用AVR LibC's Stdio發(fā)送格式化字符串

    AVR入門:使用AVR LibC's Stdio發(fā)送格式化字符串 (#21)
    的頭像 發(fā)表于 07-09 00:02 ?3080次閱讀
    AVR入門:怎么樣使用AVR LibC's Stdio發(fā)送<b class='flag-5'>格式化</b>的<b class='flag-5'>字符串</b>?

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

    一、前言 在嵌入式項(xiàng)目開(kāi)發(fā)中,字符串格式化是很常見(jiàn)的操作,我們一般都會(huì)使用 C 庫(kù)中的 sprintf 系列函數(shù)來(lái)完成格式化。 從功能上來(lái)說(shuō),這是沒(méi)有問(wèn)題的,但是在一些時(shí)間關(guān)鍵場(chǎng)合,字符串
    的頭像 發(fā)表于 04-30 13:43 ?1657次閱讀
    剖析提升<b class='flag-5'>字符串</b><b class='flag-5'>格式化</b>效率的小技巧

    strtok拆分字符串

    大家好,我是驚覺(jué),今天聊聊字符串字符串的使用場(chǎng)景非常之多,人機(jī)交互和雙機(jī)通信都會(huì)用到。比如:通過(guò)串口向單片機(jī)發(fā)送指令,以執(zhí)行操作或配置參數(shù)。單片機(jī)讀取傳感器數(shù)據(jù),數(shù)據(jù)格式字符串。一
    發(fā)表于 01-13 15:46 ?8次下載
    strtok拆分<b class='flag-5'>字符串</b>

    python字符串格式化

    python字符串格式化 格式化輸出,主要有三種方式 使用 % 進(jìn)行格式化 使用 format 函數(shù)進(jìn)行格式化 使用 f-string 進(jìn)行
    的頭像 發(fā)表于 02-21 16:28 ?1640次閱讀
    python<b class='flag-5'>字符串</b><b class='flag-5'>格式化</b>

    格式化字符串常量f-strings一些不常見(jiàn)的特性

    【導(dǎo)語(yǔ)】:本文介紹了“格式化字符串常量”f-strings一些不常見(jiàn)的特性,包括:格式化日期和時(shí)間、同時(shí)打印變量名和變量值等,嵌套使用f-strings等,此外與其他格式化
    的頭像 發(fā)表于 08-15 12:00 ?1207次閱讀

    字符串格式化輸入和輸出

    字符串是內(nèi)存中一段連續(xù)的char空間,以’\0’(數(shù)字0)結(jié)尾。
    的頭像 發(fā)表于 02-24 14:34 ?1495次閱讀
    <b class='flag-5'>字符串</b>的<b class='flag-5'>格式化</b>輸入和輸出

    java字符串轉(zhuǎn)化為日期格式

    在Java中,字符串轉(zhuǎn)化為日期格式是一個(gè)常見(jiàn)的需求。日期格式在處理時(shí)間相關(guān)的操作時(shí)非常重要,它可以用來(lái)表示一段時(shí)間的開(kāi)始和結(jié)束,也可以用來(lái)計(jì)算時(shí)間差等。本文將詳細(xì)介紹如何將一個(gè)字符串轉(zhuǎn)
    的頭像 發(fā)表于 11-17 16:38 ?3108次閱讀

    python怎么字符串變成數(shù)字

    Python是目前廣泛應(yīng)用的一種編程語(yǔ)言,它以簡(jiǎn)潔、易讀和靈活的特性被廣大開(kāi)發(fā)者所喜愛(ài)。在Python中,字符串數(shù)字是兩種常見(jiàn)的數(shù)據(jù)類型。字符串是由字符組成的一系列
    的頭像 發(fā)表于 11-22 09:47 ?3933次閱讀

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

    在LabVIEW中,常用的字符串函數(shù)廣泛覆蓋了對(duì)字符串的各種操作,包括但不限于格式化、搜索、替換、連接、計(jì)算長(zhǎng)度等。以下是一些常用的字符串函數(shù)及其簡(jiǎn)要說(shuō)明:
    的頭像 發(fā)表于 09-04 15:43 ?1008次閱讀
    主站蜘蛛池模板: 手机看片久久 | 国产农村女人一级毛片了 | 日本四虎影院 | 在线观看黄色一级片 | 亚洲精品成人久久久影院 | 两性色午夜视频免费播放 | 琪琪see色原在线20 | 永久黄网站色视频免费观看 | 女同性进行性行为视频 | 久久久久免费精品国产小说 | 禁h粗大太大好爽好涨受不了了 | 久久影视一区 | 一区精品视频 | 亚色国产| 成人a毛片手机免费播放 | 久久精品免费观看视频 | 天天曰天天爽 | 欧美成人eee在线 | 九九热精品在线 | 伊人网综合在线观看 | 欧美肥妇性| 欧美一区二区三区不卡片 | 久99热 | 天天操天天干天天玩 | 免费看大尺度视频在线观看 | 午夜高清福利 | 人人澡人人人人夜夜爽 | 深夜视频在线免费 | 在线a免费观看最新网站 | 黄录像欧美片在线观看 | 国产高清a | 99热精品久久只有精品30 | 69xx女xo69| 美日韩免费视频 | 国产巨大bbbb天美 | 2022年国产精品久久久久 | 欧美三级精品 | 亚洲综合春色另类久久 | 四虎永久在线精品国产免费 | www.狠狠操.com| 亚洲涩综合 |