伦伦影院久久影视,天天操天天干天天射,ririsao久久精品一区 ,一本大道香蕉大久在红桃,999久久久免费精品国产色夜,色悠悠久久综合88,亚洲国产精品久久无套麻豆,亚洲香蕉毛片久久网站,一本一道久久综合狠狠老

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

關于你可能不知道的printf

黃工的嵌入式技術圈 ? 來源:黃工的嵌入式技術圈 ? 作者:黃工的嵌入式技術 ? 2020-02-05 12:28 ? 次閱讀
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

前言

printf可能是我們在學習C語言的過程中最早接觸的庫函數了。其基本使用想必我們都已經非常清楚了。但是下面的這些情況你是否已經清楚地知道了呢?

示例程序

我們來看一個示例程序,看看你能否對下面的結果輸出有非常清晰的認識。

#include intmain(void) { inta=4; intb=3; intc=a/b; floatd=*(float*)(&c); longlonge=0xffffffffffffffff; printf("a/b:%f,a:%d\n",a/b,a,b);//打印0 printf("(float)a/b:%f\n",((float)a)/b);//打印1 printf("(double)a/b:%lf\n",((double)a)/b);//打印2 printf("d:%f\n",d);//打印3 printf("%.*f\n",20,(double)a/b);//打印4 printf("e:%d,a:%d\n",e,a);//打印5 printf("a:%d,++a:%d,a++:%d\n",a,++a,a++);//打印6 return0; }

編譯為32位程序:

gcc-m32-otesttest.c

在運行之前,你可以自己先猜想一下打印結果會是什么。實際運行結果:

a/b:0.000000,a:3//打印0的結果 (float)a/b:1.333333//打印1的結果 (double)a/b:1.333333//打印2的結果 d:0.000000//打印3的結果 1.33333333333333325932//打印4的結果 e:-1,a:-1//打印5的結果 a:6,++a:6,a++:4//打印6的結果

你的猜想是否都正確呢?如果猜想錯誤,那么接下來的內容你就不應該錯過了。

你是否會有以下疑問:

0.打印0的a/b為什么不是1,a為什么不是4?

1.打印1和打印2有什么區別呢?

2.打印3為什么結果會是0.000000?

3.打印4的結果為什么最后的小數位不對?其中的*是什么意思?

4.打印5中,為什么a的值是-1而不是4?

5.打印6中,結果為什么分別是6,6,4?

在解答這些問題之前,我們需要先了解一些基本內容。

可變參數中的類型提升

printf是接受變長參數的函數,傳入printf中的參數個數可以不定。而我們在變長參數探究中說到:
調用者會對每個參數執行“默認實際參數提升",提升規則如下:
——float將提升到double
——char、short和相應的signed、unsigned類型將提升到int

也就是說printf實際上只會接受到double,int,long int等類型的參數。而從來不會實際接受到float,char,short等類型參數。
我們可以通過一個示例程序來檢驗:

//badcode #include intmain(void) { char*p=NULL; printf("%d,%f,%c\n",p,p,p); return0; }

編譯報錯如下:

printf.c:Infunction‘main’: printf.c:5:12:warning:format‘%d’expectsargumentoftype‘int’,butargument2hastype‘char*’[-Wformat=] printf("%d,%f,%c\n",p,p,p); ^ printf.c:5:12:warning:format‘%f’expectsargumentoftype‘double’,butargument3hastype‘char*’[-Wformat=] printf.c:5:12:warning:format‘%c’expectsargumentoftype‘int’,butargument4hastype‘char*’[-Wformat=]

我們可以從報錯信息中看到:

%d 期望的是 int 類型參數

%f 期望的是 double 類型參數

%c 期望的也是 int 類型參數

而編譯之所以有警告是因為,char *類型無法通過默認實際參數提升,將其提升為int或double。

參數入棧順序以及計算順序

在C語言中,參數入棧順序是確定的,從右往左。而參數的計算順序卻是沒有規定的。也就是說,編譯器可以實現從右往左計算,也可以實現從左往右計算。

浮點數的有效位

對于double類型,其有效位為15~~16位(參考:對浮點數的一些理解)。

可變域寬和精度

printf中,*的使用可實現可變域寬和精度,使用時只需要用*替換域寬修飾符和精度修飾符即可。在這樣的情況下,printf會從參數列表中取用實際值作為域寬或者精度。示例程序如下:

#include intmain(void) { floata=1.33333333; char*p="hello"; printf("%.*f\n",6,a); printf("%*s\n",8,p); return0; }

運行結果:

1.333333 hello

而這里的6或者8完全可以是一個宏定義或者變量,從而做到了動態地格式控制。

格式控制符是如何處理參數的

printf有很多格式控制符,例如%d,它在處理輸入時,會從堆棧中取其對應大小,即4個字節作為對應的參數值。也就是說,當你傳入參數和格式控制符匹配或者在經過類型提升后和格式控制符匹配的時候,參數處理是沒有任何問題的。但是不匹配時,可能會出現未定義行為(有兩種情況例外,我們后面再說)。例如,%f期望一個double(8字節)類型,但是傳入的參數是int(4字節),那么在處理這個int參數值,可能會多處理4個字節,并且也會造成處理數據錯誤。

真相大白

有了前面這些內容的鋪墊,我們再來解答開始的疑問:

對于問題0,a/b的結果顯然為4字節的int類型1,而%f期望的是8字節的double,而計算結果只有4個字節,因此會繼續格式化后面4個字節的a,而整型1和后面a組合成的8字節數據,按照浮點數的方式解釋時,它的值就是0.000000了。由于前面已經讀取解釋了a的內容,因此第二個%d只能繼續讀取4個字節,也就是b的值3,最終就會出現打印a的值是3,而不是4。

對于問題1,實際上在printf中,是不需要%lf的,%f期望的就是double類型,在編譯最開始的示例程序其實就可以發現這個事實。當然了在scanf函數中,這兩者是有區別的。

對于問題2,也很簡單,2的二進制存儲形式按照浮點數方式解釋讀取時,就是該值。

對于問題3,double的有效位為15~16位,也就是之外的位數都是不可靠的。printf中的*可用于實現可變域寬和精度,前面已經解釋過了。

對于問題4,這里不給出,留給讀者思考,歡迎大家可留言區給出原因。

對于問題5,雖然參數計算順序沒有規定,但是實際上至少對于gcc來說,它是從右往左計算的。也就是說,先計算a++,而a++是先用在加,即壓入a=4,其后,a的值變為5;再計算++a,先加再用,即壓入a=5+1=6;最后a=6,壓入棧。最終從左往右壓入棧的值就分別為6,6,4。也就是最終的打印結果。但是實際情況中,這樣的代碼絕對不該出現!

至此,真相大白。

總結

雖然我們前面解釋了那些難以理解的現象,同時讀者可以參考變長參數探究和對浮點數的一些理解找到更多的信息。但是我們在實際編程中應該注意以下幾點:

格式控制符應該與對應參數類型匹配或者與類型提升后的參數類型匹配。

絕對避免出現計算結果與參數計算順序有關的代碼。

*在printf中實現可變域寬和精度。

printf不會實際接受到char,short和float類型參數。

如果%s對應的參數可能為NULL或者對應整型,那將是一場災難。

不要忽略編譯器的任何警告,除非你很清楚你在做什么。

例外情況指的是有符號整型和無符號整型之間,以及void*和char*之間。

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • 打印
    +關注

    關注

    1

    文章

    67

    瀏覽量

    19301
  • 程序
    +關注

    關注

    117

    文章

    3846

    瀏覽量

    85363
  • Printf
    +關注

    關注

    0

    文章

    84

    瀏覽量

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

掃碼添加小助手

加入工程師交流群

    評論

    相關推薦
    熱點推薦

    PicoClaw v0.2.4 發布——迄今最大更新

    2026年3月25日PicoClaw團隊發布v0.2.4版本這次更新很大,大到我們自己都覺得,不說清楚可能不知道升級了什么所以這篇文章只說一件事:這次更新對意味著什么一、AI助手會"
    的頭像 發表于 03-26 12:04 ?242次閱讀
    PicoClaw v0.2.4 發布——迄今最大更新

    的EMC瓶頸,不是經驗不夠,是體系不全

    EMC的EMC瓶頸,不是經驗不夠,是體系不全現在的狀態,是不是這樣?產品送測前,心里完全沒底,不知道哪一項會掛傳導發射超標,換了十幾種電容電感,還是壓不下去靜電打幾下,系統重啟,
    的頭像 發表于 03-10 16:41 ?404次閱讀
    <b class='flag-5'>你</b>的EMC瓶頸,不是經驗不夠,是體系不全

    的導航早已“中國芯”:北斗如何靜默取代GPS,成為日常出行主力?

    四大全球衛星系統深度對比+實測驗證指南當你說“打開導航”時,知道背后是哪顆“星辰”在為指引方向嗎?今天,我們將深入探討全球衛星定位系統的格局,并揭示一個
    的頭像 發表于 02-05 19:33 ?771次閱讀
    <b class='flag-5'>你</b>的導航早已“中國芯”:北斗如何靜默取代GPS,成為日常出行主力?

    組合導航不知道怎么選,看這里

    單一導航難適配復雜作業?MEMS組合導航來幫你,通過融合衛星定位與慣性測量技術,以“1+1>2”優勢破解全行業導航難題。 我司組合導航提供三種方案可選,從01到03到05,搭載從導航級到戰術級到消費級MEMS陀螺儀與加速度計,通過微機電技術集成在毫米芯片上,搭載衛星模塊以及其他器件,尺寸僅有65mm*70mm*45.5mm,以下是三種方案詳細介紹: 超高精度方案—ER-GNSS/MINS-01 產品定位 一款擁有測繪級超高精度的組合導航,是市面上采用MEMS技術組合導航系統中
    的頭像 發表于 01-08 15:47 ?298次閱讀

    斷電時,的后備電源真能啟動嗎?多數人不知道的UPS保養真相

    斷電那一刻,的后備電源真的能頂上嗎?深夜,機房警報突然響起,屏幕瞬間熄滅——不是演習,市電真的斷了。所有人的心都提到了嗓子眼,三秒后,服務器指示燈重新亮起,UPS電源的顯示屏穩定地閃爍著運行
    的頭像 發表于 12-05 13:37 ?1864次閱讀
    斷電時,<b class='flag-5'>你</b>的后備電源真能啟動嗎?多數人<b class='flag-5'>不知道</b>的UPS保養真相

    使用MCU200T的板子能不能做RVSTAR上的實驗例子?

    手里只有一塊MCU200T的板子,看到老師做的RVSTAR上的例子,不知道能不能在MCU200T上面做
    發表于 11-05 12:55

    現在流行來中國看賽博朋克

    中國以外的人不知道中國有多強
    的頭像 發表于 10-10 22:43 ?4561次閱讀
    現在流行來中國看賽博朋克

    關于OFDM 不知道的那些事?#OFDM #5G技術 #通信技術

    通信技術
    安泰儀器維修
    發布于 :2025年06月24日 18:31:22

    編譯錯誤: error: unrecognizable insn,不知道原因,請教!

    用start_gui.exe生成的代碼,再用SEGGER Embedded打開,編譯有錯,不知道是什么原因,請教各位大佬?
    發表于 06-23 14:45

    瑞薩RA單片機在e2 studio環境下printf編譯出錯的問題解析

    最近看到有一些網友在討論關于:瑞薩RA單片機在e2 studio環境下printf編譯出錯的問題。
    的頭像 發表于 05-24 15:51 ?1650次閱讀
    瑞薩RA單片機在e2 studio環境下<b class='flag-5'>printf</b>編譯出錯的問題解析

    不知道怎么畫原理圖了

    時,這樣的問題,也有這么多?!1、電阻的表示方法是第一種,還是第二種?中間是方框還是折線?方框做多大?現場一片混亂立馬分成N派。普通的電阻都這樣,這么多種電阻現
    的頭像 發表于 04-30 18:40 ?1070次閱讀
    <b class='flag-5'>不知道</b>怎么畫原理圖了

    PCB設計仿真,“縫合電容”我怎么可能不知道

    。 案例1: 相信很多人都遇到走線跨分割地平面的情況,例如下面的模型所展示的: 大家都知道跨分割肯定對信號有影響了,那你們能想到的優化方案是什么呢!什么,告訴我不跨分割平面不就解決了嗎!要是能不
    發表于 04-28 15:44

    PCB設計仿真,“縫合電容”我怎么可能不知道

    說到“縫合電容”,雖然我已經聽你們說過800多遍了,但是還是忍不住問一個很簡單的問題:額,它到底是啥。。。
    的頭像 發表于 04-28 15:43 ?672次閱讀
    PCB設計仿真,“縫合電容”我怎么<b class='flag-5'>可能不知道</b>

    球壓試驗裝置:可能不知道的電氣安全衛士

    球壓試驗裝置,簡單來說,是一種用于評估材料在高溫和壓力共同作用下抗形變能力的專業設備 。其核心測試原理基于一個看似簡單卻極為精妙的設計:將一個規定直徑(通常為 5mm)的鋼球,在特定壓力(一般為 20N±0.2N )下,壓在被測試材料表面,并將整個裝置置于設定高溫的烘箱中保持一段時間(常見為 60 分鐘) 。測試結束后,通過測量材料表面留下的壓痕直徑,來判斷
    的頭像 發表于 04-24 13:33 ?1422次閱讀
    球壓試驗裝置:<b class='flag-5'>你</b><b class='flag-5'>可能不知道</b>的電氣安全衛士

    想將原理圖中的電源接口改成和圖二適配,不知道該怎樣修改,新人小白求大佬幫幫忙

    板子打好后發現和買的電源接口不匹配,想將原理圖中的電源接口改成和圖二適配,不知道該怎樣修改,求大佬幫幫忙,感謝?。?! 屏幕截圖 2025-04-15 172212.png
    發表于 04-15 17:30