版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請(qǐng)進(jìn)行舉報(bào)或認(rèn)領(lǐng)
文檔簡介
第二章C和C++語言基本知識(shí)數(shù)據(jù)類型(datatypes)程序的主要任務(wù)之一是處理數(shù)據(jù),因此,我們首先須要知道有哪些數(shù)據(jù)類型,它們占用多大內(nèi)存。C++語言中,數(shù)據(jù)的類型可歸納為兩大類:第一類是C++系統(tǒng)所提供的一系列基本數(shù)據(jù)類型,稱為“預(yù)定義數(shù)據(jù)類型(pre-defineddatatype)”(和C語言中提供的數(shù)據(jù)類型基本相同);第二類是用戶自己定義的一系列數(shù)據(jù)類型,稱為“用戶自定義數(shù)據(jù)類型(user-defineddatatype)”。C++主要使用第二類。本章主要介紹“預(yù)定義數(shù)據(jù)類型”,而從第三章開始將陸續(xù)介紹‘‘用戶自定義數(shù)據(jù)類型”。C語言中本來也有一些’‘用戶自定義數(shù)據(jù)類型”,主要是數(shù)組、結(jié)構(gòu)體和聯(lián)合。但C++中由于編程需要,用戶將大量定義各種“類”,本課件中主要討論“類”。各“預(yù)定義數(shù)據(jù)類型”可用以下關(guān)鍵詞表達(dá),例如:int,char,double,bool,long等。程序可以為各類數(shù)據(jù)類型建立各個(gè)變量(variable),例如建立或聲明(declare)整型變量.i,以下語句用于聲明一個(gè)變量:intj;其中int是整型數(shù)據(jù)類型,而j則是整型變量。變量被稱為數(shù)據(jù)類型的實(shí)例(instances),建立變量的過程稱為實(shí)例化(instantiation)〇也可在聲明變量的同時(shí)將它初始化,以下語句用于定義(define)ー個(gè)變量。intj=1();或int各種數(shù)據(jù)類型的長度所有數(shù)據(jù)類型的變量都在內(nèi)存中占據(jù)一定空間,其長度可使用sizeof來確定。sizeof是一個(gè)宏而不是函數(shù)(本章§2.6.4中將介紹宏與函數(shù)的區(qū)別),它在系統(tǒng)的編譯階段就已求值了。這里它用于求某個(gè)變量在內(nèi)存中的長度,也即該變量所占有的字節(jié)數(shù)。其使用格式為:對(duì)于變量inta;可用sizeof?;或sizeof(int);其它如§izeof(double);等[例1]讀取預(yù)定義類型(即由C++系統(tǒng)本身所定義的類型)數(shù)據(jù)的長度(大部分?jǐn)?shù)據(jù)的說明見任何一本C語言書籍)//var_size.cpp//Tocheckthesizesofdifferentkindsofvariables#include<iostream>voidmain()(#include<iostream>voidmain()(std::cout?Msizeof(char)std::cout?v'sizeof(bool)std::cout?Hsizeof(short)std::cout?Msizeof(int):n?sizeof(char)?*\n*;:u?sizeof(bool)?,\n,;:**?sizeof(short)?*\n1;ゴ’vvsizeof(int)vv'\n';:"?sizeof(unsigned)?*\n:"?sizeof(unsigned)?*\n,:"?sizeof(long)?'\n';:"?sizeof(float)?'\n';:"?sizeof(double)?'\n';:"?sizeof(longlong)?'\nstd::cout?"sizeof(unsigned)std::cout?"sizeof(long)std::cout?"sizeof(float)std::cout?"sizeof(double)std::cout?"sizeof(longlong)std::cout?"sizeof(unsignedlonglong):"?sizeof(longlong)?'\n';std::cout?"sizeof(longdouble):"?sizeof(longdouble)?'\n';}/*Results:sizeof(char)
444448888*/這是在32位機(jī)上的運(yùn)行結(jié)果,操作系統(tǒng)是WindowsXP,C++平臺(tái)是VisualStudio2003。和以前I6位機(jī)上的運(yùn)行結(jié)果不同。但和64位機(jī)的Windows7操作系統(tǒng)的VS2010平臺(tái)上運(yùn)行的結(jié)果卻是相同的。整型數(shù)據(jù)例如intj;(可正可負(fù)的整數(shù))和unsignedinti;(只有正值的整數(shù))它們各占32個(gè)二進(jìn)制位(四個(gè)字節(jié))的空間。這32個(gè)二進(jìn)制位可用四種不同方式表示一個(gè)整數(shù)值:二進(jìn)制、ハ進(jìn)制、十進(jìn)制和十六進(jìn)制。以十進(jìn)制值15為例,可表示如下表中任何ー種形式:二進(jìn)制ハ進(jìn)制十進(jìn)制十六進(jìn)制111101715OxF(或Oxf)這四種表示方式之間的數(shù)值關(guān)系請(qǐng)見文件“附錄.doc”中附錄ー“不同基數(shù)制的整數(shù)表示方式”。正整數(shù)上面提到,整數(shù)有兩類:第一類為:“unsignedinti;"。使用二進(jìn)制32位表示正整數(shù)i,它不可能為負(fù)值。其范圍為〇至約為4.2xl09,即0至庁ユ」)(約為0至+42億,確切地為4,294,967,648,中國人口統(tǒng)計(jì)已夠用)。現(xiàn)以二進(jìn)制8位為例表示正整數(shù)13:13=0013=000011011286432168421在二進(jìn)制8位中,最髙有效位(MSB,MostSigniHcantBit)的值是128,最低有效位(LSB,LeastSigniHcantBit)的值是1。2.122負(fù)整數(shù)第二類整數(shù)為:“[signed]intj;”。使用32位表示可正可負(fù)的整數(shù)j,其范圍約為ー21億至+21億,即ー232/2至Q32/2A1(確切地為ー2,147,483,648至+2,147,483,647)。仍以二進(jìn)制8位為例表示+13:
對(duì)于負(fù)整數(shù),一般用補(bǔ)碼(twdscomplement)表示(即“求反加1”),使用特別方便:二進(jìn)制(以8位為例)的ー13可從其絕對(duì)值求反加一求得,表示如下:正數(shù)13表示為:13=00001101求反,得:反碼=11110010加1,得一13,表示為:反碼加1=11110011可見,負(fù)數(shù)的最髙有效位為1。讀取負(fù)碼絕對(duì)值的方法也是通過“求反加1"。如下:某負(fù)數(shù)為:X=11110011求反:X的反碼=00001100加1得x的絕對(duì)值為:X絕對(duì)值=00001101即13。所以知道此整數(shù)值為x=-13。補(bǔ)碼的另一方便處是:正數(shù)和負(fù)數(shù)相加時(shí),只須直接將它們的二進(jìn)制碼相加即可。3浮點(diǎn)型(floating-point)或?qū)嵭停╮eal)數(shù)據(jù)例如floatf;(占32個(gè)二進(jìn)制位制doubled;(占64個(gè)二進(jìn)制位制以64位浮點(diǎn)型數(shù)據(jù)為例,它在內(nèi)存中分為兩部分:指數(shù)部分x(exponent)的換算值共10位;尾數(shù)部分m(mantissa,也稱數(shù)值部分)的換算值共54位。指數(shù)部分X的換算值共10位尾數(shù)部分m的換算值共54位在存入內(nèi)存之前,浮點(diǎn)數(shù)的絕對(duì)值先被轉(zhuǎn)換為m?2X(其中0.5Wm<1.0;-1024WxW1023),例如, 1.2=0.6?2](其中m=0.6,x=1)0.0625=0.5?24(其中m=0.5,x=-3)然后m和x被分別換算后存入浮點(diǎn)型數(shù)據(jù)內(nèi)存中,換算目的是取得盡可能髙的精度,換算公式如下:將(x+1024)存入前10位,將(m-0.5)存入后54位。其中后54位(尾數(shù)部分)中每位的值如下:Sblb2b3b4b5b6b7b8-其中s為符號(hào)位:正浮點(diǎn)數(shù)的S=o,負(fù)浮點(diǎn)數(shù)的S=l。bl=0.25,b2=0.125,b3=0.0625,b4=0.03l25,b5=0.015625等?!癲oubled;”表示使用64個(gè)二進(jìn)制位表示可正可負(fù)的浮點(diǎn)數(shù)d,其范圍為±1xグ°23至土1x2+心,大致為±5.0x10.324至±1.7X10+308o讀取某些數(shù)據(jù)類型的最大和最小值的程序可參閱附錄二。了解以上所述內(nèi)容后,就不難理解第一章§1.2.2“優(yōu)越性”中|例1]出錯(cuò)的原因。字符型(character)數(shù)據(jù)例如charch;字符型變量在內(nèi)存中占據(jù)一個(gè)字節(jié)的空間。國際上通用的字符碼集是“美國信息交換標(biāo)準(zhǔn)代碼”(臺(tái)灣稱之為“美國標(biāo)準(zhǔn)資訊交換碼つ,通常稱為ASCI!碼(AmericanStandardCodeforInformationInterchange)〇其中每個(gè)字符都各有一個(gè)數(shù)值,而且大寫和小寫字符的數(shù)值各不相同。文件“附錄.doc”中附錄三是ASCII碼表。程序在輸出單個(gè)字符時(shí),將它包括于兩個(gè)單引號(hào)之間。例如:cout?T;程序在輸出ー系列字符(稱為字符串)時(shí),將它或它們包括于兩個(gè)雙引號(hào)之間。例如:cout?"Thisisacharacterstring!";反斜杠符號(hào)(ヽ)是一個(gè)特殊字符,稱為轉(zhuǎn)義符(escapecharacter),它可與其它字符組合使用,組成轉(zhuǎn)義序列符,完成特定顯示功能。常用轉(zhuǎn)義序歹!J(escapesequence)符(character)符號(hào)含義\n換行(一般同時(shí)回車)(newline)\r回車(carriagereturn)\t水平制表符(相當(dāng)于tab鍵)\b退格符(backspace)\\反斜線V單引號(hào)\H雙引號(hào)\0空字符(也用作字符串的結(jié)尾符)ー維數(shù)組數(shù)組(array)是ー種數(shù)據(jù)結(jié)構(gòu),由相同類型的數(shù)據(jù)項(xiàng)組成,在內(nèi)存中占用連續(xù)地址空間。例如inta[5]={1,3,5,7,9);內(nèi)存存儲(chǔ)內(nèi)容a[O]=la[l]=3a⑵二5a[3]=7a[4]=9其中方括號(hào)口內(nèi)的數(shù)字稱為序號(hào)(ordinalnumber)、位置號(hào)(positionnumber)或下標(biāo)(subscript,index),必須從。開始。數(shù)組內(nèi)每個(gè)單元都有一個(gè)下標(biāo)。[例1]顯示數(shù)組各單元內(nèi)容//arr_for.cpp#inciude<iostream>voidmain()(intarr[5]={l,3,5,7,9};for(inti=0;i<5;i++)std::cout?arr[i]?*\n*;//for語句見本章§2.4.23)/*Results:1*/[例2]數(shù)組各單元的初始化//arr_init_mod.cpp#include<iostream>voidmain()(intarrl[3]; /股有初始化std::cout?Marrl[3]:M; //顯示各單元值for(inti=0;i<3;i++)std::cout?arrl[i]?一;std::cout?r\nf;intarr2[4]={()}; 〃將各單元初始化為零std::cout?Harr2[4]:”; 〃顯示各單元值for(i=0;i<4;i++)std::cout?arr2[i]?'std::cout?,\n';intarr3[5]={1,2}; 〃只將前兩個(gè)單元初始化為1和2,其余單元?jiǎng)t初始化為零std::cout?Marr3[5]:M; 〃顯示各單元值for(i=0;i<5;i++)std::cout?arr3[i]?1r;std::cout?3;)/*Results:arrl[3]:-858993460-858993460-858993460arr2[4]:0000arr3[5]:12000以上程序中,如不進(jìn)行初始化,數(shù)組各單元的值將一片混亂。所以在使用數(shù)組前必須采取ー定方式進(jìn)行初始化。字符數(shù)組(characterarray):字符數(shù)組也稱字符串(characterstring),其中數(shù)組的每個(gè)單元是ー個(gè)字符。字符串可以用雙引號(hào)括起的字符序列來表示。例如charch[]=''work";即字符數(shù)組,其內(nèi)存存儲(chǔ)內(nèi)容為:ch[0]='w'ch[I]=’0’ch[2]='r'ch[3]='k'ch[4]='\0'請(qǐng)注意:字符數(shù)組必須用空字符'、〇'結(jié)尾。因此,字符數(shù)組的長度始終比它的有效字符數(shù)多1。字符型數(shù)據(jù)與字符數(shù)組的區(qū)別:.字符型數(shù)據(jù)用單引號(hào)表示,如,A'!而字符數(shù)組用雙引號(hào)表示,如“A”。.內(nèi)存存儲(chǔ)內(nèi)容不同,例如:'A"的內(nèi)存存儲(chǔ)內(nèi)容為"A”的內(nèi)存存儲(chǔ)內(nèi)容為'、〇'2.3 預(yù)處理功能為有助于執(zhí)行編譯過程(見第一章§1.3.2"C++程序的實(shí)現(xiàn)つ,經(jīng)常使用ー些預(yù)處理命令。每條預(yù)處理命令都具備一定的預(yù)處理功能。常用的預(yù)處理命令有以下三項(xiàng):宏定義命令、文件包含命令和條件編譯命令。預(yù)處理命令具有以下格式:.在左邊加一#號(hào),作為標(biāo)志。.一般獨(dú)占一行。.預(yù)處理命令不是編程語句,因此句末不加分號(hào)。,在正常編譯過程之前作為預(yù)備動(dòng)作而執(zhí)行,編譯過程結(jié)束后不占用存儲(chǔ)空間。2.3.1宏定義(macrodefinition)231.1簡單宏定義例如:#definePI3.14159#deHneSIZE80宏定義命令被執(zhí)行時(shí),用字符串替代宏名,例如用3.14159替代PL用80替代SIZE,等。[例I]計(jì)算圓周長和面積。#deHnePI3.14159voidmain()doublecircum,area,r=3;circum=2*PI*r;area=PI*r*r;}宏定義命令被執(zhí)行時(shí),不作語法檢査。例如,用戶希望將EXEC宏替代整句語句(例如inti=5;),使用以下宏:#defineEXECinti=5其中遺漏了分號(hào)(inti=5;オ是完整語句),但因不作語法檢査,因此査不出以上錯(cuò)誤,而直到編譯系統(tǒng)將使用該宏的其它語句進(jìn)行編譯時(shí)才能發(fā)現(xiàn)錯(cuò)誤。 帶參數(shù)宏定義注意:帶參數(shù)宏定義并不是函數(shù)例如:#defineADD(x,y)x+y[例1]計(jì)算兩數(shù)之対。#include<iostream>#defineADD(x,y)x+yvoidmain()(inta(40),b(50); /瀨a、b分別初始化為40、50intsum=ADD(a,b);〃宏替代結(jié)果為intsum=a+b;std::cout?sum?std::endl;}/*運(yùn)行結(jié)果:90宏替代的操作服從于運(yùn)算符的優(yōu)先級(jí),見下例:[例2]計(jì)算數(shù)的平方。//macro_3.cpp//Attentionindefiningmacro#defineSQ(x)x*x#include<iostream>voidmain()(inta=2,b=3;std::cout?SQ(a+b)?*\n*;〃由于乘法運(yùn)算符優(yōu)先級(jí)髙于加法運(yùn)算符,所以執(zhí)行a+b*a+bI/*Result:shouldbe25,butwegeta+b*a+b=ll*/問題出在優(yōu)先級(jí),應(yīng)改正如下://macro_l.cpp#defineSQ(x)(x)*(x)#include<iostream>voidmain()(inta=2,b=3;std::cout?SQ(a+b)?V; 〃即(a+b)*(a+b)}/*Result:25 */請(qǐng)注意:宏不是函數(shù)(子程序)!宏與函數(shù)的區(qū)別:宏節(jié)省時(shí)間但占用空間;函數(shù)節(jié)省空間但增加時(shí)間!(詳見本章§2.6.4)文件包含(inclusionoffiles)如果以上所述的宏定義語句很多,則可將它們包含于ー個(gè)單獨(dú)的文件中,例如“macros.h”中。此macros.h文件內(nèi)容例如可定義為://macros.h#defineMAX32#definesq(n)(n)*(n)#definePI3.14159此時(shí)用戶程序得以簡化為只包含以下一條語句:#include"macros.h"即可執(zhí)行以上所有宏命令。以上macros.h稱為頭文件(headerfiles)(或稱包含文件,includefiles)。其中系統(tǒng)所定義的頭文件名稱(例如iostream)使用尖括弧括nglebrackets)<>將頭文件名稱iostream包括起來,系統(tǒng)就根據(jù)這點(diǎn)到系統(tǒng)的文件夾中去調(diào)用。C++所提供的頭文件在包含時(shí)一般不包括字符“.h”,但舊的C語言所提供的頭文件在包含時(shí)一般應(yīng)該包括字符“.h”而用戶自定義的頭文件則用雙引號(hào)""將頭文件名稱包括起來,例如"macros.h"(一般都應(yīng)包括字符“.h")。系統(tǒng)就根據(jù)這點(diǎn)到用戶程序的文件夾中去調(diào)用。請(qǐng)注意此區(qū)別。C++語言系統(tǒng)中有很多頭文件,它們除包含宏替代定義語句外,更主要包含函數(shù)原型聲明(見本章§)和類的聲明(見第三章§3.2“類及其對(duì)象”中有關(guān)頭文件功能的描述),及其所用各種常量的定義(見第三章和第四章)。一般情況下,頭文件可以包含以下內(nèi)容:(1)類、函數(shù)、數(shù)據(jù)的聲明;(2)常量和宏的定義;(3)預(yù)處理語句等。但頭文件不宜包含類、函數(shù)、數(shù)據(jù)變量的定義。條件編譯(conditionalcompiling)條件編譯命令的功能是規(guī)定某些語句(或某個(gè)程序塊)在一定條件下オ參加編譯,否則跳過不予理睬。其主要用途為:防止遺漏定義或避免重復(fù)定義;以及在測(cè)試時(shí)增加某些測(cè)試語句,以實(shí)現(xiàn)調(diào)試跟蹤的目的。本節(jié)主要講防止遺漏定義或避免重復(fù)定義的功能。其格式為:#ifdef (常量表達(dá)式)(或#ifdefined(常量表達(dá)式))[條件編譯程序塊]#endif或#ifndef(常量表達(dá)式)(或#ifnotdefined(常量表達(dá)式))[條件編譯程序塊]#endif以上語句中,當(dāng)常量表達(dá)式無定義時(shí),就編譯兩個(gè)條件編譯命令#ifdef(或#ifndef)與#endif之間的條件編譯程序塊。條件編譯命令也可表達(dá)如:#if(常量表達(dá)式)[條件編譯程序塊]#endif此時(shí)如常量表達(dá)式非零,則滿足編譯條件,編譯兩個(gè)條件編譯命令#if與#endif之間的條件編譯程序塊?,F(xiàn)在講解其避免遺漏定義的功能:以下使用#ifndefMAX,此即(#ifnotdefinedMAX)或(#if!definedMAX)[例1]防止遺漏MAX的定義://test_ifndef_l.cpp//Topreventthemissingofmacrodefinition#include<iostream>#ifndefMAX#defineMAX32#endifvoidmain()std::cout?nMAXisM?MAX?'\n';/*Result:MAXis32 */上例中MAX沒有定義,所以執(zhí)行條件編譯語句#defineMAX32〇[例2]此例中MAX已經(jīng)定義過了,故不執(zhí)行條件編譯程序塊//test_ifndef_2.cpp//Topreventthemissingofmacrodefinition#include<iostream>#deHneMAX10#ifndefMAX#defineMAX32#endifvoidmain()(std::cout?HMAXisn?MAX?\n';)/*Result:MAXis10 */上例中MAX已被定義為等于10,所以不執(zhí)行條件編譯語句#defineMAX32。再看更重要的用途,避免重復(fù)包含頭文件:[例3]防止重復(fù)包含頭文件。設(shè)已有頭文件如下://ifndef.h//Toavoidtheduplicationofincludefiles#if!defined(_IFNDEF_H)#define_IFNDEF_Hintvar=l;//globalvariable#endif//_IFNDEF_H用戶應(yīng)用程序如下://test_ifndef_3.cpp//Topreventtheduplicationofmacrodefinition#include<iostream>#includeMD:\MyDocumentsWS2003?程序\ifndef?h”includeMD:\MyDocumentsWS2003?程序Xifndef.h" 〃看似重復(fù),其實(shí)不然voidmain()(std::cout?''變量var="?var?'\n';}/*Result:變量var=1號(hào)此程序中雖然多了一句#include“ifndef.h”,但第二句不起作用。原因在于頭文件中使用了條件編譯功能。怎么知道重復(fù)包含的第二句不起作用?因?yàn)槿绻诙涞脑?將會(huì)出錯(cuò)。如下://test_ifndef_4.cpp//Theerroroftheduplicationofmacrodefinition#include<iostream>intvar=l;intvar=l;voidmain()cout?”變量var="?var?endl;}/*Result:編譯出錯(cuò)errorC2374:"var"Z重定義:多次初始化參見“var”的聲明サ在任何頭文件中,都必須使用上述條件編譯語句以便避免重復(fù)包含。在有些頭文件中,也可能使用語句#pragmaonce,只要在頭文件的初始部分加入這條指令,就能夠保證頭文件只被編譯ー次。有時(shí)同時(shí)使用條件編譯語句#if!defined(」FNDEF一H)、#define一IFNDEF_H和#endif以及#pragma語句。2.4 控制流(flowofcontrol)語句Bohm和Jacopini的研究表明:所有程序都可以只用三類控制結(jié)構(gòu)(controlstructure):即順序結(jié)構(gòu)(sequencestructure)、選擇結(jié)構(gòu)(selectionstmcture)和循環(huán)結(jié)構(gòu)(loopingstructure,或稱重復(fù)結(jié)構(gòu),repetitionstructure)〇C和C++程序通常只使用七種控制結(jié)構(gòu):ー種順序結(jié)構(gòu)、三種選擇結(jié)構(gòu)(if、if?else和switch)和三種循環(huán)結(jié)構(gòu)(while、do?while和for)。其中順序結(jié)構(gòu)是任何編程語言所內(nèi)置或固有(inherent)的,即在沒有另外指定控制結(jié)構(gòu)的情況下,計(jì)算機(jī)總是按照編程順序逐條語句地往下執(zhí)行。因此本節(jié)主要介紹其它兩個(gè)控制結(jié)構(gòu):選擇結(jié)構(gòu)和循環(huán)結(jié)構(gòu)。選擇語句條件語句[例1]單項(xiàng)選擇結(jié)構(gòu)(single-selectionstructure)?本例中使用if語句求兩個(gè)數(shù)中的最大值//if_l.cpp//Onlyuse"if"tolookforthemaximumvalue#include<iostream>voidmain()(inti=1,j=2,x;if(i>j)x=i;if(j>i)x=j;if(j==i)x=j;std::coutv<”兩個(gè)數(shù)據(jù)中最大值:"<<x?,\n';)/*Result:兩個(gè)數(shù)藉中最大值:2*/上例中,當(dāng)if括號(hào)中的表達(dá)式符合實(shí)際情況時(shí),條件為真(true),執(zhí)行其后的語句。如不符合實(shí)際情況,則條件為假(false),不執(zhí)行其后的語句。請(qǐng)注意:切不可將邏輯判斷語句if(i==i)和賦值及邏輯判斷語句if(i=i)混為ー談,二者差別甚大。邏輯判斷語句if(j==i)只是用于判斷j是否等于i,如相等,則條件為真:否則為假。而賦值及邏輯判斷語句if(j=i)做兩件事,先將j賦值為等于i,然后再判斷賦值后的j是否不等于零,如不等于零,則條件為真;否則為假。構(gòu)(double-selectionstructure)。本例中使用if-else語句求兩個(gè)數(shù)中的最大值//max_2.cpp#include<iostream>voidmain()(inti=4,j=6,x;if(i>j) x=i;else x=j;std::cout?x?'\n';)/*Result:6 */上例中,當(dāng)if括號(hào)中的表達(dá)式符合實(shí)際情況時(shí),條件為真,執(zhí)行其后的語句。如不符合實(shí)際情況,則條件為假,改為執(zhí)行else后的語句。三元條件運(yùn)算符a>b?a:bresult=a>b?a:b;這ー語句等同于if(a>b)result=a;else result=b;它的優(yōu)點(diǎn)是比較簡練。[例1]條件運(yùn)算符用于求最大值//max_l.cpp#include<iostream>voidmain()(inti=4,j=6,x;X=i>j?i:j;std::cout?x?へn';)/*Result:6 */〃后兩句可合并為一句,如下://cout?(i>j?i:j)?'\n'; 〃其中()不可省略開關(guān)增句switch此處switch選擇結(jié)構(gòu)稱為多項(xiàng)選擇結(jié)構(gòu)(multiple-selectionstructure)〇[例11實(shí)現(xiàn)兩個(gè)數(shù)的四則運(yùn)算//switch」,cpp#include<iostream>voidmain()(doubledl,d2; 〃兩個(gè)操作數(shù)charop;〃運(yùn)算符std::cout?MPleaseinputthreedata:operand1operatoroperand?:'';std::cin?dl?op?d2;switch(op)(doubletemp;case*+':temp=dl+d2;std::cout<<dl?op?d2?'='?temp?'\n';break;casetemp=dl-d2;std::cout<<dl?op?d2?'='?temp?'\n';break;casetemp=dl*d2;std::cout?dl?op?d2?'='?temp?'\n';break;case7':temp=dl/d2;std::cout<<dl?op?d2?'='?temp?'\n';break;default:std::cout?"Error"?*\n';)}運(yùn)算結(jié)果例如:Pleaseinputthreedata:operand1operatoroperand2:2+32+3=5其中帶下劃線的字符由用戶鍵入。注意事項(xiàng):Lease后面的表達(dá)式必須是整型、字符型或枚舉型(enum)表達(dá)式。.case后面的表達(dá)式必須互不相同。.每個(gè)case語句序列最后一條語句是break,用于退出開關(guān)語句。最后ー個(gè)case語句序列可以不用break〇.帶有break的case語句序列的順序可以是任意的。break語句還可用于退出循環(huán),見本章§2.423[例5]。第九章§11.1.2“編輯框(EDIT)控件和數(shù)據(jù)交換”中有本程序的改進(jìn)版本,名為“EditComputer”的對(duì)話框項(xiàng)目,使用圖形化界面,可以自動(dòng)反復(fù)運(yùn)算。2.4.2循環(huán)語句2?4?2.1 while循環(huán)語句while(循環(huán)條件) {循環(huán)體);#include<iostream>voidmain()(intcount=1;while(count<=3)std::cout?count++?*\n*;//post-incremental,后置自增,其使用多于“前置自增”}/*Result:(共循環(huán)三次)123 */2.422do?while循環(huán)語句do{循環(huán)體}while(循環(huán)條件);在判斷循環(huán)條件之前已先執(zhí)行ー次循環(huán)體內(nèi)的程序塊!#include<iostream>voidmain()(intcount=1;do{std::cout?count++?*\n*;}while(count<=3);}/*Result:(只循環(huán)兩次)123 */2.423for循環(huán)語包for(循環(huán)變量初始值;循環(huán)條件;循環(huán)增量)循環(huán)體(可以包含多條語句)I例1]使用for循環(huán)語句的第一種形式//for_l.cpp/Z預(yù)先定義循環(huán)變量#include<iostream>voidmain()(intcount;for(count=1;count<=3;count++)std::cout?count?,\n';/*Results:*/[例2]使用for循環(huán)語句的第二種形式//for_2.cpp//在for語句內(nèi)定義循環(huán)變量#include<iostream>voidmain()|for(intcount=l;count<=3;count++)std::cout?count?*\n*;)/*Results:*/[例3]使用for循環(huán)語句的第三種形式//for_3.cpp/Z預(yù)免定義循環(huán)變量并將其初始化#include<iostream>voidmain()(intcount=l;for(;count<=3;count++)std::cout?count?*\n*;/*Results:*/[例4]使用for循環(huán)語句的第四種形式//for_4.cpp//在滴環(huán)體內(nèi)進(jìn)行循環(huán)增量#include<iostream>voidmain(){intcount=1;for(;count<=3;)std::cout?count++?'\n';}/*Results:*/[例5]使用for循環(huán)語句的第五種形式//for_5.cpp//在在環(huán)體內(nèi)判斷循環(huán)條件#include<iostream>voidmain()(intcount=l;for(;;)if(count>3)break;std::cout?count++?'\n*;
/*Results:123 */上一程序中兩個(gè)花擴(kuò)號(hào)(if(count>3)break;std::cout?count++?std::endl;}之間的內(nèi)容也稱為復(fù)合語句,當(dāng)復(fù)合語句中還包含變量或函數(shù)聲明(functiondeclarations)時(shí),它也稱為程序塊(block)。2.5 指針(pointer)251 定義指針是ー種數(shù)據(jù)類型,用于存放某個(gè)變量的地址值。此時(shí)該指針被稱為是指向該變量。ー個(gè)指針的類型決定于它所指向的變量的類型,它決定指針能夠訪問多大的數(shù)據(jù)空間。這點(diǎn)很重要。指針既可指向基本數(shù)據(jù)類型(即預(yù)定義數(shù)據(jù)類型),又可指向數(shù)組、函數(shù)、類和文件等用戶自定義數(shù)據(jù)類刑。指針可以初始化為0、NULL(即0,這是標(biāo)準(zhǔn)庫頭文件中定義的符號(hào)化常量)或ー個(gè)地址。內(nèi)容為0或NULL的指針不指向任何變量,稱為空指針。例如:inta=5;int*ptr;中的ptr。指針ptr在下式中被初始化為指向變量a。ptr=&a;以上兩條語句也可組合地表示為:int*ptr=&a;其中“&”為存地址運(yùn)算符,而則為取內(nèi)容運(yùn)算符。inta,*ptr;ptr=&a;在下冊(cè)的MFC項(xiàng)目中,用于指向特殊結(jié)構(gòu)(例如進(jìn)程、線程、同步類對(duì)象、視圖類對(duì)象、窗口、設(shè)備描述表、數(shù)據(jù)結(jié)構(gòu)的節(jié)點(diǎn)以及動(dòng)態(tài)鏈接庫模塊等)的指針還具有另ー稱呼一句柄(Handle)。數(shù)組的指針數(shù)組名是ー個(gè)指向該數(shù)組的常量指針(其值不能改變),也是該數(shù)組中首單元的地址值。指針類型單個(gè)變量的指針ー維數(shù)組的指針定義變量:inta=5;intarr[5]={1,3,5,7,9);聲明指針:int*ptr;int*ptra;將指針初始化:ptr=&a;ptra=arr;此處指向數(shù)組的指針的定義格式(int*ptra;)與普通預(yù)定義數(shù)據(jù)的指針的定義格式(int*ptr;)完全相同。但指針ptra的賦值語句卻不同,語句中不使用符號(hào)’&'。數(shù)組指針與數(shù)組名兩者的異同:(1)整型指針與整型數(shù)組:air既是數(shù)組名,又是數(shù)組地址,還是數(shù)組指針(稱為常量指針),三位一體。因此arr可在一定范圍內(nèi)與ptra等效地使用。U可在一定范圍內(nèi)與?等效地使用。但arr不準(zhǔn)重新賦值。例如:arr[0]即?arr即?ptra即ptra[O]=1air[3]即?(arr+3)即?(ptra+3)即ptra[3]=7但arr的使用不如ptra靈活,如:*arr++表達(dá)式不合法,但?ptra++表達(dá)式卻合法。見以下例子:即§2.2中[例1]的主程序voidmain(){intarr[5]={1,3,5,7,9);for(inti=0;i<5;i++)std::cout?arr[i]?4\n\〃數(shù)組名)其中的for語句可改為:for(inti=0;i<5;i++)std::cout?*(arr+i)?4\n5;〃指針//但不允許arr自身加減,即不允許://std::cout?*(arr++)?An';notallowed但指針ptra卻可以加減:int*ptra=arr;for(inti=0;i<5;i++)std::cout?*(ptra++)?fc\n';〃此處可以將指針ptra增減,指針增減詳見§253.2此處可見,一般情況下,可不必為數(shù)組另外定義指針。(2)字符串與字符數(shù)組:字符串一般使用指針(包括變量指針和常量指針兩種方式)表示其地址和內(nèi)容,此地址也即其第一個(gè)字符的地址。第一種為變量指針方式例如:char*ch_ptr="home";(ch_ptr為變量指針)。第二種為常量指針方式例如:charch_arr[5]="OK";(ch.arr為常量指針)。此處字符數(shù)組名是ー個(gè)指向該數(shù)組的常量指針,它不能重新賦值。字符串內(nèi)容及其地址的顯示方式,牽涉到數(shù)據(jù)的類型轉(zhuǎn)換問題。請(qǐng)參閱本章§2.7“數(shù)據(jù)類型轉(zhuǎn)換”。指針的賦值和增減和其它變量一樣,指針也可賦值和增減。但指針有它特殊之處,請(qǐng)看下面。指針的賦值所有變量,包括指針在內(nèi),在賦值運(yùn)算中要求等式兩側(cè)的變量具有相同數(shù)據(jù)類型(以后本章§2.7“數(shù)據(jù)類型轉(zhuǎn)換”和第五章§5.4.1“數(shù)據(jù)和對(duì)象的類型轉(zhuǎn)換”中將會(huì)詳述)。例如:intj=5,k;k=j;又如:[例?指針的賦值//ptr_=_l.cpp//assigningapointer#include<iostream>voidmain()intj=5,k=11,*ptr=&j;std::cout?*ptr?,\n*;ptr=&k;std::cout?*ptr?*\n*;}/*Results:511號(hào)在以下語句中,第三句出錯(cuò):inti=5;doubled=1.2,*ptr=&d;ptr=&i;//error:*=*:cannotconvertfrom'int*'to'double*'這個(gè)錯(cuò)誤是由于等號(hào)兩側(cè)變量的類型不同,請(qǐng)參閱本章§2.7“數(shù)據(jù)類型轉(zhuǎn)換'在字符指針的賦值運(yùn)算中,還有一個(gè)條件。我們知道,字符串一般使用指針表示其地址,也即其第一個(gè)字符的地址。對(duì)字符串賦值也即對(duì)指針賦值。因此,對(duì)字符指針重新賦值的條件是“該指針不可以是常量指針”。上面§2.5.2“數(shù)組的指針”中提到:字符數(shù)組名是ー個(gè)指向該數(shù)組的常量指針,不能重新賦值。因此,使用數(shù)組形式表示的字符串(因它使用常量指針)無法使用普通賦值運(yùn)算符“=”對(duì)字符串重新賦值。請(qǐng)見下例:I例2]不同情況下字符指針的賦值運(yùn)算//overld_str_=_2.cpp//seetheconditionthatcharacterstringscanorcannotbere-assigned#include<iostream>#include<string.h>voidmain(){chararray[]="OK"5 〃常量指針char*ptrl="home"; 〃變量指針char*ptr2="World"; 〃變量指針std::cout.setf(std::ios::hex|std::ios::uppercase); 〃顯示十六進(jìn)制和大寫字母,詳見第八章std::cout?"Originalstringsare:"?std::endl;std::cout?"array[]="<<array?','?"*ptrl="?ptrl?".withaddressesas:"?std::endl;std::cout?"array="?(int)array?','<<"ptrl="?(int)ptrl?std::endl;ptrl=ptr2;std::cout?"Afteroperationptrl=ptr2,stringsbecome:"?std::endl;std::cout?"array[]="<<array?','?"*ptrl="?ptrl?".withaddressesas:"?std::endl;std::cout?"array="<<(int)array?\'?"ptrl="?(int)ptrl?std::endl;ptrl=array;std::cout?"Afteroperationptrl=array,stringsbecome:"?std::endl;std::cout?"array[]="<<array?','?"*ptrl="?ptrl?".withaddressesas:"?std::endl;std::cout?"array="<<(int)array?','?"ptrl="?(int)ptrl?std::endl;std::cout?"'array=ptrl'isnotallowed!"?std::endl;//array=ptrl;//error:cannotconvertfrom'char*'to'char[3]'}/*Results:Originalstringsare:array[]=OK,*ptrl=home.withaddressesas:array=1244884,ptrl=4518348Afteroperationptrl=ptr2,stringsbecome:array[]=OK,*ptrl=World.withaddressesas:array=1244884,ptrl=4518340Afteroperationptrl=array,stringsbecome:array[]=OK,*ptrl=OK.withaddressesas:array=1244884,ptrl=1244884'array=ptrl'isnotallowed!*/從以上程序可以看出,指向數(shù)組的常量指針是不允許重新賦值的。253.2 指針的增減與其它預(yù)定義數(shù)據(jù)類型的變量相同,指針也可以隨意增量(incremented?以++表示)或減量(decremented,以ー-表示)。但這并不意味它的值只是增ー或減ー。事實(shí)上,它的數(shù)值變化是和它的類型(也即它所指向的數(shù)據(jù)類型)有關(guān)的。請(qǐng)看以下兩例:[例整型指針的增量//arr_ptrl.cpp//Showtheeffectofincrementofapointerofintegertype#include<iostream>intarrロ={1,2,3};voidmain()(int*ptra=arr;for(inti=0;i<3;i++,ptra++)std::coutvv”指針?biāo)赶虻牡刂?M?ptra?M;指針?biāo)赶虻臄?shù)據(jù)=M?*ptra?,\n,;)/*Results:指針?biāo)赶虻牡刂?00455FA0;指針?biāo)赶虻臄?shù)據(jù)=1指針?biāo)赶虻牡刂?00455FA4;指針?biāo)赶虻臄?shù)據(jù)=2指針?biāo)赶虻牡刂?00455FA8;指針?biāo)赶虻臄?shù)據(jù)=3以上程序中,指針ptra每增量一次,其數(shù)值即增加4,這正好是ー個(gè)整型數(shù)據(jù)的長度。[例2]浮點(diǎn)型指針的增量//arr_ptr2.cpp//Showtheeffectofincrementofapointerofdoubletype#include<iostream>doublearr[]={1.1,2.2,3.3};voidmain()(double*ptra=arr;for(inti=0;i<3;i++,ptra++)std::cout<<"指針?biāo)赶虻牡刂?"?ptra?"5指針?biāo)赶虻臄?shù)據(jù)="?*ptra?'\n';}/*Results:指針?biāo)赶虻牡刂?004564A8;指針?biāo)赶虻臄?shù)據(jù)=1.1指針?biāo)赶虻牡刂?004564B0;指針?biāo)赶虻臄?shù)據(jù)=2.2指針?biāo)赶虻牡刂?004564B85指針?biāo)赶虻臄?shù)據(jù)=3.3*/以上程序中,指針ptra每增量一次,其數(shù)值即增加8,這正好是ー個(gè)double型數(shù)據(jù)的長度。指向指針的指針例如:inti=5;int*ptr=&i; 〃ー級(jí)指針int**pp=&ptr; 〃二級(jí)指針即:指向整型指針ptr的指針 指向整型變量i的指針整型變量pp=&ptr ?Ptr=&i >i=5可表示為: i=*ptr;或i=**pp;第五章中將用到“二級(jí)指針”的概念。255 指針數(shù)組(arrayofpointers)在實(shí)際應(yīng)用中,常常使用ー維ー級(jí)字符指針數(shù)組來存放ー些字符串,以便于操作。[例1]使用ー維ー級(jí)字符指針數(shù)組來存放一星期中的各天。//arr_ptr3.cpp//Showthefunctionofacharacterarray#include<iostream>#include<conio.h>//forgetch()char*name[]={''Monday",''Tuesday","Wednesday”,"Thursday","Friday","Saturday","Sunday");voidmain(){intweek;while(1)(std::cout?"Inputsequentialnumber:";std::cin?week;if(week<l||week>7)break;std::cout?"Thisis"?name[week-l]?*\n';/Z指針數(shù)組第1單元的序號(hào)為〇,而用戶輸入的〃week值卻從1開始,所以week應(yīng)減1}std::cout?"Illegalnumber.Programended!"?'\n';getch();)/*Results:Inputsequentialnumber:!ThisisTuesdayInputsequentialnumber:5ThisisFridayInputsequentialnumber:0Illegalnumber.Programended! */2.6函數(shù)(function)262 函數(shù)定義和函數(shù)原型(函數(shù)聲明)函數(shù)的功能及函數(shù)定義(functiondefinition)[例1]不用函數(shù)的例子(此程序示意用,不能上機(jī))#include<iostream>voidmain()(doublex=8.(),y=64.0,z=125.0;std::cout?用于求解x的立方根的程序塊vv'\n';std::cout?用于求解y的立方根的程序塊vv'\n';std::cout?用于求解z的立方根的程序塊?'\n';)/*Results:245 */上例中求解立方根的程序塊可以使用逼近算法,包含數(shù)十條語句,精度髙時(shí)甚至超過100條,則整個(gè)程序?qū)㈤L達(dá)300多條,占據(jù)太多空間??捎靡韵路椒ü?jié)省空間開銷。[例2]使用函數(shù)的例子(此程序示意用,不能上機(jī))#include<iostream>doublecuberoot(double);〃函數(shù)原型voidmain()doublex=8.0,y=64.0,z=125.0;std::cout?cuberoot(x)?ヘn';std::cout?cuberoot(y)?,\n’;std::cout?cuberoot(z)?,\n’;)doublecuberoot(doublei)(用于求解double型數(shù)據(jù)的立方根的程序塊,包含至少數(shù)十條語句,甚至超過ー百條。}/*Results:245 */如此使用函數(shù),能將程序壓縮很多,由三百多條壓縮至ー百多條,從而節(jié)省空間,并具有良好的可讀性。函數(shù)的定義格式如下:〈返回值〉 く函數(shù)名〉V(參數(shù)表)>(く函數(shù)體〉)其中第一行也稱函數(shù)頭或首部(functionheader),一般能在一行內(nèi)寫完,但多行也可。〈返回值〉也稱〈類型說明符〉,也即,函數(shù)的返回值也代表函數(shù)的類型。如無返回值,則使用void表ZjSo函數(shù)定義的關(guān)鍵部分是函數(shù)體(functionbody),有時(shí)也可稱為塊(block)。設(shè)計(jì)函數(shù)時(shí)一般考慮使函數(shù)具有單ー功能。各C++系統(tǒng)對(duì)于函數(shù)返回值都有自己規(guī)定的缺省值,目前本課件所用VC++6,0版本規(guī)定:函數(shù)返回值的缺省值是int,即返回整型值。函數(shù)原型['聲明](functionprototype)ANSI(AmericanNationalStandardsInstitute,美國國家標(biāo)準(zhǔn)學(xué)會(huì))曾對(duì)C語言規(guī)定:任何函數(shù)必須使用函數(shù)原型(或稱函數(shù)原型聲明,或簡稱函數(shù)聲明,函數(shù)聲明,functiondeclaration),VisualC++也沿用此規(guī)定它規(guī)定所有函數(shù)在使用之前,必須先聲明它的函數(shù)原型,以便于檢査錯(cuò)誤。在ANSIC之前,編譯系統(tǒng)并不對(duì)函數(shù)進(jìn)行參數(shù)類型檢査,如果在函數(shù)調(diào)用中參數(shù)類型、參數(shù)順序或參數(shù)數(shù)量有錯(cuò),編譯系統(tǒng)都無法察覺,直至函數(shù)運(yùn)行結(jié)果中出現(xiàn)錯(cuò)誤,オ被發(fā)現(xiàn)。使用函數(shù)原型有助于克服這個(gè)缺陷。ANSIC語言使用函數(shù)原型后,編譯系統(tǒng)能夠及早地在編譯階段就驗(yàn)證函數(shù)調(diào)用的參數(shù)列表(包括參數(shù)的類型和數(shù)量)是否正確,不至于產(chǎn)生不必要的錯(cuò)誤或不易發(fā)現(xiàn)的邏輯錯(cuò)誤。函數(shù)原型的另ー個(gè)重要功能是參數(shù)的強(qiáng)制類型轉(zhuǎn)換(coercionofargument),將函數(shù)調(diào)用時(shí)的參數(shù)強(qiáng)制轉(zhuǎn)換為函數(shù)原型中所指定的類型。(以后將在本章§2.7“數(shù)據(jù)類型轉(zhuǎn)換”和第五章§5.4中敘述)大多數(shù)情況下,用戶在調(diào)用函數(shù)時(shí),只須知道函數(shù)原型而對(duì)函數(shù)定義不感興趣。函數(shù)原型的格式如下:〈返回值〉 〈函數(shù)名〉〈(參數(shù)表)>i可以看出:函數(shù)原型的格式是函數(shù)首部加分號(hào)。函數(shù)原型的參數(shù)表中可以包括參數(shù)類型和參數(shù)名兩者,如:intsq(皿);也可以只包括參數(shù)類型。如:intsa(int):這兩種形式都可以使用。[例1J不允許省略函數(shù)原型[聲明]。//fun_prot_l.cpp//functionprototypeshouldnotbeomitted#include<iostream>intadd(int,int);IIIfomitted,errorC2065:'add':undeclaredidentifiervoidmain()(inta=2,b=3;std::cout?add(a,b)?'\n';}intadd(intx,inty){return(x+y);)/*Result:5 */函數(shù)原型(函數(shù)聲明)和函數(shù)定義在表現(xiàn)形式上的差別在于:.函數(shù)原型(函數(shù)聲明)沒有函數(shù)體,而函數(shù)定義則必須具有函數(shù)體。.函數(shù)原型(函數(shù)聲明)必須用分號(hào)結(jié)尾,而函數(shù)定義的結(jié)尾則不用分號(hào)。函數(shù)原型和函數(shù)定義的形參類型和數(shù)量以及返回值都必須相同。一般將所有函數(shù)原型(函數(shù)聲明)都放在ー個(gè)頭文件中,這是頭文件的ー個(gè)重要用途。而該頭文件一般位于程序首部。在第三章§3.2.4“類的聲明與其實(shí)現(xiàn)部分的分離”ー節(jié)中,我們將會(huì)看到,按照軟件工程的觀點(diǎn),類的成員函數(shù)的函數(shù)聲明(即函數(shù)原型)和函數(shù)定義應(yīng)是分離的。.6.2 有參與無參函數(shù).1 無參[數(shù)的]函數(shù)[例1]函數(shù)正常結(jié)束//fun_l.cpp#include<iostream>voidmain()(inti=1;std::cout?i?'\n';=2;std::cout?i?'\n';}/*Results:1*/大部分函數(shù)都是運(yùn)行至函數(shù)末尾時(shí)オ結(jié)束。事實(shí)上,在某些函數(shù)中遇見return語句時(shí)也結(jié)束,如下例。[例2]函數(shù)遇return時(shí)結(jié)束//fun_2.cpp#include<iostream>voidmain()(inti=1;std::cout?i?'\n';return;=2;std::cout?i?'\n';}/*Result:*/此時(shí)最后兩條語句根本無用![例3]兩個(gè)無參函數(shù)(顯示出錯(cuò)信息的函數(shù))//err_msg_l.cpp//toshowanerrormessaage#include<iostream>voiderr_msg();〃函數(shù)原型voidmain()(intflag=-1;if(flag<0)err_msg();)voiderr__msg()(std::cout?‘'Error!”?*\n*;}/*Result:Error!*/262.2 有參[數(shù)的]函數(shù)有參函數(shù)具有一個(gè)或更多參數(shù)(argument)。當(dāng)函數(shù)的形參多于ー個(gè)時(shí),組成“參數(shù)列表(argumentlist)”或“形參列表”。[例1]求平方的函數(shù)#include<iostream>intsq_int(inta);voidmain()(inti=5,j;j=sq_int(丄); 〃當(dāng)然也可和上一句合并為sq_int⑸std::cout?j?‘\n‘;)intsq_int(inta) {returna*a;}/*Result:25 */以上函數(shù)中,intsq_int(inta);中的"a”稱為形參(形式參數(shù))(dummyargument),"j=sqJnt(丄)メ中的“i”稱為實(shí)參(實(shí)際參數(shù))(realoractualargument)。I例2]顯示出錯(cuò)序號(hào)的有參函數(shù)//err_msg_2.cpp#include<iostream>voiderr_msg(int);voidmain(){intflag=-1;if(flag<0)err__msg(-flag);}voiderr_msg(intnum)(std::cout?"ErrorNo."?num?‘\n‘;}/*Results:ErrorNo.1 */其中num是形參,-flag是實(shí)參。有時(shí)也可將實(shí)參與運(yùn)算符結(jié)合起來使用,例如下例:I例3]實(shí)參經(jīng)過一定運(yùn)算后,再使用條件運(yùn)算符進(jìn)行非線性運(yùn)算,形成飽和特性,對(duì)實(shí)參加以限制,然后再輸入函數(shù)內(nèi)。//argument_operat_2.cpp//usingconditionaloperatoractingonrealargument#include<iostream>intsq(int);voidmain()for(intj=l;j<=10;j++)
std::cout?sq((j<6)?j:6)?std::cout?*\nf;)intsq(intk){,returnk*k;)/*Result:14916253636363636號(hào)程序中各變量間的關(guān)系*2.623 參數(shù)表可變的函數(shù)個(gè)別情況下,雖然知道參數(shù)的類型,但其數(shù)量卻不確定。此時(shí)可使用參數(shù)表可變的函數(shù).其實(shí)c語言中已有這類函數(shù),例如標(biāo)準(zhǔn)輸出函數(shù)printf。在C++中,這類函數(shù)的函數(shù)原型具有以下形式的參數(shù)表:(inti,...).這個(gè)整型變量i的數(shù)值代表處理實(shí)參的循環(huán)次數(shù).編譯系統(tǒng)只檢査第一個(gè)實(shí)參是否為整型,而不過問其余實(shí)參的類型.使用參數(shù)表可變的函數(shù)時(shí),程序必須包含<cstdarg>頭文件.參數(shù)表可變的函數(shù)必須使用ー個(gè)va」ist類型(大部分情況下它是char?類型)的變量和三個(gè)宏.在處理實(shí)參前使用宏va-start:在處理實(shí)參過程中使用宏va-arg;在處理實(shí)參后使用宏va-end.試看兩個(gè)例子.[例1]用于顯示不同數(shù)量整型變量的程序.//argument_var_3.cpp//modifiedversionofexamplein”視頻學(xué)VisualC++(光盤)明日科技梁水王家勝編著”#include<iostream>#include<cstdarg>voidshow(inti,...); 〃參數(shù)表可變函數(shù)的函數(shù)原型voidmain()(std::coutvv”第一次調(diào)用:H;show(4,8,12,432,1005); 〃調(diào)用函數(shù)std::coutvぐ第二次調(diào)用:“;show(2,l08,1234); 〃調(diào)用函數(shù))voidshow(inti,...) 〃定義參數(shù)表可變函數(shù)vajistlist;va_start(list,i);while(i—)//vajist類型變量,or"char*list;vajistlist;va_start(list,i);while(i—)〃將int型的參數(shù)傳給va!〃將int型的參數(shù)傳給va!變量
隔出操作intval=va_arg(list,int);std::cout?val?'
std::cout?,\n,; /輪出回車符va.end(list); 〃將指針設(shè)置為NULL}/*Results:第一次調(diào)用:8124321005第二次調(diào)用:1081234號(hào)ピtypedefchar*va_list;#endif#define_VA_LIST_DEFINED#endif#ifdef_M_IX86#defineJNTSIZEOF(n)((sizeof(n)+sizeof(int)-1)&-(sizeof(int)-1))#defineva_start(ap,v)(ap=(va_list)&v+_INTSIZEOF(v))#defineva_arg(ap,t)(*(t*)((ap+=JNTSIZEOF(t))-JNTSIZEOF(t)))#defineva_end(ap)(ap=(va_list)0)*/主函數(shù)第一次顯示了四個(gè)實(shí)參,第二次顯示了兩個(gè)實(shí)參。第二次調(diào)用時(shí)堆棧內(nèi)容如右圖所示: i=21081234[例2]用于顯示多組不同類型相同數(shù)目的變量。//argument_var__4.cpp//modifiedversionofexamplein"視頻學(xué)VisualC++(光盤)明日科技梁水王家勝編著”#include<iostream>#include<cstdarg>voiddisplay(inti,...); 〃參數(shù)表可變函數(shù)的函數(shù)原型voidmain(){displaysノ第一組的第一值為”,8,“第二值為”,12.34, 〃調(diào)用函數(shù))voiddisplay(inti9...)|)voiddisplay(inti9...)|vajistlist;va_start(list,i);while(i—)〃定義參數(shù)表可變函數(shù)〃valist類型^^量〃建務(wù)掃描參數(shù)表的起始指針,并保存vajist型變量中的信息char*chl=va_arg(list,char*);intval=va_arg(list,int);char*ch2=va_arg(list,char*);〃將char?型的參數(shù)傳給chi變量〃將intchar*chl=va_arg(list,char*);intval=va_arg(list,int);char*ch2=va_arg(list,char*);〃將char?型的參數(shù)傳給ch2變量doubledbl=va_arg(list,double);//將double型的參數(shù)傳給dbl變量std::cout?chl?val?"?"?ch2?dbl?'\n'; /獺出操作va_end(list);〃將指針設(shè)置為NULLva_end(list);}/*Results:第一組的第一值為8;第二值為12.34第二組的第一值為!6S第二值為34.56以上主函數(shù)顯示了兩組變量的數(shù)值,每組有四個(gè)不同類型但相同順序的變量。下面列舉四個(gè)宏的定義,供參考:[摘自頭文件stdarg.h]指針總是指向下一個(gè)待訪問變量。#defineJNTSIZEOF(n)((sizeof(n)+sizeof(int)-1)&~(sizeof(int)-1))〃將變量長度擴(kuò)展為4的倍數(shù)〃按位運(yùn)算符(bitwiseoperators)中:&是按位與,?是按位非。#defineva_start(ap,v)(ap=(va_list)&v+_INTSIZEOF(v))〃其中&v是參數(shù)表中第一個(gè)參數(shù),即i([例1]中i=2)〃ーINTSIZEOF(v)是此參數(shù)長度的4的倍數(shù)?!ㄒ虼?就將list指針指向棧上第一個(gè)待訪問變量(如[例1]中108)#defineva_arg(ap,t)(*(t*)((ap+=JNTSIZEOF(t))-JNTSIZEOF(t)))〃(ap+=」NTSIZEOFQ))用于將福針指向下ー個(gè)變番〃ー」NTSIZEOF⑴指針值不變,但將地址值改為指向本變量〃*(t*)用于指針類型轉(zhuǎn)換,接著讀取本變量內(nèi)容#defineva_end(ap)(ap=(va_list)0)〃復(fù)篇這是規(guī)范做法,可省展。262.4 指針用作函數(shù)參數(shù)指針用作函數(shù)參數(shù)時(shí)具有以下優(yōu)點(diǎn):.對(duì)于用戶自定義的較復(fù)雜數(shù)據(jù)類型(例如結(jié)構(gòu)體、類等),使用指針能夠節(jié)省時(shí)間和空間,因只須在棧上復(fù)制指針的副本(只占4個(gè)字節(jié))而不是整個(gè)數(shù)據(jù)類型的副本(可能多達(dá)數(shù)十或數(shù)百個(gè)字節(jié))。.可以從子函數(shù)內(nèi)改變子函數(shù)以外的參量。本章§2.9"函數(shù)調(diào)用的基本概念”將詳述此點(diǎn)。[例1]通過指針修改結(jié)構(gòu)體中的子項(xiàng)//ptr_argl.cpp//pointerusedasfunctionargument〃通過指針修改結(jié)構(gòu)中的子項(xiàng)〃只用于教學(xué)目的#include<iostream>structperson{intyear;intmonth;intday;doublesalary;〃其它各東);voidmodify(person*ptr,doubleval);〃函數(shù)原型voidmain()(personsample,*ptrl=&sample;modify(ptrl,1234.5);std::cout?sample.salary?,\n,;}voidmodify(person*ptr,doubleval)|ptr->salary=val;}/?Results:1234,5 */下面例子中display()函數(shù)通過指針來調(diào)用不同數(shù)組。サ例2]使用指針作為函數(shù)參數(shù),既顯示成功信息又顯示出錯(cuò)信息。//err_msg_4.cpp//Pointerusedasfunctionargument#include<iostream>voiddisplay(char**ptr,intnum);char*msg[]={"Coolingwaterrunning!","VentilationOK!","LubricationOK!"};char*err_msg[]={"Nocoolingwater!","Noventilation!","Nolubrication!"};voidmain() 〃只是為了講解(intflag=1;display(msg,flag);flag=-2;display(err_msg,-flag);}voiddisplay(char**ptr,intnum)(,,std::cout?ptr[num-l]?'\n';}/*Results:Coolingwaterrunning!Noventilation!*/此處使用指針后,在調(diào)用函數(shù)display(char**ptr,intnum)時(shí),不必復(fù)制占用很多空間的數(shù)組副本,只需復(fù)制占用很少空間的指針副本。(詳見本章§2.9"函數(shù)調(diào)用的基本概念”)但指針使用并不方便,因此在各種面向?qū)ο蟪绦蛟O(shè)計(jì)語言中,“指針”現(xiàn)已逐漸被“引用”(見§2.8“引用”)所取代。263具有返回值的函數(shù)以前提到過,函數(shù)的類型是由函數(shù)的返回值所代表的?,F(xiàn)在看個(gè)例子。[例11整型函數(shù)用于求最大值,返回整型值#include<iostream>intmax(inta,intb);voidmain()(inti=l,j=2,result;result=max(i,j);std::cout?MmaxvalueisM?result?,\n,;}intmax(inta,intb)(intresult;result=a>b?a:b;returnresult;}/*Result:maxvalueis2 */由于函數(shù)的返回值是整型變量,因此我們稱max為整型函數(shù)。另外,函數(shù)max()可簡化為:intmax(inta,intb)(returna>b?a:b;}264 函數(shù)與宏的區(qū)別下面具體看一下宏與函數(shù)的區(qū)別:宏節(jié)省時(shí)間但占用空間;函數(shù)節(jié)省空間但增加時(shí)間開銷!宏的例子:(此程序示意用,不能上機(jī))#include<iostream>#definecubrt(x) (用于求解double型數(shù)據(jù)的立方根的程序塊,至少數(shù)十條語句)voidmain()(doublex=8.0,y=64.0,z=125.0;std::cout?cubrt(x)?endl;std::cout?cubrt(y)?endl;std::cout?cubrt(z)?endl;
/*Results:24函數(shù)的例子(相同功能):(此程序示意用,不能上機(jī))#include<iostream>doublecuberoot(double);〃函數(shù)原型voidmain()(doublex=8.0,y=64.0,z=125.0;std::cout?cuberoot(x)?endl;std::cout?cuberoot(y)?endl;std::cout?cuberoot(z)?endl;)doublecuberoot(doublei)(用于求解double型數(shù)據(jù)的立方根的程序塊,至少數(shù)十條語句)/*Results:245 */宏調(diào)用 函數(shù)調(diào)用voidmain()voidmain()voidmain()cuberoot(x)cuberoot(y)cuberoot(z)doublecuberoot(doublei)求立方根的程序塊cuberoot(x)cuberoot(y)cuberoot(z)doublecuberoot(doublei)求立方根的程序塊(例如有一百多條語句)cubrt(x)求立方根的程序塊(例如有一百多條語句)cubrt(y)
求立方根的程序塊(例如有一百多條語句)cubrt(z)求立方根的程序塊(例如有一百多條語句)265 函數(shù)指針與其指針數(shù)組函數(shù)既然有地址,就可以具有指針。[例1]函數(shù)指針//fun_ptr_l.cpp//Toshowtheuseofafunctionpointer#include<iostream>voidfun();voidmain()void(*ptr)()=fun; 〃ptr即函數(shù)指針,指向函數(shù)fun〃?uuiva121111。''void(*p—)(and"ptr-fun;“ptr();}voidfun()(std::cout?Mfun()iscalle
溫馨提示
- 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請(qǐng)下載最新的WinRAR軟件解壓。
- 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請(qǐng)聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
- 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁內(nèi)容里面會(huì)有圖紙預(yù)覽,若沒有圖紙預(yù)覽就沒有圖紙。
- 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
- 5. 人人文庫網(wǎng)僅提供信息存儲(chǔ)空間,僅對(duì)用戶上傳內(nèi)容的表現(xiàn)方式做保護(hù)處理,對(duì)用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對(duì)任何下載內(nèi)容負(fù)責(zé)。
- 6. 下載文件中如有侵權(quán)或不適當(dāng)內(nèi)容,請(qǐng)與我們聯(lián)系,我們立即糾正。
- 7. 本站不保證下載資源的準(zhǔn)確性、安全性和完整性, 同時(shí)也不承擔(dān)用戶因使用這些下載資源對(duì)自己和他人造成任何形式的傷害或損失。
最新文檔
- 2025年度商務(wù)樓食堂承包經(jīng)營服務(wù)合同4篇
- 2025年度國際旅游包機(jī)服務(wù)合同執(zhí)行2篇
- 2025年廠外施工臨時(shí)用水用電合同范本3篇
- 2025年度服裝出口銷售合同定制版3篇
- 2024版簡易工程承包合同
- 2025年度政府大數(shù)據(jù)服務(wù)中心外包服務(wù)合同范本4篇
- 2024聘用醫(yī)療設(shè)備維修工程師合同范本3篇
- 2025年度新型建筑鋼結(jié)構(gòu)工程承包合同4篇
- 2025年度智慧停車系統(tǒng)設(shè)計(jì)與施工承包合同4篇
- 2025年度人工智能產(chǎn)業(yè)孵化器運(yùn)營管理合同2篇
- 衡水市出租車駕駛員從業(yè)資格區(qū)域科目考試題庫(全真題庫)
- 護(hù)理安全用氧培訓(xùn)課件
- 《三國演義》中人物性格探析研究性課題報(bào)告
- 注冊(cè)電氣工程師公共基礎(chǔ)高數(shù)輔導(dǎo)課件
- 土方勞務(wù)分包合同中鐵十一局
- 乳腺導(dǎo)管原位癌
- 冷庫管道應(yīng)急預(yù)案
- 司法考試必背大全(涵蓋所有法律考點(diǎn))
- 公共部分裝修工程 施工組織設(shè)計(jì)
- 《學(xué)習(xí)教育重要論述》考試復(fù)習(xí)題庫(共250余題)
- 裝飾裝修施工及擔(dān)保合同
評(píng)論
0/150
提交評(píng)論