C語(yǔ)言各種變量的初始化方式
數(shù)值類變量初始化
整型、浮點(diǎn)型的變量可以在定義的同時(shí)進(jìn)行初始化,一般都初始化為0
。
int inum = 0; float fnum = 0.00f; double dnum = 0.00;
字符型變量初始化
字符型變量也可在定義的同時(shí)進(jìn)行初始化,一般初始化為'\0'
。
char ch = '\0';
字符串初始化
字符串初始化的方法比較多,我這里簡(jiǎn)單介紹三種,因?yàn)樽址举|(zhì)上是由一個(gè)個(gè)字符組成的字符數(shù)組,所以其初始化的最終目的,就是將字符數(shù)組里面的一個(gè)個(gè)字符都初始化為'\0'
。
方法一:使用空的字符串""
。
char str[10] = "";
方法二:使用memset
。
char str[10]; memset(str, 0, sizeof(str));
方法三:寫一個(gè)循環(huán)。
char str[10]; for(int i = 0; i < 10; i++) { str[i] = '\0'; }
這里比較推薦的是第二種初始化方法。也即使用memset
進(jìn)行初始化。
很多人對(duì)memset
這個(gè)函數(shù)一知半解,只知道它可以初始化很多數(shù)據(jù)類型的變量,卻不知道其原理是什么樣的,這里做一下簡(jiǎn)要的說(shuō)明:memset
是按照字節(jié)進(jìn)行填充的。
先看下面的一段代碼:
int num; memset(&num, 0, sizeof(int)); printf("step1=%d\n", num); memset(&num, 1, sizeof(int)); printf("step2=%d\n", num);
在討論之前,我們先看一下運(yùn)行結(jié)果
chenyc@DESKTOP-IU8FEL6:~/src$ gcc -o memset memset.c -g chenyc@DESKTOP-IU8FEL6:~/src$ ./memset step1 = 0 step2 = 16843009 chenyc@DESKTOP-IU8FEL6:~/src$
看到這個(gè)運(yùn)行結(jié)果,是不是和你想象中的不一樣呢?step1 = 0
相信大家都好理解,可 step2 = 16843009
很多人就不能理解了。按照一般的慣性思維,不是應(yīng)該 = 1
才對(duì)么?
這就是我要說(shuō)的,memset是按照字節(jié)進(jìn)行填充的。
我們知道,int
型是4個(gè)字節(jié)(每個(gè)字節(jié)有8位),按二進(jìn)制表示出來(lái)就應(yīng)該是:
00000000 00000000 00000000 00000000
按照按字節(jié)填充的原則,step1 的結(jié)果就是將4個(gè)字節(jié)全部填充0,所以得到的結(jié)果仍然是0:
00000000 00000000 00000000 00000000
而 step2 則是將每個(gè)字節(jié)都填充為1 (注意是每個(gè)字節(jié),而不是每個(gè)byte位) ,所以相對(duì)應(yīng)的結(jié)果就應(yīng)該是:
00000001 00000001 00000001 00000001
大家可以自己將上面那個(gè)二進(jìn)制數(shù)轉(zhuǎn)換成十進(jìn)制看看,看看是不是16843009
。
所以嚴(yán)格來(lái)說(shuō),memset函數(shù)本身并不具有初始化的功能,而是一個(gè)單純的按字節(jié)填充函數(shù),只是人們?cè)谑褂玫倪^(guò)程中,擴(kuò)展出了初始化的作用。
字符串初始化有一個(gè)小竅門,我們知道字符串本質(zhì)上是字符數(shù)組,因此它具有兩個(gè)特性,
- 字符串在內(nèi)存里是連續(xù)的,
- 字符串遇
'\0'
結(jié)束。
所以我們?cè)诔跏蓟臅r(shí)候,總是愿意給字符串本身長(zhǎng)度加1的長(zhǎng)度的內(nèi)存進(jìn)行初始化。
char year[4+1]; memset(year, 0, sizeof(year)); strcpy(year,"2018");
指針初始化
一般來(lái)說(shuō),指針都是初始化為NULL
。
int *pnum = NULL; int num = 0; pnum = #
指針是個(gè)讓人又愛(ài)又恨的東西,一般的整形、字符串等,初始化之后就可以直接拿來(lái)用了,可指針如果初始化為NULL
后,沒(méi)有給該指針重新分配內(nèi)存,則會(huì)出現(xiàn)難以預(yù)料的錯(cuò)誤(最最常見(jiàn)的就是操作空指針引起的段錯(cuò)誤)。
在動(dòng)態(tài)內(nèi)存管理中,由于變量的內(nèi)存是分配在堆中的,所以一般用malloc
、calloc
等函數(shù)申請(qǐng)過(guò)動(dòng)態(tài)內(nèi)存,在使用完后需要及時(shí)釋放,一般釋放掉動(dòng)態(tài)內(nèi)存后要及時(shí)將指針置空,這也是很多人容易忽略的。
char *p = NULL; p=(char *)malloc(100); if(NULL == p) { printf("Memory Allocated at: %x\n",p); } else { printf("Not Enough Memory!\n"); } free(p); p = NULL; //這一行給指針置空必不可少,否則很可能后面操作了這個(gè)野指針而不自知,從而導(dǎo)致出現(xiàn)嚴(yán)重的問(wèn)題
很多人經(jīng)常會(huì)犯的一個(gè)錯(cuò)誤,我們知道,在指針作為實(shí)參進(jìn)行參數(shù)傳遞時(shí),該指針就已經(jīng)退化成了數(shù)組,所以很多人就想到用memset
來(lái)對(duì)該指針進(jìn)行初始化:
void fun(char *pstr) { memset(pstr, 0, sizeof(pstr)); ... }
這種寫法是不正確的。我們姑且不管指針能不能用memset
來(lái)進(jìn)行初始化,指針首先保存的是一個(gè)4字節(jié)的地址,所以sizeof(pstr)
永遠(yuǎn)只能 = 4
,這樣的初始化就毫無(wú)意義。
結(jié)構(gòu)體初始化
結(jié)構(gòu)體的初始化就比較簡(jiǎn)單了,基本也都是采用memset
的方式。
typedef struct student { int id; char name[20]; char sex; }STU; STU stu1; memset((char *)&stu1, 0, sizeof(stu1));
關(guān)于初始化結(jié)構(gòu)體的長(zhǎng)度問(wèn)題,也即memset
的第三個(gè)參數(shù),一般來(lái)說(shuō),傳入數(shù)據(jù)類型和變量名效果是一樣的,上例中,下面寫法是等價(jià)的效果:
memset((char *)&stu1, 0, sizeof(STU));
但是對(duì)于結(jié)構(gòu)體數(shù)組的初始化,長(zhǎng)度就需要注意一下了,還是以上例來(lái)做說(shuō)明:
STU stus[10]; memset((char *)&stus, 0, sizeof(stus)); //正確,數(shù)組本身在內(nèi)存里就是連續(xù)的,sizeof取出的就是數(shù)組的字節(jié)長(zhǎng)度 memset((char *)&stus, 0, sizeof(STU)); //錯(cuò)誤,只會(huì)初始化第一個(gè)STU結(jié)構(gòu)體,后面還有9個(gè)STU元素并未初始化 memset((char *)&stus, 0, sizeof(STU)*10); //正確,效果與第一個(gè)是一樣的
有些人習(xí)慣將memset
的第二個(gè)參數(shù)寫成以下形式:
memset((char *)&stu1, 0x00, sizeof(stu1));
只要理解了memset
是按字節(jié)進(jìn)行填充的,就知道這樣寫也是正確的,完全沒(méi)有問(wèn)題。
總結(jié)
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
關(guān)于C語(yǔ)言多線程pthread庫(kù)的相關(guān)函數(shù)說(shuō)明
下面小編就為大家?guī)?lái)一篇關(guān)于C語(yǔ)言多線程pthread庫(kù)的相關(guān)函數(shù)說(shuō)明。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-05-05C++項(xiàng)目開(kāi)發(fā)實(shí)現(xiàn)圖書管理系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了C++項(xiàng)目開(kāi)發(fā)實(shí)現(xiàn)圖書管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-03-03C++11中std::function基礎(chǔ)用法詳解
std::function是C++11標(biāo)準(zhǔn)庫(kù)中提供的一種可調(diào)用對(duì)象的通用類型,它可以存儲(chǔ)任意可調(diào)用對(duì)象,本文就來(lái)和大家講講它的基礎(chǔ)用法,希望對(duì)大家有所幫助2023-04-04利用C語(yǔ)言實(shí)現(xiàn)猜數(shù)字游戲
這篇文章主要為大家詳細(xì)介紹了利用C語(yǔ)言實(shí)現(xiàn)猜數(shù)字游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-02-02C++中產(chǎn)生臨時(shí)對(duì)象的情況及其解決方案
這篇文章主要介紹了C++中產(chǎn)生臨時(shí)對(duì)象的情況及其解決方案,以值傳遞的方式給函數(shù)傳參,類型轉(zhuǎn)換以及函數(shù)需要返回對(duì)象時(shí),并給對(duì)應(yīng)給出了詳細(xì)的解決方案,通過(guò)圖文結(jié)合的方式講解的非常詳細(xì),需要的朋友可以參考下2024-05-05C語(yǔ)言實(shí)現(xiàn)學(xué)生個(gè)人消費(fèi)管理系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言學(xué)生個(gè)人消費(fèi)管理系統(tǒng)開(kāi)發(fā),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-08-08