女人自慰AV免费观看内涵网,日韩国产剧情在线观看网址,神马电影网特片网,最新一级电影欧美,在线观看亚洲欧美日韩,黄色视频在线播放免费观看,ABO涨奶期羡澄,第一导航fulione,美女主播操b

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

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

3天內不再提示

關于你可能不知道的printf

黃工的嵌入式技術圈 ? 來源:黃工的嵌入式技術圈 ? 作者:黃工的嵌入式技術 ? 2020-02-05 12:28 ? 次閱讀

前言

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

示例程序

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

#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有什么區(qū)別呢?

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

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

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

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

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

可變參數(shù)中的類型提升

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

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

//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 類型參數(shù)

%f 期望的是 double 類型參數(shù)

%c 期望的也是 int 類型參數(shù)

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

參數(shù)入棧順序以及計算順序

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

浮點數(shù)的有效位

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

可變域寬和精度

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

#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完全可以是一個宏定義或者變量,從而做到了動態(tài)地格式控制。

格式控制符是如何處理參數(shù)的

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

真相大白

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

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

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

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

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

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

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

至此,真相大白。

總結

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

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

絕對避免出現(xiàn)計算結果與參數(shù)計算順序有關的代碼。

*在printf中實現(xiàn)可變域寬和精度。

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

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

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

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

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

    關注

    1

    文章

    66

    瀏覽量

    18945
  • 程序
    +關注

    關注

    117

    文章

    3820

    瀏覽量

    82396
  • Printf
    +關注

    關注

    0

    文章

    84

    瀏覽量

    14081
收藏 人收藏

    評論

    相關推薦
    熱點推薦

    不知道怎么畫原理圖了

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

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

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

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

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

    球壓試驗裝置:可能不知道的電氣安全衛(wèi)士

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

    ADS805E測量跳躍幅度很大,不知道是為什么?

    我按照手冊上直流耦合連接電路,輸入是1.5V~3.5V范圍,SEL與VREF相連,用430定時器輸出轉換時鐘,結果測量直流時,AD測量結果在幾個值之間跳躍,跳躍幅度很大,不知道是為什么,引腳電壓控制用3.3V。
    發(fā)表于 01-21 07:46

    測的值不定然后開始減小直到為0,不知道怎么回事?

    ADS1247對寄存器可讀可寫 可是配置完02這個地址的時(寫0x30)在vrefout測不到2.048v,測的值不定然后開始減小直到為0,不知道怎么回事?
    發(fā)表于 01-21 06:27

    有關UV固化不知道的技術細節(jié)

    UV固化技術中,光引發(fā)劑吸收UV光產(chǎn)生活性自由基觸發(fā)聚合反應,包括鏈引發(fā)、增長、終止和交聯(lián)。反應速率受UV光強、光引發(fā)劑濃度、單體類型等因素影響,UV固化技術具有快速固化特性。
    的頭像 發(fā)表于 12-06 09:33 ?723次閱讀

    盤點Linux系統(tǒng)中的常見命令

    作為運維同學怎能不知道Linux系統(tǒng)中的lsmod、lsof、lspci、lsscsi命令呢,今天就來盤一盤她及實例。
    的頭像 發(fā)表于 12-03 09:48 ?699次閱讀
    盤點Linux系統(tǒng)中的常見命令

    關于陶瓷電路板不知道的事

    陶瓷電路板(Ceramic Circuit Board),又稱陶瓷基板,是一種以陶瓷材料為基體,通過精密的制造工藝在表面形成電路圖形的高技術產(chǎn)品,快來看看哪些是您還不知道的事?
    的頭像 發(fā)表于 10-21 11:55 ?839次閱讀
    <b class='flag-5'>關于</b>陶瓷電路板<b class='flag-5'>你</b><b class='flag-5'>不知道</b>的事

    又一電工不知道,施耐德變頻器怎么復位,如果不告訴知道怎么復位嗎?

    維修 我給他說,可能以前沒接觸過施耐德的變頻器,就不知道它還有一個蓋子,要復位的話,就要把蓋子打開,第一次找不到也正常,要是經(jīng)常斷電重啟的話,對設備不好。 變頻器修理 之所以分享給大家,就是想告訴那些沒接觸過施耐
    的頭像 發(fā)表于 10-12 15:15 ?948次閱讀
    又一電工<b class='flag-5'>不知道</b>,施耐德變頻器怎么復位,如果不告訴<b class='flag-5'>你</b>,<b class='flag-5'>你</b><b class='flag-5'>知道</b>怎么復位嗎?

    使用CS1232 采集一個小信號,信號有可能是正,也可能是負 ,是不知道怎么回事情?

    我使用CS1232 采集一個小信號,信號有可能是正,也可能是負 但是不知道怎么回事情,采集的數(shù)據(jù)就經(jīng)常不對 在上圖的電路中,當我把P4 短路,采集到的電壓盡然是為8388608 我看了一下SDO
    發(fā)表于 09-25 14:36

    知道貼片電感故障時可能出現(xiàn)的癥狀嗎?

    知道貼片電感故障時可能出現(xiàn)的癥狀嗎?
    的頭像 發(fā)表于 08-17 14:20 ?706次閱讀
    <b class='flag-5'>你</b><b class='flag-5'>知道</b>貼片電感故障時<b class='flag-5'>可能</b>出現(xiàn)的癥狀嗎?

    AMC1100使用前需要烘烤,不知道烘烤溫度和烘烤時間是多少?

    1:AMC1100使用前需要烘烤,但是不知道 烘烤溫度和烘烤時間是多少?能在datasheet上查看到嗎? 2:datasheet上 MSL參數(shù) Level-1-260C-UNLIM中UNLIM
    發(fā)表于 08-09 08:11

    新手嘗試做一個LED驅動電路,不知道電路有沒有問題

    TP8006穩(wěn)流驅動,最后留出LED插口。 因為不是很懂硬件設計,不知道做的模塊能不能行,很希望各位提提意見,讓我認識認識硬件設計上的缺陷。
    發(fā)表于 07-24 18:35

    不知道怎么進行負載測試發(fā)電機嗎?

    測試一般的流程是怎樣的,知道嗎? ? 1、試驗前準備:確保發(fā)電機和試驗設備處于良好的工作狀態(tài),檢查發(fā)電機的電源和燃料供應是否正常,確保試驗設備與發(fā)電機的連接正確可靠。 2、確定試驗載荷:根據(jù)您的實際要求確定試驗
    的頭像 發(fā)表于 07-03 17:36 ?1487次閱讀