C語言printf及引數壓棧順序

2021-06-18 06:01:16 字數 3227 閱讀 8356

1.呼叫格式為  printf("《格式化字串》", 《參量表》);

其中格式化字串包括兩部分內容: 一部分是正常字元, 這些字元將按原樣輸出; 另一部分是格式化規定字元, 以"%"開始, 後跟乙個或幾個規定字元, 用來確定輸出內容格式。參量表是需要輸出的一系列引數, 其個數必須與格式化字串所說明的輸出引數個數一樣多, 各引數之間用","分開, 且順序一一對應, 否則將會出現意想不到的錯誤。

2.格式化字元

%d 十進位制有符號整數

%u 十進位制無符號整數

%f 浮點數

%s 字串

%c 單個字元

%p 指標的值

%e 指數形式的浮點數

%x, %x 無符號以十六進製制表示的整數

%0 無符號以八進位制表示的整數

%g 自動選擇合適的表示法

說明:

(1). 可以在"%"和字母之間插進數字表示最大場寬。 例如: %3d 表示輸出3位整型數, 不夠3位右對齊。 %9.2f 表示輸出場寬為9的浮點數, 其中小數字為2, 整數字為6, 小數點佔一位, 不夠9位右對齊。%8s 表示輸出8個字元的字串, 不夠8個字元右對齊。 如果字串的長度、或整型數字數超過說明的場寬, 將按其實際長度輸出。 但對浮點數, 若整數部分位數超過了說明的整數字寬度, 將按實際整數字輸出; 若小數部分位數超過了說明的小數字寬度, 則按說明的寬度以四捨五入輸出。另外, 若想在輸出值前加一些0, 就應在場寬項前加個0。 例如: %04d 表示在輸出乙個小於4位的數值時, 將在前面補0使其總寬度為4位。如果用浮點數表示字元或整型量的輸出格式, 小數點後的數字代表最大寬度, 小數點前的數字代表最小寬度。 例如: %6.9s 表示顯示乙個長度不小於6且不大於9的字串。若大於9, 則第9個字元以後的內容將被刪除。

(2). 可以在"%"和字母之間加小寫字母l, 表示輸出的是長型數。例如: %ld 表示輸出long整數, %lf 表示輸出double浮點數。

(3). 可以控制輸出左對齊或右對齊, 即在"%"和字母之間加入乙個"-" 號可說明輸出為左對齊, 否則為右對齊。例如: %-7d 表示輸出7位整數左對齊,%-10s 表示輸出10個字元左對齊。

3. 一些特殊規定字元

\n換行

\f清屏並換頁

\r回車

\t tab符

\xhh表示乙個ascii碼用16進表示,

其中hh是1到2個16進製制數

int a=1234;

printf("a=%d\n",a); //a=1234

printf("a=%2d\n",a); //a=1234 超過2位,按實際輸出

printf("a=%6d\n",a); //a= 1234 不足6位,右對齊

printf("a=%06d\n",a); //a=001234 不足6位,前面補0

printf("a=%-6d\n",a); //a=1234 '-'左對齊

int* i=&a;

printf("i=%p\n",i); //i=0012ff44 輸出指標的值,即位址

float m=8888.8888; //float 單精度型浮點數 有效位數是6位或7位,根據不同的浮點數會有不同

float m1=8888.8888f; //在後面加上f或f,編譯警告:truncation from'const double'to 'float'

//編譯器預設浮點數為double

float m2=8888.888f;

double n=8888.8888;

double n1=8888888888.88888888; //double 雙精度型浮點數 有效位數是15位

printf("m=%f\n m1=%f\n m2=%f\n n=%lf\n n1=%f\n",m,m1,m2,n,n1); // m=8888.888672

// m1=8888.888672

// m2=8888.887695

// n=8888.888800

// n1=8888888888.888889

//%f的預設輸出小數字數就是6位不管有沒有l

/*printf的%f說明符的確既可以輸出float型又可以輸出 double型。 根據"預設引數提公升"規則(在printf這樣的函式的

可變引數列表中 ,不論作用域內有沒有原型,都適用這一規則)float型會被提公升為double型。因此printf()只會看到

雙精度數。嚴格地講,%lf在printf下是未定義的,但是很多系統可能會接受它。要確保可移植性,就要堅持使用%f。*/

printf("m4=%4.2f\n",m); //寬度總共4位,小數兩位,小數點一位,整數一位,這裡整數超過寬度規定,按實際整數字輸出

printf("m5=%9.6f\n",m); //浮點數小數部分不足6位,右對齊

printf("m6=%9.2f\n",m); //整數部分不足6位,右對齊;小數部分超過2位,四捨五入

char c[20]="hello,world!";

printf("c=%s\n",c);

printf("c=%6.9s\n",c); //c=hello,wor 6.9s表示輸出乙個長度長度不小於6且不大於9的字串。若大於9, 則第9個字元以後的內容將被刪除。

引數壓棧順序

int arr=;

int *ptr=arr;

*(ptr++)+=123;

printf("%d,%d",*ptr,*(++ptr));

輸出結果:8,8

解釋:c中printf計算引數時是從右往左壓棧的。

即當呼叫函式printf()時,首先從右往左將讀入的引數壓入棧,然後

函式被呼叫時,再從棧頂開始取資料進行計算。所以可以理解為「從右向左求值並壓棧」。

不光printf函式是這樣,函式都一樣,所有引數也是自右像左計算的。

原因是,如果乙個函式有多個引數,比如

int fun (int a, int b, int c);

呼叫時,總是從最後乙個引數開始壓棧。 也就是c先進棧,其次是b,最後才是a。

同理, 如果你這樣寫:

int main ()

先計算最後乙個引數(即最右乙個++i),壓入1,以此類推,再壓入2,3和字串"%d,%d,%d"的首位址。

所以 , 程式的輸出是 3,2,1

printf函式的引數壓棧問題

最近看到一些程式設計師的筆試題目,經常會考到printf函式的引數壓棧問題,總體來講就是引數從右向左依次壓棧,再出棧,但是今天看到乙個看似很簡單的題目,卻一直找不到頭緒。題目如下 cpp view plain copy include void main 輸出看似很簡單,但是結果卻打出所料。輸出是 ...

C語言 printf計算順序和壓棧順序初探

不同編譯器產生的結果不同,本文測試環境為vs2013和vc6.先看以下 int x 3 printf d d d d d d x,x x,x,x x vs2013輸出結果 434434 vc6輸出結果 444433 那麼問題來了 1 兩個編譯器printf的壓棧順序和計算順序是怎樣的?2 兩個編譯器...

C語言函式引數壓棧順序為何是從右到左?

上學期學習了組合語言,並在作業系統實驗中使用了彙編 c語言混合程式設計,中間也了解了一些c語言與組合語言的對應關係。由於組合語言是底層的程式語言,各種函式引數都要直接控制棧進行訪問,在混合程式設計中,要用彙編來呼叫c函式,當然就要知道引數的壓棧情況了。當知道c函式的引數壓棧順序是從右到左時,我覺得很...