用法
.h? ? extern NSString * const LoginURL;
.m? NSString * const kInitURL=@"";
const 和 #define區(qū)別 (2012-12-11 14:14:07)轉(zhuǎn)載▼
(1) 編譯器處理方式不同
define宏是在預(yù)處理階段展開了罪。
const常量是編譯運(yùn)行階段使用骄瓣。
(2) 類型和安全檢查不同
define宏沒有類型,不做任何類型檢查涩哟,僅僅是展開。
const常量有具體的類型心软,在編譯階段會(huì)執(zhí)行類型檢查扎瓶。
(3) 存儲(chǔ)方式不同
define宏僅僅是展開,有多少地方使用浊闪,就展開多少次恼布,不會(huì)分配內(nèi)存。
const常量會(huì)在內(nèi)存中分配(可以是堆中也可以是棧中)搁宾。
(4)const? 可以節(jié)省空間折汞,避免不必要的內(nèi)存分配。 例如:
#define PI 3.14159 //常量宏
const doulbe Pi=3.14159; //此時(shí)并未將Pi放入ROM中 ......
double i=Pi; //此時(shí)為Pi分配內(nèi)存盖腿,以后不再分配爽待!
double I=PI; //編譯期間進(jìn)行宏替換,分配內(nèi)存
double j=Pi; //沒有內(nèi)存分配
double J=PI; //再進(jìn)行宏替換奸忽,又一次分配內(nèi)存堕伪!
const定義常量從匯編的角度來看,只是給出了對(duì)應(yīng)的內(nèi)存地址栗菜,而不是象#define一樣給出的是立即數(shù)欠雌,所以,const定義的常量在程序運(yùn)行過程中只有一份拷貝疙筹,而 #define定義的常量在內(nèi)存中有若干個(gè)拷貝富俄。
(5) 提高了效率。 編譯器通常不為普通const常量分配存儲(chǔ)空間而咆,而是將它們保存在符號(hào)表中霍比,這使得它成為一個(gè)編譯期間的常量,沒有了存儲(chǔ)與讀內(nèi)存的操作暴备,使得它的效率也很高悠瞬。
const 與 #define的比較
C++ 語言可以用const來定義常量,也可以用 #define來定義常量。但是前者比后者有更多的優(yōu)點(diǎn):
(1)? const常量有數(shù)據(jù)類型浅妆,而宏常量沒有數(shù)據(jù)類型望迎。編譯器可以對(duì)前者進(jìn)行類型安全檢查。而對(duì)后者只進(jìn)行字符替換凌外,沒有類型安全檢查辩尊,并且在字符替換可能會(huì)產(chǎn)生意料不到的錯(cuò)誤(邊際效應(yīng))。
(2)? 有些集成化的調(diào)試工具可以對(duì)const常量進(jìn)行調(diào)試康辑,但是不能對(duì)宏常量進(jìn)行調(diào)試摄欲。
l? 【規(guī)則5-2-1】在C++ 程序中只使用const常量而不使用宏常量,即const常量完全取代宏常量疮薇。
5.3 常量定義規(guī)則
l? 【規(guī)則5-3-1】需要對(duì)外公開的常量放在頭文件中胸墙,不需要對(duì)外公開的常量放在定義文件的頭部。為便于管理惦辛,可以把不同模塊的常量集中存放在一個(gè)公共的頭文件中劳秋。
l? 【規(guī)則5-3-2】如果某一常量與其它常量密切相關(guān),應(yīng)在定義中包含這種關(guān)系胖齐,而不應(yīng)給出一些孤立的值玻淑。
例如:
const? float? RADIUS = 100;
const? float? DIAMETER = RADIUS * 2;
5.4 類中的常量
有時(shí)我們希望某些常量只在類中有效。由于#define定義的宏常量是全局的呀伙,不能達(dá)到目的补履,于是想當(dāng)然地覺得應(yīng)該用const修飾數(shù)據(jù)成員來實(shí)現(xiàn)。const數(shù)據(jù)成員的確是存在的剿另,但其含義卻不是我們所期望的箫锤。const數(shù)據(jù)成員只在某個(gè)對(duì)象生存期內(nèi)是常量,而對(duì)于整個(gè)類而言卻是可變的雨女,因?yàn)轭惪梢詣?chuàng)建多個(gè)對(duì)象谚攒,不同的對(duì)象其const數(shù)據(jù)成員的值可以不同。
不能在類聲明中初始化const數(shù)據(jù)成員氛堕。以下用法是錯(cuò)誤的馏臭,因?yàn)轭惖膶?duì)象未被創(chuàng)建時(shí),編譯器不知道SIZE的值是什么讼稚。
class A
{…
const int SIZE = 100; // 錯(cuò)誤括儒,企圖在類聲明中初始化const數(shù)據(jù)成員
int array[SIZE];? ? ? // 錯(cuò)誤,未知的SIZE
};
const數(shù)據(jù)成員的初始化只能在類構(gòu)造函數(shù)的初始化表中進(jìn)行锐想,例如
class A
{…
A(int size);? ? ? // 構(gòu)造函數(shù)
const int SIZE ;
};
A::A(int size) : SIZE(size) // 構(gòu)造函數(shù)的初始化表
{
…
}
A? a(100);? // 對(duì)象 a 的SIZE值為100
A? b(200);? // 對(duì)象 b 的SIZE值為200
怎樣才能建立在整個(gè)類中都恒定的常量呢帮寻?別指望const數(shù)據(jù)成員了,應(yīng)該用類中的枚舉常量來實(shí)現(xiàn)赠摇。例如
class A
{…
enum { SIZE1 = 100, SIZE2 = 200}; // 枚舉常量
int array1[SIZE1];
int array2[SIZE2];
};
枚舉常量不會(huì)占用對(duì)象的存儲(chǔ)空間固逗,它們?cè)诰幾g時(shí)被全部求值浅蚪。枚舉常量的缺點(diǎn)是:它的隱含數(shù)據(jù)類型是整數(shù),其最大值有限抒蚜,且不能表示浮點(diǎn)數(shù)(如PI=3.14159)掘鄙。sizeof(A) = 1200;其中枚舉部長空間。
enum? EM { SIZE1 = 100, SIZE2 = 200}; // 枚舉常量? ? sizeof(EM) = 4嗡髓;