統(tǒng)計(jì)輸入的行數(shù)
標(biāo)準(zhǔn)庫(kù)保證輸入文本流以行序列的形式出現(xiàn),每一行均以換行符結(jié)束。因此,統(tǒng)計(jì)行數(shù)等價(jià)于統(tǒng)計(jì)換行符的個(gè)數(shù)。
#include <stdio.h>/* count lines in input */main(){ int c, nl; nl = 0; while ((c = getchar()) != EOF) if (c == '/n') ++nl; printf("%d/n", nl);} 在該程序中,while 循環(huán)語(yǔ)句的循環(huán)體是一個(gè) if 語(yǔ)句,它控制自增語(yǔ)句++nl。if 語(yǔ)句先測(cè)試圓括號(hào)中的條件,如果該條件為真,則執(zhí)行其后的語(yǔ)句(或括在花括號(hào)中的一組語(yǔ)句)。這里再次用縮進(jìn)方式表明語(yǔ)句之間的控制關(guān)系。
雙等于號(hào)==是 C 語(yǔ)言中表示“等于”關(guān)系的運(yùn)算符(類似于 Pascal 中的單等于號(hào)=及 Fortran 中的.EQ.)。由于 C 語(yǔ)言將單等于號(hào)=作為賦值運(yùn)算符,因此使用雙等于號(hào)==表示相等的邏輯關(guān)系,以示區(qū)分。這里提醒注意,在表示“等于”邏輯關(guān)系的時(shí)候(應(yīng)該用==),C 語(yǔ)言初學(xué)者有時(shí)會(huì)錯(cuò)誤地寫(xiě)成單等于號(hào)=。在后面我們將看到,即使這樣誤用了,其結(jié)果通常仍然是合法的表達(dá)式,因此系統(tǒng)不會(huì)給出警告信息。
單引號(hào)中的字符表示一個(gè)整型值,該值等于此字符在機(jī)器字符集中對(duì)應(yīng)的數(shù)值,我們稱之為字符常量。但是,它只不過(guò)是小的整型數(shù)的另一種寫(xiě)法而已。例如,'A'是一個(gè)字符常量;在 ASCII 字符集中其值為 65(即字符 A 的內(nèi)部表示值為 65)。當(dāng)然,用'A'要比用 65 好,因?yàn)椤?#39;A'的意義更清楚,且與特定的字符集無(wú)關(guān)。
字符串常量中使用的轉(zhuǎn)義字符序列也是合法的字符常量,比如,'/n'代表?yè)Q行符的值,在 ASCII 字符集中其值為 10。我們應(yīng)當(dāng)注意到,'/n'是單個(gè)字符,在表達(dá)式中它不過(guò)是一個(gè)整型數(shù)而已;而"/n"是一個(gè)僅包含一個(gè)字符的字符串常量。
下面編寫(xiě)一個(gè)統(tǒng)計(jì)空格、制表符與換行符個(gè)數(shù)的程序。
#include <stdio.h>main(){ /* blanks, tabs, and newlines */ int c, nb, nt, nl; nb = 0; nt = 0; nl = 0; while( (c = getchar()) != EOF) { if(c == ' ') ++nb; if(c == '/t') ++nt; if(c == '/n') ++nl; } printf("%d %d %d /n", nb, nt, nl);} 統(tǒng)計(jì)輸入的單詞個(gè)數(shù)
這里對(duì)單詞的定義比較寬松,它是任何其中不包含空格、制表符或換行符的字符序列。下面這段程序是 UNIX 系統(tǒng)中 wc 程序的骨干部分:
#include <stdio.h>#define IN 1 /* inside a word */#define OUT 0 /* outside a word *//* count lines, words, and characters in input */main(){ int c, nl, nw, nc, state; state = OUT; nl = nw = nc = 0; while ((c = getchar()) != EOF) { ++nc; if (c == '/n') ++nl; if (c == ' ' || c == '/n' || c == '/t') state = OUT; else if (state == OUT) { state = IN; ++nw; } } printf("%d %d %d/n", nl, nw, nc);} 程序執(zhí)行時(shí),每當(dāng)遇到單詞的第一個(gè)字符,它就作為一個(gè)新單詞加以統(tǒng)計(jì)。state 變量記錄程序當(dāng)前是否正位于一個(gè)單詞之中,它的初值是“不在單詞中”,即初值被賦為 OUT。我們?cè)谶@里使用了符號(hào)常量 IN 與 OUT,而沒(méi)有使用其對(duì)應(yīng)的數(shù)值 1 與 0,這樣程序更易讀。在較小的程序中,這種做法也許看不出有什么優(yōu)勢(shì),但在較大的程序中,如果從一開(kāi)始就這樣做,因此而增加的一點(diǎn)工作量與提高程序可讀性帶來(lái)的好處相比是值得的。讀者也會(huì)發(fā)現(xiàn),如果程序中的幻數(shù)都以符號(hào)常量的形式出現(xiàn),對(duì)程序進(jìn)行大量修改就會(huì)相對(duì)容易得多。
下列語(yǔ)句 nl = nw = nc = 0; 將把其中的 3 個(gè)變量 nl、nw 與 nc 都設(shè)置為 0。這種用法很常見(jiàn),但要注意這樣一個(gè)事實(shí):在兼有值與賦值兩種功能的表達(dá)式中,賦值結(jié)合次序是由右至左。所以上面這條語(yǔ)句等同于 n1 = (nw = (nc = 0));
運(yùn)算符||代表 OR(邏輯或),所以下列語(yǔ)句 if (c == ' ' || c== '/n' || c == '/t') 的意義是“如果 c 是空格,或 c 是換行符,或 c 是制表符”(前面講過(guò),轉(zhuǎn)義字符序列/t 是制表符的可見(jiàn)表示形式)。相應(yīng)地,運(yùn)算符&&代表 AND(邏輯與),它僅比||高一個(gè)優(yōu)先級(jí)。由&&或||連接的表達(dá)式由左至右求值,并保證在求值過(guò)程中只要能夠判斷最終的結(jié)果為真或假,求值就立即終止。如果 c 是空格,則沒(méi)有必要再測(cè)試它是否為換行符或制表符,這樣就不必執(zhí)行后面兩個(gè)測(cè)試。在這里,這一點(diǎn)并不特別重要,但在某些更復(fù)雜的情況下這樣做就有必要了,不久我們將會(huì)看到這種例子。
這段程序中還包括一個(gè) else 部分,它指定當(dāng) if 語(yǔ)句中的條件部分為假時(shí)所要執(zhí)行的動(dòng)作。其一般形式為:
if (表述式) 語(yǔ)句 1else 語(yǔ)句 2
其中,if-else 中的兩條語(yǔ)句有且僅有一條語(yǔ)句被執(zhí)行。如果表達(dá)式的值為真,則執(zhí)行語(yǔ)句 1,否則執(zhí)行語(yǔ)句 2。這兩條語(yǔ)句都既可以是單條語(yǔ)句,也可以是括在花括號(hào)內(nèi)的語(yǔ)句序列。在單詞計(jì)數(shù)程序中,else 之后的語(yǔ)句仍是一個(gè) if 語(yǔ)句,該 if 語(yǔ)句控制了包含在花括號(hào)內(nèi)的兩條語(yǔ)句。



















