熱火朝天準(zhǔn)備了兩天的華為機(jī)試,終于考了。考前很虛,很多東西沒(méi)看。好在華為的機(jī)試不考死記硬背的東西,考的編程的規(guī)范、思維。今年的考題只有一道,時(shí)間20分鐘。所有人都是C/C++,沒(méi)有java。真正編代碼的時(shí)間也就五六分鐘,題很簡(jiǎn)單。但我覺(jué)的,還是很有水平的題。
題目:將字符串中的每一個(gè)字符,變成他的下一個(gè)字符。大小寫不變,如果是z就改成a。(其他的沒(méi)說(shuō),如果是Z改成A)。比如,輸入為abcf,輸出應(yīng)為bcdg。
函數(shù)原型給定了:void change_letter(char *pInStr, char *pOutStr)
下面是我的作答,僅供參考:
#include<stdio.h> void change_letter(char *pInStr, char *pOutStr) { if(pInStr == NULL || pOutStr == NULL) return; while(*pInStr) { if(*pInStr == 'z') *pOutStr = 'a'; else if(*pInStr == 'Z') *pOutStr = 'A'; else *pOutStr = *pInStr + 1; pInStr++; pOutStr++; } *pOutStr = '\0'; } void main() { char *test = "guozhaoyanguoqi"; char *out = (char *)malloc(strlen(test)); *out = '\0'; //char out[100]; change_letter(test, out); printf("%s\n", out); }
這里,有幾個(gè)值得注意的地方,這些細(xì)節(jié)也正是考官的加分點(diǎn)或扣分點(diǎn)。
1,函數(shù)的開(kāi)頭要判斷是否申請(qǐng)了內(nèi)存:if(pInStr == NULL || pOutStr == NULL),因?yàn)樗麄円蟛粶?zhǔn)自己加頭文件。默認(rèn)的就是#include<stdio.h>,所以沒(méi)用assert。扯淡的是,考場(chǎng)上由于緊張,我只判斷了pInStr,忘了判斷輸出了!它大爺?shù)?
2,while循環(huán)里可以寫while(*pInStr != '\0'),也可以像我上面寫的那樣while(*pInStr),本著簡(jiǎn)單的原則我采用后者。
3,while循環(huán)出來(lái)之后,輸出字符串一定要加字符串結(jié)束符'\0'.while循環(huán)出來(lái)之后,指針pOutStr指向的是最后一個(gè)字符的下一個(gè)字節(jié),所以這里直接賦值‘\0’就可以了。從程序健壯性考慮一定要賦,但也有意想不到的隱情,請(qǐng)看4
4,主函數(shù)里要給out申請(qǐng)內(nèi)存。有兩種方式,一種是char out[100];直接申請(qǐng)100個(gè)。如果while循環(huán)里不加'\0',將會(huì)看到如下錯(cuò)誤結(jié)果:
如果加上‘\0’的話,一切會(huì)正常。但我想工程上不會(huì)讓這么浪費(fèi)內(nèi)存吧,因此采用第二種方式是必須的。
第二種方式,char *out = (char *)malloc(strlen(test));即用多少申請(qǐng)多少。這時(shí),while循環(huán)出來(lái)之后,從程序結(jié)果上來(lái)看加不加'\0'都是一樣的,但真的么?我將程序修改如下:
#include<stdio.h> #include<stdlib.h> void change_letter(char *pInStr, char *pOutStr) { if(pInStr == NULL || pOutStr == NULL) return; while(*pInStr) { if(*pInStr == 'z') *pOutStr = 'a'; else if(*pInStr == 'Z') *pOutStr = 'A'; else *pOutStr = *pInStr + 1; pInStr++; pOutStr++; } //*pOutStr = '\0'; } void main() { char *test = "guozhaoyanguoqi"; // *out = NULL; // printf("strlen(test) = %d", strlen(test)); char *out = (char*)malloc(strlen(test)); // *out = '\0'; // char out[100]; change_letter(test, out); printf("%s\n", out); printf("strlen(out) = %d\n", strlen(out)); }運(yùn)行結(jié)果:
雖然字符串的結(jié)果是對(duì)的, 但打印出來(lái)strlen(out)=19,而源字符串的長(zhǎng)度是15. 這里就出錯(cuò)了,如果此時(shí)對(duì)out沒(méi)有察覺(jué),用out作其他處理時(shí)會(huì)引發(fā)致命錯(cuò)誤!這些大公司考字符串不是沒(méi)有意義的。
對(duì)了對(duì)比明確,我將主程序修改如下:
void main() { char *test = "guozhaoyanguoqi"; printf("strlen(test) = %d", strlen(test)); char *out = (char*)malloc(strlen(test)); // *out = '\0'; // char out[100]; change_letter(test, out); printf("%s\n", out); printf("strlen(out) = %d\n", strlen(out)); }僅僅是在申明char *out前加了一句話,竟然編譯不通過(guò):
錯(cuò)誤信息是:
C:\Documents and Settings\Administrator\桌面\快捷方式\huawei.c(25) : warning C4013: 'strlen' undefined; assuming extern returning int C:\Documents and Settings\Administrator\桌面\快捷方式\huawei.c(26) : error C2143: syntax error : missing ';' before 'type' C:\Documents and Settings\Administrator\桌面\快捷方式\huawei.c(29) : error C2065: 'out' : undeclared identifier C:\Documents and Settings\Administrator\桌面\快捷方式\huawei.c(29) : warning C4047: 'function' : 'char *' differs in levels of indirection from 'int ' C:\Documents and Settings\Administrator\桌面\快捷方式\huawei.c(29) : warning C4024: 'change_letter' : different types for formal and actual parameter 2 Error executing cl.exe.將主程序改為:
void main() { char *test = "guozhaoyanguoqi"; char *out = NULL; printf("strlen(test) = %d\n", strlen(test)); out = (char*)malloc(strlen(test)); // *out = '\0'; // char out[100]; change_letter(test, out); printf("%s\n", out); printf("strlen(out) = %d\n", strlen(out)); }就能正常編譯運(yùn)行了!乖乖,多虧了寫這個(gè)博客! 看來(lái)c語(yǔ)言里,變量的申明和c++還是有所不同的,c++是即申明即用,但c略微不同,有時(shí)還不易察覺(jué)。像這里,如果沒(méi)有printf("strlen(test) = %d\n", strlen(test));這句話,采用char*out = (char*)malloc(。。。)這種方式?jīng)]有啥問(wèn)題。但就多了一句printf這句話,這樣申明就不中了。看來(lái)還是要規(guī)規(guī)矩矩的來(lái)!先申明char *out = NULL;然后再out=(char*)(。。。)。程序運(yùn)行結(jié)果如下 :
僅僅是對(duì)輸入字符串的值做了改變,輸出和輸入的兩個(gè)字符串的長(zhǎng)度竟然不一樣,這都是while循環(huán)里字符串結(jié)尾沒(méi)有加‘\0’惹的禍!如果您不查看strlen,這個(gè)禍還有可能發(fā)現(xiàn)不了!
將while里的加‘\0’帶上,如上的主程序,運(yùn)行結(jié)果一切正常,如下:
引發(fā)的三個(gè)未解決的疑問(wèn):
1,out的申明長(zhǎng)度問(wèn)題 ,是該 out = (char*)malloc(strlen(test));還是該申明strlen(test)+1個(gè)長(zhǎng)度?我用前者,也沒(méi)有報(bào)錯(cuò)!按理說(shuō),應(yīng)該申明 strlen(test)+1更嚴(yán)密!否則的話最后while出來(lái)之后的*pOutStr的賦值占用的是非法內(nèi)存,是未申明的內(nèi)存。是不是有什么后患???
2,在out = (char*)malloc(strlen(test)+1)申明內(nèi)存之后,是不是要用*out = '\0';對(duì)這塊內(nèi)存進(jìn)行初始化一下? 我以前沒(méi)有這個(gè)意識(shí),最近看書籍,有的書上講到這么初始化一下,不寫這句也沒(méi)發(fā)現(xiàn)啥問(wèn)題。作何解釋???
3,我連著申明兩個(gè)字符串內(nèi)存空間 ,如
char *test = "guozhaoyanguoqi";
char *out == (char*)malloc(strlen(test)+1);
沒(méi)有啥問(wèn)題。當(dāng)?shù)谝粋€(gè)語(yǔ)句后隨便加一個(gè)語(yǔ)句,再申明out的時(shí)候怎么就亂報(bào)錯(cuò)???必須采用
char *out = NULL;
out = (char*)malloc(。。。。)
這種方式才能夠申明通過(guò)。這兩種申明方式本質(zhì)上有什么區(qū)別嗎?? 這就是傳說(shuō)中的字節(jié)對(duì)齊
ps:走出考場(chǎng),我覺(jué)的寫的程序能得90分最起碼,吃飯時(shí)想到忘了判斷pOutStr為NULL,得85分吧!文章寫到一半,發(fā)現(xiàn)申明out的長(zhǎng)度應(yīng)該是strlen(pInStr)+1,得分80. 半路殺出個(gè)程咬金,無(wú)意發(fā)現(xiàn)申明字符串時(shí)的正確方式,應(yīng)該是char*out = NULL;然后out = (char*)malloc(。。。),這樣才嚴(yán)密!我靠,80分不到了。。。。。
歡迎大牛前來(lái)指點(diǎn)后生的三個(gè)疑問(wèn)。
更多文章、技術(shù)交流、商務(wù)合作、聯(lián)系博主
微信掃碼或搜索:z360901061

微信掃一掃加我為好友
QQ號(hào)聯(lián)系: 360901061
您的支持是博主寫作最大的動(dòng)力,如果您喜歡我的文章,感覺(jué)我的文章對(duì)您有幫助,請(qǐng)用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點(diǎn)擊下面給點(diǎn)支持吧,站長(zhǎng)非常感激您!手機(jī)微信長(zhǎng)按不能支付解決辦法:請(qǐng)將微信支付二維碼保存到相冊(cè),切換到微信,然后點(diǎn)擊微信右上角掃一掃功能,選擇支付二維碼完成支付。
【本文對(duì)您有幫助就好】元
