日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

编程必备的32个修养,你占了几个?

發(fā)布時間:2023/12/19 编程问答 47 豆豆
生活随笔 收集整理的這篇文章主要介紹了 编程必备的32个修养,你占了几个? 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.

要學嵌入式,關(guān)注@我要學嵌入式,嵌入式猛男的加油站。

什么是好的程序員?是不是懂得很多技術(shù)細節(jié)?還是懂底層編程?還是編程速度比較快?

我覺得都不是。

對于一些技術(shù)細節(jié)來說和底層的技術(shù),只要看幫助,查資料就能找到,對于速度快,只要編得多也就熟能生巧了。

我認為好的程序員應該有以下幾方面的素質(zhì):

  • 1、有專研精神,勤學善問、舉一反三。

  • 2、積極向上的態(tài)度,有創(chuàng)造性思維。

  • 3、與人積極交流溝通的能力,有團隊精神。

  • 4、謙虛謹慎,戒驕戒燥。

  • 5、寫出的代碼質(zhì)量高。包括:代碼的穩(wěn)定、易讀、規(guī)范、易維護、專業(yè)。

這些都是程序員的修養(yǎng),這里我想談談“編程修養(yǎng)”,也就是上述中的第5點。我覺得,如果我要了解一個作者,我會看他所寫的小說,如果我要了解一個畫家,我會看他所畫的圖畫,如果我要了解一個工人,我會看他所做出來的產(chǎn)品。

同樣,如果我要了解一個程序員,我想首先我最想看的就是他的程序代碼,程序代碼可以看出一個程序員的素質(zhì)和修養(yǎng),程序就像一個作品,有素質(zhì)有修養(yǎng)的程序員的作品必然是一圖精美的圖畫,一首美妙的歌曲,一本賞心悅目的小說。

我看過許多程序,沒有注釋,沒有縮進,胡亂命名的變量名,等等,等等,我把這種人統(tǒng)稱為沒有修養(yǎng)的程序,這種程序員,是在做創(chuàng)造性的工作嗎?

不,完全就是在搞破壞,他們與其說是在編程,還不如說是在對源程序進行“加密”,這種程序員,見一個就應該開除一個,因為他編的程序所創(chuàng)造的價值,遠遠小于需要在上面進行維護的價值。

程序員應該有程序員的修養(yǎng),那怕再累,再沒時間,也要對自己的程序負責。我寧可要那種動作慢,技術(shù)一般,但有良好的寫程序風格的程序員,也不要那種技術(shù)強、動作快的“搞破壞”的程序員。有句話叫“字如其人”,我想從程序上也能看出一個程序員的優(yōu)劣。

因為,程序是程序員的作品,作品的好壞直截關(guān)系到程序員的聲譽和素質(zhì)。而“修養(yǎng)”好的程序員一定能做出好的程序和軟件。

有個成語叫“獨具匠心”,意思是做什么都要做得很專業(yè),很用心,如果你要做一個“匠”,也就是造詣高深的人,那么,從一件很簡單的作品上就能看出你有沒有“匠”的特性,我覺得做一個程序員不難,但要做一個“程序匠”就不簡單了。編程序很簡單,但編出有質(zhì)量的程序就難了。

我在這里不討論過深的技術(shù),我只想在一些容易讓人忽略的東西上說一說,雖然這些東西可能很細微,但如果你不注意這些細微之處的話,那么他將會極大的影響你的整個軟件質(zhì)量,以及整個軟件程的實施,所謂“千里之堤,毀于蟻穴”。

“細微之處見真功”,真正能體現(xiàn)一個程序的功底恰恰在這些細微之處。

這就是程序員的——編程修養(yǎng)。

我總結(jié)了在用C/C++語言(主要是C語言)進行程序?qū)懽魃系娜€“修養(yǎng)”,通過這些,你可以寫出質(zhì)量高的程序,同時也會讓看你程序的人漬漬稱道,那些看過你程序的人一定會說:“這個人的編程修養(yǎng)不錯”。

01?


版權(quán)和版本

好的程序員會給自己的每個函數(shù),每個文件,都注上版權(quán)和版本。

對于C/C++的文件,文件頭應該有類似這樣的注釋:

/************************************************************************ * * 文件名:network.c * * 文件描述:網(wǎng)絡通訊函數(shù)集 * * 創(chuàng)建人:Hao Chen, 2003年2月3日 * * 版本號:1.0 * * 修改記錄: * ************************************************************************/而對于函數(shù)來說,應該也有類似于這樣的注釋:/*================================================================ * * 函 數(shù) 名:XXX * * 參 數(shù): * * type name [IN] : descripts * * 功能描述: * * .............. * * 返 回 值:成功TRUE,失敗FALSE * * 拋出異常: * * 作 者:ChenHao 2003/4/2 * ================================================================*/

這樣的描述可以讓人對一個函數(shù),一個文件有一個總體的認識,對代碼的易讀性和易維護性有很大的好處。這是好的作品產(chǎn)生的開始。

02?


縮進、空格、換行、空行、對齊

i) 縮進應該是每個程序都會做的,只要學程序過程序就應該知道這個,但是我仍然看過不縮進的程序,或是亂縮進的程序,如果你的公司還有寫程序不縮進的程序員,請毫不猶豫的開除他吧。

并以破壞源碼罪起訴他,還要他賠償讀過他程序的人的精神損失費。縮進,這是不成文規(guī)矩,我再重提一下吧,一個縮進一般是一個TAB鍵或是4個空格。(最好用TAB鍵)

ii) 空格。空格能給程序代來什么損失嗎?沒有,有效的利用空格可以讓你的程序讀進來更加賞心悅目。而不一堆表達式擠在一起。看看下面的代碼:

ha=(ha*128+*key++)%tabPtr->size;ha = ( ha * 128 + *key++ ) % tabPtr->size;

有空格和沒有空格的感覺不一樣吧。一般來說,語句中要在各個操作符間加空格,函數(shù)調(diào)用時,要以各個參數(shù)間加空格。如下面這種加空格的和不加的:

if ((hProc=OpenProcess(PROCESS_ALL_ACCESS,FALSE,pid))==NULL){ }if ( ( hProc = OpenProcess(PROCESS_ALL_ACCESS, FALSE, pid) ) == NULL ){ }

iii) 換行。不要把語句都寫在一行上,這樣很不好。如:

for(i=0;i<len;i++) if((a[i]<'0'||a[i]>'9')&&(a[i]<'a'||a[i]>'z')) break;

我靠,這種即無空格,又無換行的程序在寫什么啊?加上空格和換行吧。

for ( i=0; i<len; i++) {if ( ( a[i] < '0' || a[i] > '9' ) &&( a[i] < 'a' || a[i] > 'z' ) ) {break;} }

好多了吧?有時候,函數(shù)參數(shù)多的時候,最好也換行,如:

CreateProcess(NULL,cmdbuf,NULL,NULL,bInhH,dwCrtFlags,envbuf,NULL,&siStartInfo,&prInfo );

條件語句也應該在必要時換行:

if ( ch >= '0' || ch <= '9' || ch >= 'a' || ch <= 'z' || ch >= 'A' || ch <= 'Z' )

iv) 空行。不要不加空行,空行可以區(qū)分不同的程序塊,程序塊間,最好加上空行。如:

HANDLE hProcess; PROCESS_T procInfo;/* open the process handle */ if((hProcess = OpenProcess(PROCESS_ALL_ACCESS, FALSE, pid)) == NULL) {return LSE_MISC_SYS; }memset(&procInfo, 0, sizeof(procInfo)); procInfo.idProc = pid; procInfo.hdProc = hProcess; procInfo.misc |= MSCAVA_PROC;return(0);

v) 對齊。用TAB鍵對齊你的一些變量的聲明或注釋,一樣會讓你的程序好看一些。如:

typedef struct _pt_man_t_ {int numProc; /* Number of processes */int maxProc; /* Max Number of processes */int numEvnt; /* Number of events */int maxEvnt; /* Max Number of events */HANDLE* pHndEvnt; /* Array of events */DWORD timeout; /* Time out interval */HANDLE hPipe; /* Namedpipe */TCHAR usr[MAXUSR];/* User name of the process */int numMsg; /* Number of Message */int Msg[MAXMSG];/* Space for intro process communicate */ } PT_MAN_T;

怎么樣?感覺不錯吧。

這里主要講述了如果寫出讓人賞心悅目的代碼,好看的代碼會讓人的心情愉快,讀起代碼也就不累,工整、整潔的程序代碼,通常更讓人歡迎,也更讓人稱道。現(xiàn)在的硬盤空間這么大,不要讓你的代碼擠在一起,這樣它們會抱怨你虐待它們的。好了,用“縮進、空格、換行、空行、對齊”裝飾你的代碼吧,讓他們從沒有秩序的土匪中變成一排排整齊有秩序的正規(guī)部隊吧。

03?


程序注釋

養(yǎng)成寫程序注釋的習慣,這是每個程序員所必須要做的工作。我看過那種幾千行,卻居然沒有一行注釋的程序。

這就如同在公路上駕車卻沒有路標一樣。用不了多久,連自己都不知道自己的意圖了,還要花上幾倍的時間才看明白,這種浪費別人和自己的時間的人,是最為可恥的人。

是的,你也許會說,你會寫注釋,真的嗎?

注釋的書寫也能看出一個程序員的功底。一般來說你需要至少寫這些地方的注釋:文件的注釋、函數(shù)的注釋、變量的注釋、算法的注釋、功能塊的程序注釋。主要就是記錄你這段程序是干什么的?你的意圖是什么?你這個變量是用來做什么的?等等。

不要以為注釋好寫,有一些算法是很難說或?qū)懗鰜淼?#xff0c;只能意會,我承認有這種情況的時候,但你也要寫出來,正好可以訓練一下自己的表達能力。而表達能力正是那種悶頭搞技術(shù)的技術(shù)人員最缺的,你有再高的技術(shù),如果你表達能力不行,你的技術(shù)將不能得到充分的發(fā)揮。因為,這是一個團隊的時代。

好了,說幾個注釋的技術(shù)細節(jié):

i) 對于行注釋(“//”)比塊注釋(“/* */”)要好的說法,我并不是很同意。因為一些老版本的C編譯器并不支持行注釋,所以為了你的程序的移植性,請你還是盡量使用塊注釋。

ii) 你也許會為塊注釋的不能嵌套而不爽,那么你可以用預編譯來完成這個功能。使用“#if 0”和“#endif”括起來的代碼,將不被編譯,而且還可以嵌套。

04


函數(shù)的[in][out]參數(shù)

我經(jīng)常看到這樣的程序:

FuncName(char* str) {int len = strlen(str);..... }char* GetUserName(struct user* pUser) {return pUser->name; }

不!請不要這樣做。

你應該先判斷一下傳進來的那個指針是不是為空。如果傳進來的指針為空的話,那么,你的一個大的系統(tǒng)就會因為這一個小的函數(shù)而崩潰。一種更好的技術(shù)是使用斷言(assert),這里我就不多說這些技術(shù)細節(jié)了。當然,如果是在C++中,引用要比指針好得多,但你也需要對各個參數(shù)進行檢查。

寫有參數(shù)的函數(shù)時,首要工作,就是要對傳進來的所有參數(shù)進行合法性檢查。而對于傳出的參數(shù)也應該進行檢查,這個動作當然應該在函數(shù)的外部,也就是說,調(diào)用完一個函數(shù)后,應該對其傳出的值進行檢查。

當然,檢查會浪費一點時間,但為了整個系統(tǒng)不至于出現(xiàn)“非法操作”或是“Core Dump”的系統(tǒng)級的錯誤,多花這點時間還是很值得的。

05?


對系統(tǒng)調(diào)用的返回進行判斷

繼續(xù)上一條,對于一些系統(tǒng)調(diào)用,比如打開文件,我經(jīng)常看到,許多程序員對fopen返回的指針不做任何判斷,就直接使用了。然后發(fā)現(xiàn)文件的內(nèi)容怎么也讀出不,或是怎么也寫不進去。還是判斷一下吧:

fp = fopen("log.txt", "a"); if ( fp == NULL ){printf("Error: open file error\n");return FALSE; }

其它還有許多啦,比如:socket返回的socket號,malloc返回的內(nèi)存。請對這些系統(tǒng)調(diào)用返回的東西進行判斷。

06?


if語句對出錯的處理

我看見你說了,這有什么好說的。還是先看一段程序代碼吧。

if ( ch >= '0' && ch <= '9' ){/* 正常處理代碼 */ }else{/* 輸出錯誤信息 */printf("error ......\n");return ( FALSE ); }

這種結(jié)構(gòu)很不好,特別是如果“正常處理代碼”很長時,對于這種情況,最好不要用else。先判斷錯誤,如:

if ( ch < '0' || ch > '9' ){/* 輸出錯誤信息 */printf("error ......\n");return ( FALSE ); } /* 正常處理代碼 */ ......

這樣的結(jié)構(gòu),不是很清楚嗎?突出了錯誤的條件,讓別人在使用你的函數(shù)的時候,第一眼就能看到不合法的條件,于是就會更下意識的避免。

07?


頭文件中的#ifndef

千萬不要忽略了頭件的中的#ifndef,這是一個很關(guān)鍵的東西。比如你有兩個C文件,這兩個C文件都include了同一個頭文件。而編譯時,這兩個C文件要一同編譯成一個可運行文件,于是問題來了,大量的聲明沖突。

還是把頭文件的內(nèi)容都放在#ifndef和#endif中吧。不管你的頭文件會不會被多個文件引用,你都要加上這個。一般格式是這樣的:

#ifndef <標識> #define <標識>...... ......#endif

<標識>在理論上來說可以是自由命名的,但每個頭文件的這個“標識”都應該是唯一的。標識的命名規(guī)則一般是頭文件名全大寫,前后加下劃線,并把文件名中的“.”也變成下劃線,如:stdio.h

#ifndef _STDIO_H_ #define _STDIO_H_......#endif

(BTW:預編譯有多很有用的功能。你會用預編譯嗎?)

08?


在堆上分配內(nèi)存

可能許多人對內(nèi)存分配上的“棧 stack”和“堆 heap”還不是很明白。包括一些科班出身的人也不明白這兩個概念。我不想過多的說這兩個東西。簡單的來講,stack上分配的內(nèi)存系統(tǒng)自動釋放,heap上分配的內(nèi)存,系統(tǒng)不釋放,哪怕程序退出,那一塊內(nèi)存還是在那里。stack一般是靜態(tài)分配內(nèi)存,heap上一般是動態(tài)分配內(nèi)存。

由malloc系統(tǒng)函數(shù)分配的內(nèi)存就是從堆上分配內(nèi)存。從堆上分配的內(nèi)存一定要自己釋放。用free釋放,不然就是術(shù)語——“內(nèi)存泄露”(或是“內(nèi)存漏洞”)—— Memory Leak。于是,系統(tǒng)的可分配內(nèi)存會隨malloc越來越少,直到系統(tǒng)崩潰。還是來看看“棧內(nèi)存”和“堆內(nèi)存”的差別吧。

棧內(nèi)存分配 —————

char *AllocStrFromStack() {char pstr[100];return pstr; }

堆內(nèi)存分配 —————

char *AllocStrFromHeap(int len) {char *pstr;if ( len <= 0 ) return NULL;return ( char* ) malloc( len ); }

對于第一個函數(shù),那塊pstr的內(nèi)存在函數(shù)返回時就被系統(tǒng)釋放了。于是所返回的char*什么也沒有。而對于第二個函數(shù),是從堆上分配內(nèi)存,所以哪怕是程序退出時,也不釋放,所以第二個函數(shù)的返回的內(nèi)存沒有問題,可以被使用。但一定要調(diào)用free釋放,不然就是Memory Leak!

在堆上分配內(nèi)存很容易造成內(nèi)存泄漏,這是C/C++的最大的“克星”,如果你的程序要穩(wěn)定,那么就不要出現(xiàn)Memory Leak。所以,我還是要在這里千叮嚀萬囑付,在使用malloc系統(tǒng)函數(shù)(包括calloc,realloc)時千萬要小心。

記得有一個UNIX上的服務應用程序,大約有幾百的C文件編譯而成,運行測試良好,等使用時,每隔三個月系統(tǒng)就是down一次,搞得許多人焦頭爛額,查不出問題所在。只好,每隔兩個月人工手動重啟系統(tǒng)一次。

出現(xiàn)這種問題就是Memery Leak在做怪了,在C/C++中這種問題總是會發(fā)生,所以你一定要小心。一個Rational的檢測工作——Purify,可以幫你測試你的程序有沒有內(nèi)存泄漏。

我保證,做過許多C/C++的工程的程序員,都會對malloc或是new有些感冒。當你什么時候在使用malloc和new時,有一種輕度的緊張和惶恐的感覺時,你就具備了這方面的修養(yǎng)了。

對于malloc和free的操作有以下規(guī)則:

  • 配對使用,有一個malloc,就應該有一個free。(C++中對應為new和delete)

  • 盡量在同一層上使用,不要像上面那種,malloc在函數(shù)中,而free在函數(shù)外。最好在同一調(diào)用層上使用這兩個函數(shù)。

  • malloc分配的內(nèi)存一定要初始化。free后的指針一定要設置為NULL。

  • 注:雖然現(xiàn)在的操作系統(tǒng)(如:UNIX和Win2k/NT)都有進程內(nèi)存跟蹤機制,也就是如果你有沒有釋放的內(nèi)存,操作系統(tǒng)會幫你釋放。但操作系統(tǒng)依然不會釋放你程序中所有產(chǎn)生了Memory Leak的內(nèi)存。

    所以,最好還是你自己來做這個工作。(有的時候不知不覺就出現(xiàn)Memory Leak了,而且在幾百萬行的代碼中找無異于海底撈針,Rational有一個工具叫Purify,可能很好的幫你檢查程序中的Memory Leak)

    09?


    變量的初始化

    接上一條,變量一定要被初始化再使用。C/C++編譯器在這個方面不會像JAVA一樣幫你初始化,這一切都需要你自己來,如果你使用了沒有初始化的變量,結(jié)果未知。好的程序員從來都會在使用變量前初始化變量的。如:

  • 對malloc分配的內(nèi)存進行memset清零操作。(可以使用calloc分配一塊全零的內(nèi)存)

  • 對一些棧上分配的struct或數(shù)組進行初始化。(最好也是清零)

  • 不過話又說回來了,初始化也會造成系統(tǒng)運行時間有一定的開銷,所以,也不要對所有的變量做初始化,這個也沒有意義。好的程序員知道哪些變量需要初始化,哪些則不需要。如:以下這種情況,則不需要。

    char *pstr; /* 一個字符串 */ pstr = ( char* ) malloc( 50 ); if ( pstr == NULL ) exit(0); strcpy( pstr, "Hello Wrold" );

    但如果是下面一種情況,最好進行內(nèi)存初始化。(指針是一個危險的東西,一定要初始化)

    char **pstr; /* 一個字符串數(shù)組 */ pstr = ( char** ) malloc( 50 ); if ( pstr == NULL ) exit(0);/* 讓數(shù)組中的指針都指向NULL */ memset( pstr, 0, 50*sizeof(char*) );

    而對于全局變量,和靜態(tài)變量,一定要聲明時就初始化。因為你不知道它第一次會在哪里被使用。所以使用前初始這些變量是比較不現(xiàn)實的,一定要在聲明時就初始化它們。如:

    Links *plnk = NULL; /* 對于全局變量plnk初始化為NULL */

    10?


    和c文件的使用

    H文件和C文件怎么用呢?

    一般來說,H文件中是declare(聲明),C文件中是define(定義)。因為C文件要編譯成庫文件(Windows下是.obj/.lib,UNIX下是.o/.a),如果別人要使用你的函數(shù),那么就要引用你的H文件,所以,H文件中一般是變量、宏定義、枚舉、結(jié)構(gòu)和函數(shù)接口的聲明,就像一個接口說明文件一樣。而C文件則是實現(xiàn)細節(jié)。

    H文件和C文件最大的用處就是聲明和實現(xiàn)分開。這個特性應該是公認的了,但我仍然看到有些人喜歡把函數(shù)寫在H文件中,這種習慣很不好。(如果是C++話,對于其模板函數(shù),在VC中只有把實現(xiàn)和聲明都寫在一個文件中,因為VC不支持export關(guān)鍵字)。

    而且,如果在H文件中寫上函數(shù)的實現(xiàn),你還得在makefile中把頭文件的依賴關(guān)系也加上去,這個就會讓你的makefile很不規(guī)范。

    最后,有一個最需要注意的地方就是:帶初始化的全局變量不要放在H文件中!

    例如有一個處理錯誤信息的結(jié)構(gòu):

    char* errmsg[] = {/* 0 */ "No error",/* 1 */ "Open file error",/* 2 */ "Failed in sending/receiving a message",/* 3 */ "Bad arguments",/* 4 */ "Memeroy is not enough",/* 5 */ "Service is down; try later",/* 6 */ "Unknow information",/* 7 */ "A socket operation has failed",/* 8 */ "Permission denied",/* 9 */ "Bad configuration file format",/* 10 */ "Communication time out",............ };

    請不要把這個東西放在頭文件中,因為如果你的這個頭文件被5個函數(shù)庫(.lib或是.a)所用到,于是他就被鏈接在這5個.lib或.a中,而如果你的一個程序用到了這5個函數(shù)庫中的函數(shù),并且這些函數(shù)都用到了這個出錯信息數(shù)組。那么這份信息將有5個副本存在于你的執(zhí)行文件中。

    如果你的這個errmsg很大的話,而且你用到的函數(shù)庫更多的話,你的執(zhí)行文件也會變得很大。

    正確的寫法應該把它寫到C文件中,然后在各個需要用到errmsg的C文件頭上加上 extern char* errmsg[]; 的外部聲明,讓編譯器在鏈接時才去管他,這樣一來,就只會有一個errmsg存在于執(zhí)行文件中,而且,這樣做很利于封裝。

    我曾遇到過的最瘋狂的事,就是在我的目標文件中,這個errmsg一共有112個副本,執(zhí)行文件有8M左右。當我把errmsg放到C文件中,并為一千多個C文件加上了extern的聲明后,所有的函數(shù)庫文件尺寸都下降了20%左右,而我的執(zhí)行文件只有5M了。一下子少了3M啊。

    備注

    有朋友對我說,這個只是一個特例,因為,如果errmsg在執(zhí)行文件中存在多個副本時,可以加快程序運行速度,理由是errmsg的多個復本會讓系統(tǒng)的內(nèi)存換頁降低,達到效率提升。像我們這里所說的errmsg只有一份,當某函數(shù)要用errmsg時,如果內(nèi)存隔得比較遠,會產(chǎn)生換頁,反而效率不高。

    這個說法不無道理,但是一般而言,對于一個比較大的系統(tǒng),errmsg是比較大的,所以產(chǎn)生副本導致執(zhí)行文件尺寸變大,不僅增加了系統(tǒng)裝載時間,也會讓一個程序在內(nèi)存中占更多的頁面。

    而對于errmsg這樣數(shù)據(jù),一般來說,在系統(tǒng)運行時不會經(jīng)常用到,所以還是產(chǎn)生的內(nèi)存換頁也就不算頻繁。

    權(quán)衡之下,還是只有一份errmsg的效率高。即便是像logmsg這樣頻繁使用的的數(shù)據(jù),操作系統(tǒng)的內(nèi)存調(diào)度算法會讓這樣的頻繁使用的頁面常駐于內(nèi)存,所以也就不會出現(xiàn)內(nèi)存換頁問題了。

    11?


    出錯信息的處理

    你會處理出錯信息嗎?哦,它并不是簡單的輸出。看下面的示例:

    if ( p == NULL ){printf ( "ERR: The pointer is NULL\n" ); }

    告別學生時代的編程吧。這種編程很不利于維護和管理,出錯信息或是提示信息,應該統(tǒng)一處理,而不是像上面這樣,寫成一個“硬編碼”。第10條對這方面的處理做了一部分說明。如果要管理錯誤信息,那就要有以下的處理:

    /* 聲明出錯代碼 */ #define ERR_NO_ERROR 0 /* No error */ #define ERR_OPEN_FILE 1 /* Open file error */ #define ERR_SEND_MESG 2 /* sending a message error */ #define ERR_BAD_ARGS 3 /* Bad arguments */ #define ERR_MEM_NONE 4 /* Memeroy is not enough */ #define ERR_SERV_DOWN 5 /* Service down try later */ #define ERR_UNKNOW_INFO 6 /* Unknow information */ #define ERR_SOCKET_ERR 7 /* Socket operation failed */ #define ERR_PERMISSION 8 /* Permission denied */ #define ERR_BAD_FORMAT 9 /* Bad configuration file */ #define ERR_TIME_OUT 10 /* Communication time out *//* 聲明出錯信息 */ char* errmsg[] = { /* 0 */ "No error", /* 1 */ "Open file error", /* 2 */ "Failed in sending/receiving a message", /* 3 */ "Bad arguments", /* 4 */ "Memeroy is not enough", /* 5 */ "Service is down; try later", /* 6 */ "Unknow information", /* 7 */ "A socket operation has failed", /* 8 */ "Permission denied", /* 9 */ "Bad configuration file format", /* 10 */ "Communication time out", };/* 聲明錯誤代碼全局變量 */ long errno = 0;/* 打印出錯信息函數(shù) */ void perror( char* info) {if ( info ){printf("%s: %s\n", info, errmsg[errno] );return;}printf("Error: %s\n", errmsg[errno] ); }

    這個基本上是ANSI的錯誤處理實現(xiàn)細節(jié)了,于是當你程序中有錯誤時你就可以這樣處理:

    bool CheckPermission( char* userName ) {if ( strcpy(userName, "root") != 0 ){errno = ERR_PERMISSION_DENIED;return (FALSE); }... }main() {...if (! CheckPermission( username ) ){perror("main()"); } ... }

    一個即有共性,也有個性的錯誤信息處理,這樣做有利同種錯誤出一樣的信息,統(tǒng)一用戶界面,而不會因為文件打開失敗,A程序員出一個信息,B程序員又出一個信息。而且這樣做,非常容易維護。代碼也易讀。

    當然,物極必反,也沒有必要把所有的輸出都放到errmsg中,抽取比較重要的出錯信息或是提示信息是其關(guān)鍵,但即使這樣,這也包括了大多數(shù)的信息。

    12?


    用函數(shù)和循環(huán)語句中的被計算量

    看一下下面這個例子:

    for( i=0; i<1000; i++ ){GetLocalHostName( hostname );... }

    GetLocalHostName的意思是取得當前計算機名,在循環(huán)體中,它會被調(diào)用1000次啊。這是多么的沒有效率的事啊。應該把這個函數(shù)拿到循環(huán)體外,這樣只調(diào)用一次,效率得到了很大的提高。雖然,我們的編譯器會進行優(yōu)化,會把循環(huán)體內(nèi)的不變的東西拿到循環(huán)外面,但是,你相信所有編譯器會知道哪些是不變的嗎?我覺得編譯器不可靠。最好還是自己動手吧。

    同樣,對于常用函數(shù)中的不變量,如:

    GetLocalHostName(char* name) {char funcName[] = "GetLocalHostName";sys_log( "%s begin......", funcName );...sys_log( "%s end......", funcName ); }

    如果這是一個經(jīng)常調(diào)用的函數(shù),每次調(diào)用時都要對funcName進行分配內(nèi)存,這個開銷很大啊。把這個變量聲明成static吧,當函數(shù)再次被調(diào)用時,就會省去了分配內(nèi)存的開銷,執(zhí)行效率也很好。

    13


    函數(shù)名和變量名的命名

    我看到許多程序?qū)ψ兞棵秃瘮?shù)名的取名很草率,特別是變量名,什么a,b,c,aa,bb,cc,還有什么flag1,flag2, cnt1, cnt2,這同樣是一種沒有“修養(yǎng)”的行為。即便加上好的注釋。好的變量名或是函數(shù)名,我認為應該有以下的規(guī)則:

  • 直觀并且可以拼讀,可望文知意,不必“解碼”。

  • 名字的長度應該即要最短的長度,也要能最大限度的表達其含義。

  • 不要全部大寫,也不要全部小寫,應該大小寫都有,如:GetLocalHostName 或是 UserAccount。

  • 可以簡寫,但簡寫得要讓人明白,如:ErrorCode -> ErrCode, ServerListener -> ServLisner,UserAccount -> UsrAcct 等。

  • 為了避免全局函數(shù)和變量名字沖突,可以加上一些前綴,一般以模塊簡稱做為前綴。

  • 全局變量統(tǒng)一加一個前綴或是后綴,讓人一看到這個變量就知道是全局的。

  • 用匈牙利命名法命名函數(shù)參數(shù),局部變量。但還是要堅持“望文生意”的原則。

  • 與標準庫(如:STL)或開發(fā)庫(如:MFC)的命名風格保持一致。


  • 14?


    函數(shù)的傳值和傳指針

    向函數(shù)傳參數(shù)時,一般而言,傳入非const的指針時,就表示,在函數(shù)中要修改這個指針把指內(nèi)存中的數(shù)據(jù)。如果是傳值,那么無論在函數(shù)內(nèi)部怎么修改這個值,也影響不到傳過來的值,因為傳值是只內(nèi)存拷貝。

    什么?你說這個特性你明白了,好吧,讓我們看看下面的這個例程:

    void GetVersion(char* pStr) {pStr = malloc(10);strcpy ( pStr, "2.0" ); }main() {char* ver = NULL;GetVersion ( ver );......free ( ver ); }

    我保證,類似這樣的問題是一個新手最容易犯的錯誤。程序中妄圖通過函數(shù)GetVersion給指針ver分配空間,但這種方法根本沒有什么作用,原因就是——這是傳值,不是傳指針。

    你或許會和我爭論,我分明傳的時指針啊?再仔細看看,其實,你傳的是指針其實是在傳值。

    15


    修改別人程序的修養(yǎng)

    當你維護別人的程序時,請不要非常主觀臆斷的把已有的程序刪除或是修改。我經(jīng)常看到有的程序員直接在別人的程序上修改表達式或是語句。修改別人的程序時,請不要刪除別人的程序,如果你覺得別人的程序有所不妥,請注釋掉,然后添加自己的處理程序,必竟,你不可能100%的知道別人的意圖。

    所以為了可以恢復,請不依賴于CVS或是SourceSafe這種版本控制軟件,還是要在源碼上給別人看到你修改程序的意圖和步驟。這是程序維護時,一個有修養(yǎng)的程序員所應該做的。

    如下所示,這就是一種比較好的修改方法:

    /* * ----- commented by haoel 2003/04/12 ------ * * char* p = ( char* ) malloc( 10 ); * memset( p, 0, 10 ); *//* ------ Added by haoel 2003/04/12 ----- */ char* p = ( char* )calloc( 10, sizeof char ); /* ---------------------------------------- */ ...

    當然,這種方法是在軟件維護時使用的,這樣的方法,可以讓再維護的人很容易知道以前的代碼更改的動作和意圖,而且這也是對原作者的一種尊敬。

    以“注釋 — 添加”方式修改別人的程序,要好于直接刪除別人的程序。

    16


    把相同或近乎相同的代碼形成函數(shù)和宏

    有人說,最好的程序員,就是最喜歡“偷懶”的程序,其中不無道理。

    如果你有一些程序的代碼片段很相似,或直接就是一樣的,請把他們放在一個函數(shù)中。而如果這段代碼不多,而且會被經(jīng)常使用,你還想避免函數(shù)調(diào)用的開銷,那么就把他寫成宏吧。

    千萬不要讓同一份代碼或是功能相似的代碼在多個地方存在,不然如果功能一變,你就要修改好幾處地方,這種會給維護帶來巨大的麻煩,所以,做到“一改百改”,還是要形成函數(shù)或是宏。

    17?


    表達式中的括號

    如果一個比較復雜的表達式中,你并不是很清楚各個操作符的憂先級,即使是你很清楚優(yōu)先級,也請加上括號,不然,別人或是自己下一次讀程序時,一不小心就看走眼理解錯了,為了避免這種“誤解”,還有讓自己的程序更為清淅,還是加上括號吧。

    比如,對一個結(jié)構(gòu)的成員取地址:

    GetUserAge( &( UserInfo->age ) );

    雖然,&UserInfo->age中,->操作符的優(yōu)先級最高,但加上一個括號,會讓人一眼就看明白你的代碼是什么意思。

    再比如,一個很長的條件判斷:

    if ( ( ch[0] >= '0' || ch[0] <= '9' ) &&( ch[1] >= 'a' || ch[1] <= 'z' ) &&( ch[2] >= 'A' || ch[2] <= 'Z' ) )

    括號,再加上空格和換行,你的代碼是不是很容易讀懂了?

    18?


    函數(shù)參數(shù)中的const

    對于一些函數(shù)中的指針參數(shù),如果在函數(shù)中只讀,請將其用const修飾,這樣,別人一讀到你的函數(shù)接口時,就會知道你的意圖是這個參數(shù)是[in],如果沒有const時,參數(shù)表示[in/out],注意函數(shù)接口中的const使用,利于程序的維護和避免犯一些錯誤。

    雖然,const修飾的指針,如:const char* p,在C中一點用也沒有,因為不管你的聲明是不是const,指針的內(nèi)容照樣能改,因為編譯器會強制轉(zhuǎn)換,但是加上這樣一個說明,有利于程序的閱讀和編譯。因為在C中,修改一個const指針所指向的內(nèi)存時,會報一個Warning。這會引起程序員的注意。

    C++中對const定義的就很嚴格了,所以C++中要多多的使用const,const的成員函數(shù),const的變量,這樣會對讓你的代碼和你的程序更加完整和易讀。(關(guān)于C++的const我就不多說了)

    19?


    函數(shù)的參數(shù)個數(shù)(多了請用結(jié)構(gòu))

    函數(shù)的參數(shù)個數(shù)最好不要太多,一般來說6個左右就可以了,眾多的函數(shù)參數(shù)會讓讀代碼的人一眼看上去就很頭昏,而且也不利于維護。如果參數(shù)眾多,還請使用結(jié)構(gòu)來傳遞參數(shù)。這樣做有利于數(shù)據(jù)的封裝和程序的簡潔性。

    也利于使用函數(shù)的人,因為如果你的函數(shù)個數(shù)很多,比如12個,調(diào)用者很容易搞錯參數(shù)的順序和個數(shù),而使用結(jié)構(gòu)struct來傳遞參數(shù),就可以不管參數(shù)的順序。

    而且,函數(shù)很容易被修改,如果需要給函數(shù)增加參數(shù),不需要更改函數(shù)接口,只需更改結(jié)構(gòu)體和函數(shù)內(nèi)部處理,而對于調(diào)用函數(shù)的程序來說,這個動作是透明的。

    20


    函數(shù)的返回類型,不要省略

    我看到很多程序?qū)懞瘮?shù)時,在函數(shù)的返回類型方面不太注意。如果一個函數(shù)沒有返回值,也請在函數(shù)前面加上void的修飾。

    而有的程序員偷懶,在返回int的函數(shù)則什么不修飾(因為如果不修飾,則默認返回int),這種習慣很不好,還是為了原代碼的易讀性,加上int吧。

    所以函數(shù)的返回值類型,請不要省略。

    另外,對于void的函數(shù),我們往往會忘了return,由于某些C/C++的編譯器比較敏感,會報一些警告,所以即使是void的函數(shù),我們在內(nèi)部最好也要加上return的語句,這有助于代碼的編譯。

    21


    goto語句的使用

    N年前,軟件開發(fā)的一代宗師——迪杰斯特拉(Dijkstra)說過:“goto statment is harmful !!”,并建議取消goto語句。因為goto語句不利于程序代碼的維護性。

    這里我也強烈建議不要使用goto語句,除非下面的這種情況:

    #define FREE(p) if(p) { \free(p); \p = NULL; \ }main() {char *fname=NULL, *lname=NULL, *mname=NULL;fname = ( char* ) calloc ( 20, sizeof(char) );if ( fname == NULL ){goto ErrHandle;}lname = ( char* ) calloc ( 20, sizeof(char) );if ( lname == NULL ){goto ErrHandle;}mname = ( char* ) calloc ( 20, sizeof(char) );if ( mname == NULL ){goto ErrHandle;}......ErrHandle:FREE(fname);FREE(lname);FREE(mname);ReportError(ERR_NO_MEMOEY); }

    也只有在這種情況下,goto語句會讓你的程序更易讀,更容易維護。(在用嵌C來對數(shù)據(jù)庫設置游標操作時,或是對數(shù)據(jù)庫建立鏈接時,也會遇到這種結(jié)構(gòu))

    22?


    宏的使用

    很多程序員不知道C中的“宏”到底是什么意思?

    特別是當宏有參數(shù)的時候,經(jīng)常把宏和函數(shù)混淆。我想在這里我還是先講講“宏”,宏只是一種定義,他定義了一個語句塊,當程序編譯時,編譯器首先要執(zhí)行一個“替換”源程序的動作,把宏引用的地方替換成宏定義的語句塊,就像文本文件替換一樣。這個動作術(shù)語叫“宏的展開”

    使用宏是比較“危險”的,因為你不知道宏展開后會是什么一個樣子。例如下面這個宏:

    #define MAX(a, b) a>b?a:b

    當我們這樣使用宏時,沒有什么問題:MAX( num1, num2 ); 因為宏展開后變成 num1>num2?num1:num2;

    但是,如果是這樣調(diào)用的,MAX( 17+32, 25+21 ); 呢,編譯時出現(xiàn)錯誤,原因是,宏展開后變成:17+32>25+21?17+32:25+21,哇,這是什么啊?

    所以,宏在使用時,參數(shù)一定要加上括號,上述的那個例子改成如下所示就能解決問題了。

    #define MAX( (a), (b) ) (a)>(b)?(a):(b)

    即使是這樣,也不這個宏也還是有Bug,因為如果我這樣調(diào)用 MAX(i++, j++); , 經(jīng)過這個宏以后,i和j都被累加了兩次,這絕不是我們想要的。

    所以,在宏的使用上還是要謹慎考慮,因為宏展開是的結(jié)果是很難讓人預料的。而且雖然,宏的執(zhí)行很快(因為沒有函數(shù)調(diào)用的開銷),但宏會讓源代碼澎漲,使目標文件尺寸變大,(如:一個50行的宏,程序中有1000個地方用到,宏展開后會很不得了),相反不能讓程序執(zhí)行得更快(因為執(zhí)行文件變大,運行時系統(tǒng)換頁頻繁)。

    因此,在決定是用函數(shù),還是用宏時得要小心。

    23?


    static的使用

    static關(guān)鍵字,表示了“靜態(tài)”,一般來說,他會被經(jīng)常用于變量和函數(shù)。一個static的變量,其實就是全局變量,只不過他是有作用域的全局變量。比如一個函數(shù)中的static變量:

    char* getConsumerName() {static int cnt = 0;....cnt++; .... }

    cnt變量的值會跟隨著函數(shù)的調(diào)用次而遞增,函數(shù)退出后,cnt的值還存在,只是cnt只能在函數(shù)中才能被訪問。而cnt的內(nèi)存也只會在函數(shù)第一次被調(diào)用時才會被分配和初始化,以后每次進入函數(shù),都不為static分配了,而直接使用上一次的值。

    對于一些被經(jīng)常調(diào)用的函數(shù)內(nèi)的常量,最好也聲明成static(參見第12條)

    但static的最多的用處卻不在這里,其最大的作用的控制訪問,在C中如果一個函數(shù)或是一個全局變量被聲明為static,那么,這個函數(shù)和這個全局變量,將只能在這個C文件中被訪問,如果別的C文件中調(diào)用這個C文件中的函數(shù),或是使用其中的全局(用extern關(guān)鍵字),將會發(fā)生鏈接時錯誤。這個特性可以用于數(shù)據(jù)和程序保密。

    24?


    函數(shù)的代碼尺寸

    一個函數(shù)完成一個具體的功能,一般來說,一個函數(shù)中的代碼最好不要超過600行左右,越少越好,最好的函數(shù)一般在100行以內(nèi),300行左右的孫函數(shù)就差不多了。有證據(jù)表明,一個函數(shù)中的代碼如果超過500行,就會有和別的函數(shù)相同或是相近的代碼,也就是說,就可以再寫另一個函數(shù)。

    另外,函數(shù)一般是完成一個特定的功能,千萬忌諱在一個函數(shù)中做許多件不同的事。函數(shù)的功能越單一越好,一方面有利于函數(shù)的易讀性,另一方面更有利于代碼的維護和重用,功能越單一表示這個函數(shù)就越可能給更多的程序提供服務,也就是說共性就越多。

    雖然函數(shù)的調(diào)用會有一定的開銷,但比起軟件后期維護來說,增加一些運行時的開銷而換來更好的可維護性和代碼重用性,是很值得的一件事。

    25?


    typedef的使用

    typedef是一個給類型起別名的關(guān)鍵字。不要小看了它,它對于你代碼的維護會有很好的作用。比如C中沒有bool,于是在一個軟件中,一些程序員使用int,一些程序員使用short,會比較混亂,最好就是用一個typedef來定義,如:

    typedef char bool;

    一般來說,一個C的工程中一定要做一些這方面的工作,因為你會涉及到跨平臺,不同的平臺會有不同的字長,所以利用預編譯和typedef可以讓你最有效的維護你的代碼,如下所示:

    #ifdef SOLARIS2_5 typedef boolean_t BOOL_T; #else typedef int BOOL_T; #endiftypedef short INT16_T; typedef unsigned short UINT16_T; typedef int INT32_T; typedef unsigned int UINT32_T;#ifdef WIN32 typedef _int64 INT64_T; #else typedef long long INT64_T; #endiftypedef float FLOAT32_T; typedef char* STRING_T; typedef unsigned char BYTE_T; typedef time_t TIME_T; typedef INT32_T PID_T;

    使用typedef的其它規(guī)范是,在結(jié)構(gòu)和函數(shù)指針時,也最好用typedef,這也有利于程序的易讀和可維護性。如:

    typedef struct _hostinfo {HOSTID_T host;INT32_T hostId;STRING_T hostType;STRING_T hostModel;FLOAT32_T cpuFactor;INT32_T numCPUs;INT32_T nDisks;INT32_T memory;INT32_T swap; } HostInfo;typedef INT32_T (*RsrcReqHandler)(void *info,JobArray *jobs,AllocInfo *allocInfo,AllocList *allocList);

    C++中這樣也是很讓人易讀的:

    typedef CArray<HostInfo, HostInfo&> HostInfoArray;

    于是,當我們用其定義變量時,會顯得十分易讀。如:

    HostInfo* phinfo; RsrcReqHandler* pRsrcHand;

    這種方式的易讀性,在函數(shù)的參數(shù)中十分明顯。

    關(guān)鍵是在程序種使用typedef后,幾乎所有的程序中的類型聲明都顯得那么簡潔和清淅,而且易于維護,這才是typedef的關(guān)鍵。

    26?


    為常量聲明宏

    最好不要在程序中出現(xiàn)數(shù)字式的“硬編碼”,如:

    int user[120];

    為這個120聲明一個宏吧。為所有出現(xiàn)在程序中的這樣的常量都聲明一個宏吧。比如TimeOut的時間,最大的用戶數(shù)量,還有其它,只要是常量就應該聲明成宏。如果,突然在程序中出現(xiàn)下面一段代碼,

    for ( i=0; i<120; i++){.... }

    120是什么?為什么會是120?這種“硬編碼”不僅讓程序很讀,而且也讓程序很不好維護,如果要改變這個數(shù)字,得同時對所有程序中這個120都要做修改,這對修改程序的人來說是一個很大的痛苦。所以還是把常量聲明成宏,這樣,一改百改,而且也很利于程序閱讀。

    #define MAX_USR_CNT 120for ( i=0; i<MAX_USER_CNT; i++){.... }

    這樣就很容易了解這段程序的意圖了。

    有的程序員喜歡為這種變量聲明全局變量,其實,全局變量應該盡量的少用,全局變量不利于封裝,也不利于維護,而且對程序執(zhí)行空間有一定的開銷,一不小心就造成系統(tǒng)換頁,造成程序執(zhí)行速度效率等問題。所以聲明成宏,即可以免去全局變量的開銷,也會有速度上的優(yōu)勢。

    27?


    不要為宏定義加分號

    有許多程序員不知道在宏定義時是否要加分號,有時,他們以為宏是一條語句,應該要加分號,這就錯了。當你知道了宏的原理,你會贊同我為會么不要為宏定義加分號的。看一個例子:

    #define MAXNUM 1024;

    這是一個有分號的宏,如果我們這樣使用:

    half = MAXNUM/2;if ( num < MAXNUM )

    等等,都會造成程序的編譯錯誤,因為,當宏展開后,他會是這個樣子的:

    half = 1024;/2;if ( num < 1024; )

    是的,分號也被展進去了,所以造成了程序的錯誤。請相信我,有時候,一個分號會讓你的程序出現(xiàn)成百個錯誤。所以還是不要為宏加最后一個分號,哪怕是這樣:

    #define LINE "================================="#define PRINT_LINE printf(LINE)#define PRINT_NLINE(n) while ( n-- >0 ) { PRINT_LINE; }

    都不要在最后加上分號,當我們在程序中使用時,為之加上分號,

    main() {char *p = LINE;PRINT_LINE; }

    這一點非常符合習慣,而且,如果忘加了分號,編譯器給出的錯誤提示,也會讓我們很容易看懂的。

    28?


    ||和&&的語句執(zhí)行順序

    條件語句中的這兩個“與”和“或”操作符一定要小心,它們的表現(xiàn)可能和你想像的不一樣,這里條件語句中的有些行為需要和說一下:

    express1 || express2

    先執(zhí)行表達式express1如果為“真”,express2將不被執(zhí)行,express2僅在express1為“假”時才被執(zhí)行。因為第一個表達式為真了,整個表達式都為真,所以沒有必要再去執(zhí)行第二個表達式了。

    express1 && express2

    先執(zhí)行表達式express1如果為“假”,express2將不被執(zhí)行,express2僅在express1為“真”時才被執(zhí)行。因為第一個表達式為假了,整個表達式都為假了,所以沒有必要再去執(zhí)行第二個表達式了。

    于是,他并不是你所想像的所有的表達式都會去執(zhí)行,這點一定要明白,不然你的程序會出現(xiàn)一些莫明的運行時錯誤。

    例如,下面的程序:

    if ( sum > 100 && ( ( fp=fopen( filename,"a" ) ) != NULL ) {fprintf(fp, "Warring: it beyond one hundred\n");...... }fprintf( fp, " sum is %id \n", sum ); fclose( fp );

    本來的意圖是,如果sum > 100 ,向文件中寫一條出錯信息,為了方便,把兩個條件判斷寫在一起,于是,如果sum<=100時,打開文件的操作將不會做,最后,fprintf和fclose就會發(fā)現(xiàn)未知的結(jié)果。

    再比如,如果我想判斷一個字符是不是有內(nèi)容,我得判斷這個字符串指針是不為空(NULL)并且其內(nèi)容不能為空(Empty),一個是空指針,一個是空內(nèi)容。我也許會這樣寫:

    if ( ( p != NULL ) && ( strlen(p) != 0 ))

    于是,如果p為NULL,那么strlen(p)就不會被執(zhí)行,于是,strlen也就不會因為一個空指針而“非法操作”或是一個“Core Dump”了。

    記住一點,條件語句中,并非所有的語句都會執(zhí)行,當你的條件語句非常多時,這點要尤其注意。

    29?


    盡量用for而不是while做循環(huán)

    基本上來說,for可以完成while的功能,我是建議盡量使用for語句,而不要使用while語句,特別是當循環(huán)體很大時,for的優(yōu)點一下就體現(xiàn)出來了。

    因為在for中,循環(huán)的初始、結(jié)束條件、循環(huán)的推進,都在一起,一眼看上去就知道這是一個什么樣的循環(huán)。剛出學校的程序一般對于鏈接喜歡這樣來:

    p = pHead;while ( p ){......p = p->next; }

    當while的語句塊變大后,你的程序?qū)⒑茈y讀,用for就好得多:

    for ( p=pHead; p; p=p->next ){.. }

    一眼就知道這個循環(huán)的開始條件,結(jié)束條件,和循環(huán)的推進。大約就能明白這個循環(huán)要做個什么事?而且,程序維護進來很容易,不必像while一樣,在一個編輯器中上上下下的搗騰。

    30?

    請sizeof類型而不是變量

    許多程序員在使用sizeof中,喜歡sizeof變量名,例如:

    int score[100]; char filename[20]; struct UserInfo usr[100];

    在sizeof這三個的變量名時,都會返回正確的結(jié)果,于是許多程序員就開始sizeof變量名。這個習慣很雖然沒有什么不好,但我還是建議sizeof類型。

    我看到過這個的程序:

    pScore = (int*) malloc( SUBJECT_CNT ); memset( pScore, 0, sizeof(pScore) ); ...

    此時,sizeof(pScore)返回的就是4(指針的長度),不會是整個數(shù)組,于是,memset就不能對這塊內(nèi)存進行初始化。為了程序的易讀和易維護,我強烈建議使用類型而不是變量,如:

    對于score:sizeof(int) * 100 /* 100個int */ 對于filename:sizeof(char) * 20 /* 20個char */ 對于usr:sizeof(struct UserInfo) * 100 /* 100個UserInfo */

    這樣的代碼是不是很易讀?一眼看上去就知道什么意思了。

    另外一點,sizeof一般用于分配內(nèi)存,這個特性特別在多維數(shù)組時,就能體現(xiàn)出其優(yōu)點了。如,給一個字符串數(shù)組分配內(nèi)存,

    /* * 分配一個有20個字符串, * 每個字符串長100的內(nèi)存 */char* *p;/* * 錯誤的分配方法 */ p = (char**)calloc( 20*100, sizeof(char) );/* * 正確的分配方法 */ p = (char**) calloc ( 20, sizeof(char*) ); for ( i=0; i<20; i++){/*p = (char*) calloc ( 100, sizeof(char) );*/p[i] = (char*) calloc ( 100, sizeof(char) ); }

    (注:上述語句被注釋掉的是原來的,是錯誤的,由dasherest朋友指正,謝謝)

    為了代碼的易讀,省去了一些判斷,請注意這兩種分配的方法,有本質(zhì)上的差別。

    31?


    不要忽略Warning

    對于一些編譯時的警告信息,請不要忽視它們。雖然,這些Warning不會妨礙目標代碼的生成,但這并不意味著你的程序就是好的。

    竟,并不是編譯成功的程序才是正確的,編譯成功只是萬里長征的第一步,后面還有大風大浪在等著你。從編譯程序開始,不但要改正每個error,還要修正每個warning。這是一個有修養(yǎng)的程序員該做的事。

    一般來說,一面的一些警告信息是常見的:

    1)聲明了未使用的變量。(雖然編譯器不會編譯這種變量,但還是把它從源程序中注釋或是刪除吧)

    2)使用了隱晦聲明的函數(shù)。(也許這個函數(shù)在別的C文件中,編譯時會出現(xiàn)這種警告,你應該這使用之前使用extern關(guān)鍵字聲明這個函數(shù))

    3)沒有轉(zhuǎn)換一個指針。(例如malloc返回的指針是void的,你沒有把之轉(zhuǎn)成你實際類型而報警,還是手動的在之前明顯的轉(zhuǎn)換一下吧)

    4)類型向下轉(zhuǎn)換。(例如:float f = 2.0; 這種語句是會報警告的,編譯會告訴你正試圖把一個double轉(zhuǎn)成float,你正在閹割一個變量,你真的要這樣做嗎?還是在2.0后面加個f吧,不然,2.0就是一個double,而不是float了)

    不管怎么說,編譯器的Warning不要小視,最好不要忽略,一個程序都做得出來,何況幾個小小的Warning呢?

    32?


    書寫Debug版和Release版的程序

    程序在開發(fā)過程中必然有許多程序員加的調(diào)試信息。我見過許多項目組,當程序開發(fā)結(jié)束時,發(fā)動群眾刪除程序中的調(diào)試信息,何必呢?

    為什么不像VC++那樣建立兩個版本的目標代碼?一個是debug版本的,一個是Release版的。那些調(diào)試信息是那么的寶貴,在日后的維護過程中也是很寶貴的東西,怎么能說刪除就刪除呢?

    利用預編譯技術(shù)吧,如下所示聲明調(diào)試函數(shù):?

    #ifdef DEBUG void TRACE(char* fmt, ...) { ...... } #else #define TRACE(char* fmt, ...) #endif

    ?于是,讓所有的程序都用TRACE輸出調(diào)試信息,只需要在在編譯時加上一個參數(shù)“-DDEBUG”,如:

    cc -DDEBUG -o target target.c

    于是,預編譯器發(fā)現(xiàn)DEBUG變量被定義了,就會使用TRACE函數(shù)。而如果要發(fā)布給用戶了,那么只需要把取消“-DDEBUG”的參數(shù),于是所有用到TRACE宏,這個宏什么都沒有,所以源程序中的所有TRACE語言全部被替換成了空。一舉兩得,一箭雙雕,何樂而不為呢?

    順便提一下,兩個很有用的系統(tǒng)宏,一個是“FILE”,一個是“LINE”,分別表示,所在的源文件和行號,當你調(diào)試信息或是輸出錯誤時,可以使用這兩個宏,讓你一眼就能看出你的錯誤,出現(xiàn)在哪個文件的第幾行中。這對于用C/C++做的大工程非常的管用。

    綜上所述32條,都是為了三大目的——

    • 1、程序代碼的易讀性。

    • 2、程序代碼的可維護性,

    • 3、程序代碼的穩(wěn)定可靠性。

    有修養(yǎng)的程序員,就應該要學會寫出這樣的代碼!這是任何一個想做編程高手所必需面對的細小的問題,編程高手不僅技術(shù)要強,基礎要好,而且最重要的是要有“修養(yǎng)”!

    好的軟件產(chǎn)品絕不僅僅是技術(shù),而更多的是整個軟件的易維護和可靠性。

    軟件的維護有大量的工作量花在代碼的維護上,軟件的Upgrade,也有大量的工作花在代碼的組織上,所以好的代碼,清淅的,易讀的代碼,將給大大減少軟件的維護和升級成本。


    精彩推薦:點擊下方圖片即可跳轉(zhuǎn)閱讀

    你真的會讀datasheet嗎?

    STM32奪命100問!你知道幾個?

    STM32中常用的C語言知識點,開始復習!

    蘇聯(lián)的三進制電腦,為什么被二進制干掉了?

    猛男如何學嵌入式,關(guān)注@我要學嵌入式,嵌入式男人的加油站。

    -END-

    我是張巧龍,一名教電子的大學老師,歡迎關(guān)注!

    總結(jié)

    以上是生活随笔為你收集整理的编程必备的32个修养,你占了几个?的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

    如果覺得生活随笔網(wǎng)站內(nèi)容還不錯,歡迎將生活随笔推薦給好友。

    精品国产一区二区三区免费 | 亚洲精品综合欧美二区变态 | 女人高潮特级毛片 | 国产精品美女www爽爽爽视频 | 亚洲精品在线免费看 | 日韩电影在线观看一区二区 | 丝袜美腿在线视频 | 国产99久久精品一区二区300 | 中文字幕av电影下载 | 午夜12点| 日韩女同一区二区三区在线观看 | 亚洲精品www. | 日韩在线视频观看 | 高清在线一区二区 | 久久久久女人精品毛片九一 | 精品国产一区二区三区四 | 亚洲国产欧美一区二区三区丁香婷 | 日韩一级网站 | 中文在线字幕免 | 日韩一区二区免费在线观看 | 中文字幕在线观看1 | 国产午夜三级 | 国产视频久久 | 久久久久免费精品国产 | 在线午夜| 日韩1页| 五月激情片 | wwwav视频| 国产在线观看中文字幕 | 午夜精品一区二区三区视频免费看 | 日韩视频精品在线 | 黄色在线免费观看网站 | 日本在线观看一区二区 | 天天做天天爱夜夜爽 | 波多野结衣久久资源 | 亚洲国产成人在线播放 | 九九热1| 欧美一区二区三区免费观看 | 午夜在线国产 | 国产专区在线看 | 欧美精品一级视频 | 在线观看视频国产一区 | a在线观看视频 | 日韩视频图片 | 丁香花五月 | 成人av电影免费观看 | 四虎影视www | www.五月天激情 | 一本一道波多野毛片中文在线 | 玖玖视频免费在线 | 中文字幕中文字幕中文字幕 | av成人免费网站 | 一区二区三区精品在线视频 | 天天操天天操天天操天天 | 中国老女人日b | 91最新视频在线观看 | 久久免费视频在线观看30 | 精品亚洲国产视频 | av电影不卡在线 | 国产精品久久久久久久久久妇女 | 久久精品a | 亚洲欧美日本一区二区三区 | 青草视频网 | 国产精品第三页 | 久久免费精彩视频 | 午夜国产一区二区三区四区 | 久久综合操 | 日日日操操| 麻豆影视在线免费观看 | 欧美在线观看禁18 | 字幕网在线观看 | 2021国产视频 | 99热手机在线观看 | 久久天堂网站 | 天天爽综合网 | 亚洲精品视频久久 | 午夜婷婷网 | 美女国产精品 | 色播五月激情综合网 | 亚洲色五月 | 成人激情开心网 | 久久黄色a级片 | 久久激情视频 久久 | 在线视频欧美日韩 | 国产大片黄色 | 国产精品无av码在线观看 | 日韩久久精品一区二区三区下载 | 99热在线观看免费 | 六月婷色 | www黄色av| 国产不卡精品视频 | 在线亚洲观看 | 久久免费公开视频 | 久操视频在线观看 | 精品久久1 | 在线观看免费日韩 | 国产视 | 99精品视频免费看 | 成人免费在线视频 | 综合伊人av | 97超碰精品 | 一区二区欧美激情 | 伊人天天操 | 亚洲六月丁香色婷婷综合久久 | 欧美一级久久 | 国产网站在线免费观看 | www.av免费 | 91大神免费在线观看 | 一区二区三区中文字幕在线观看 | 丁香综合五月 | 久久久久久久久久久久99 | 天天射综合网站 | 欧美另类tv| 国产精品免费观看在线 | 黄色精品一区 | 97超在线视频 | 亚洲 欧美 91 | 国产精品久久久久久久久久久杏吧 | 日韩久久精品一区二区 | 日韩黄色在线 | 中文字幕在线不卡国产视频 | 色www免费视频 | www久久久| 亚洲国产一区二区精品专区 | 国产视频每日更新 | 久久人视频 | 国产日韩欧美在线一区 | 四虎在线免费观看视频 | 91精品国自产在线观看 | 久久综合色天天久久综合图片 | 超碰在线公开免费 | 久久爱www.| www.五月天色 | 狠狠色丁香婷婷综合久久片 | 在线观看a视频 | 久久影视网 | 亚洲1级片 | 久久九九网站 | a√天堂资源 | 欧美最猛性xxxxx免费 | 91精品久久久久久综合五月天 | 狠狠久久婷婷 | 天天人人 | 国产精品成人a免费观看 | 中文字幕欲求不满 | 日韩高清激情 | 在线看毛片网站 | 五月婷在线 | 91av视频在线免费观看 | 国产69久久精品成人看 | 国产1区2区3区精品美女 | 色资源在线 | 狠狠干夜夜操天天爽 | 日韩一区正在播放 | a级国产乱理论片在线观看 伊人宗合网 | 国产精品系列在线观看 | 成人一区二区三区在线 | 国产精品久久久久久久久大全 | av导航福利| 精品国产乱码久久久久久1区2匹 | 韩日精品中文字幕 | 国产四虎在线 | 手机看片 | 精品国产一区二区三区四区在线观看 | 天天插天天操天天干 | 97超碰在线久草超碰在线观看 | 亚洲天堂在线观看完整版 | 欧美一级黄色片 | aa级黄色大片 | 91麻豆精品国产自产在线 | 久艹视频在线免费观看 | 黄色电影网站在线观看 | 在线观看精品一区 | 久久天天综合网 | 国产美女视频免费观看的网站 | 久99久中文字幕在线 | 黄色aaa级片 | 色综合久久久网 | 91精品国产乱码久久桃 | 日韩视频欧美视频 | 中文字幕日本在线观看 | 日日操日日 | 国产精品一区二区在线观看 | 91桃色在线免费观看 | 精品人人人人 | av女优中文字幕在线观看 | 91九色蝌蚪视频网站 | 久久久久久久久久久久久久av | 91视频免费播放 | 在线看片91 | 欧美一级在线看 | 天天爱天天射 | 久久婷婷国产色一区二区三区 | 日韩视频免费 | 99热国产在线观看 | 天天干,天天操,天天射 | 久久国产成人午夜av影院潦草 | 国产欧美日韩一区 | 婷婷六月天综合 | 在线观看爱爱视频 | 99电影456麻豆 | 99爱在线观看 | 女女av在线 | 黄色影院在线免费观看 | 国内外成人在线 | 亚洲欧美成aⅴ人在线观看 四虎在线观看 | 国产精品综合久久久 | 国产一区二区三区高清播放 | 精品国产视频在线观看 | 黄色一级片视频 | 免费黄av| 日韩成人中文字幕 | 夜夜夜| 成年人视频在线免费 | 91男人影院 | 99视频在线精品 | 狠狠干狠狠色 | 日韩av中文字幕在线 | 天天色天天操综合 | 国产高清精| 91在线视频免费播放 | 欧美日韩国语 | 91入口在线观看 | 久久精品欧美一区 | 99久久精品电影 | 九九久| 亚州激情视频 | 亚洲第一av在线播放 | 久久精品—区二区三区 | 久久不射影院 | 96在线| 久久免费视频7 | 精品欧美一区二区三区久久久 | 免费精品 | 成人全视频免费观看在线看 | 麻豆免费视频 | 夜夜澡人模人人添人人看 | 亚洲aⅴ免费在线观看 | 成人中文字幕av | 丁香六月在线观看 | 在线观看av中文字幕 | 久久激情日本aⅴ | 激情综合网婷婷 | 久草在线国产 | 亚洲资源 | 久久精品综合网 | www.激情五月.com | 亚洲伦理中文字幕 | 久久精品美女视频网站 | 一区二区三区三区在线 | 久久久国产成人 | 久久精品一区二区三 | 成人蜜桃 | 午夜视频久久久 | 在线免费看片 | 国产手机在线观看视频 | 最近中文字幕高清字幕免费mv | 成人va天堂 | 91九色蝌蚪视频 | 天天操偷偷干 | 成人免费视频网站 | 日韩在线观看电影 | 国产福利a | 麻花传媒mv免费观看 | av国产网站 | 亚洲91av| 亚洲成人动漫在线观看 | 国产色一区 | 国产精品自拍在线 | 97热在线观看 | 欧美成人h版在线观看 | 久久视频在线 | 久草在线手机观看 | 亚洲国产大片 | 欧洲精品视频一区二区 | 亚洲成人xxx| 久爱精品在线 | 一区二区精品在线视频 | 日韩一级成人av | 亚洲aⅴ在线观看 | 五月婷婷网站 | 99视频免费| 亚洲精品日韩av | 天天搞天天 | 欧美视频日韩 | 亚洲人xxx | 男女精品久久 | 中文字幕视频网站 | 成人h在线播放 | 在线一区av | 国产精品区在线观看 | 亚洲天堂色婷婷 | 99中文字幕视频 | 日韩欧美69 | 欧美伦理一区二区 | 欧美日韩在线精品一区二区 | 久久免费在线观看 | 激情五月婷婷综合网 | 欧美一级性 | 久久这里只有精品视频首页 | 亚洲午夜久久久久久久久 | 国产成人一区二区精品非洲 | 狠狠ri| 在线观看视频一区二区三区 | 亚洲国产中文字幕在线观看 | 免费成人av在线 | 中文在线天堂资源 | 一区二区三区久久精品 | 精品国产一区二区三区久久久蜜臀 | 天天躁日日躁狠狠躁 | 久久精品资源 | 久久精品视频在线观看免费 | 在线观看你懂的网站 | 久久九九免费 | 在线观看视频你懂的 | 91精品视频在线看 | 日本精品久久久久影院 | 亚洲精品视频免费在线观看 | 国产精品高清av | 成人在线视频在线观看 | 中文字幕中文中文字幕 | 中文不卡视频 | 在线免费中文字幕 | 九色91福利 | 成人精品一区二区三区电影免费 | 月下香电影| 美女视频黄色免费 | 激情在线网站 | 日韩av看片 | 精品亚洲成a人在线观看 | 天天射射天天 | 亚洲国产视频网站 | 久久成人在线 | 亚洲最新在线视频 | www.伊人网 | 精品国产视频在线 | 国产精品99免视看9 国产精品毛片一区视频 | 日韩精品一区二区三区视频播放 | 高清精品在线 | 97影视| 91久久国产自产拍夜夜嗨 | 国产精品自产拍在线观看中文 | 91天堂在线观看 | 麻豆视传媒官网免费观看 | 国产高清在线视频 | 久久黄色片 | 亚洲高清av| 婷婷在线免费视频 | 亚洲最大av网站 | 久久精品成人欧美大片古装 | 午夜少妇av| 国产精品久久久久毛片大屁完整版 | 91精品在线看 | 午夜精品一区二区三区在线视频 | 四虎永久免费 | 亚洲激情在线观看 | 伊人中文在线 | 亚洲国产精品电影在线观看 | 啪啪小视频网站 | 日韩精品久久久 | 99成人免费视频 | 黄色三级在线 | 天天射天天做 | 欧美中文字幕久久 | 亚洲理论在线观看 | 91高清免费看| 人人爽久久久噜噜噜电影 | 四虎成人精品在永久免费 | 国产精品少妇 | 国产91在线观 | 欧美韩日在线 | 日本爱爱片 | a国产精品| 天天摸天天操天天舔 | 激情综合网天天干 | 精品1区2区 | 在线免费观看视频一区二区三区 | 色婷婷国产在线 | 午夜在线观看一区 | 欧美爽爽爽 | 日韩成人在线免费观看 | 欧美在线91 | 国产精品黄色 | 久久国产精品电影 | 久久精品这里都是精品 | 久久爱影视i | 免费日韩 精品中文字幕视频在线 | 亚洲一区美女视频在线观看免费 | 亚洲乱码精品 | 欧美性大战久久久久 | 99精品偷拍视频一区二区三区 | 日韩av伦理片 | 国产在线观看你懂得 | 国产短视频在线播放 | 欧美激情精品一区 | 国产一区免费观看 | 黄色av网站在线观看免费 | 91夜夜夜| 国产97在线观看 | 日韩精品欧美精品 | 日韩av片免费在线观看 | 亚洲国产免费网站 | 欧美日本一区 | 91免费高清在线观看 | 99综合影院在线 | 国产午夜亚洲精品 | 成年人免费在线观看网站 | 亚洲最大av在线播放 | 国产精品女主播一区二区三区 | 国产精品欧美激情在线观看 | 91大神电影 | 久久久国产电影 | 亚洲精品国产高清 | 色综合久久久久久中文网 | 亚洲精品一区二区精华 | 国产精品久久电影网 | 97人人澡人人爽人人模亚洲 | 亚洲小视频在线观看 | 天天在线免费视频 | 日韩精品一区二区三区免费观看 | 亚洲成av人片在线观看香蕉 | 91成人免费电影 | 18pao国产成视频永久免费 | 久久精品99久久久久久 | 韩国一区二区三区视频 | 4438全国亚洲精品在线观看视频 | 草在线视频 | 国产片网站 | 免费h精品视频在线播放 | 91视频高清 | 在线一二三四区 | 午夜三级在线 | 欧美成人性网 | 久久国产一二区 | 69国产盗摄一区二区三区五区 | www.啪啪.com | 91tv国产成人福利 | 久久久www成人免费毛片 | 亚洲精品午夜久久久久久久久久久 | 91精品视屏 | 日本bbbb摸bbbb | a久久免费视频 | 美女视频黄免费的 | 99爱国产精品 | 婷婷六月中文字幕 | 黄色精品网站 | 国产精品视频永久免费播放 | 免费男女羞羞的视频网站中文字幕 | 超碰日韩在线 | 干干操操 | 亚洲精品在线电影 | 一区久久久 | 成人黄色电影在线 | 激情五月激情综合网 | 有码中文字幕 | 激情欧美xxxx | 综合久久网站 | 免费看国产视频 | 中文字幕二区三区 | 狠狠操欧美 | 激情五月婷婷激情 | 日本一区二区三区视频在线播放 | 欧美成年黄网站色视频 | 国产片免费在线观看视频 | 亚洲欧洲成人 | 久久96国产精品久久99软件 | 蜜桃视频在线观看一区 | 国产色资源| 国产成人一区二区啪在线观看 | 欧美久久电影 | 伊人天天狠天天添日日拍 | 久久国产视屏 | 伊人视频 | 天天天天天天干 | 亚洲免费在线播放视频 | 免费毛片一区二区三区久久久 | 福利电影久久 | 国产精品资源网 | 黄色国产区 | 国产美女网站在线观看 | 日韩高清免费电影 | 日本 在线 视频 中文 有码 | 成年人在线观看网站 | 国产精品电影在线 | 免费在线观看一区 | 亚洲精品国偷拍自产在线观看 | 96看片| 欧美日韩三级在线观看 | 久久久精品国产一区二区 | 亚洲精品乱码久久久久久蜜桃欧美 | 久久手机精品视频 | 中文字幕av专区 | 在线视频欧美日韩 | 日韩精品免费在线观看 | 91夫妻视频| 天堂在线一区 | 麻豆传媒视频在线播放 | 8x成人免费视频 | 国产成人精品亚洲精品 | 九九热在线观看视频 | 欧洲亚洲国产视频 | 在线亚洲高清视频 | 免费欧美精品 | 黄色片软件网站 | a极黄色片 | 久久精品国产免费看久久精品 | 亚洲最新av网站 | 亚洲精品久久久久中文字幕二区 | 亚洲日本中文字幕在线观看 | 久久av免费 | 亚洲天堂视频在线 | 亚洲精品在线免费看 | 一本一道久久a久久精品蜜桃 | 国产理论在线 | 国产精品一二 | 日日夜夜狠狠干 | 免费av在线 | av片在线看 | 超碰国产在线播放 | 国产高清在线免费观看 | 婷婷5月色| 玖玖玖精品 | 99国产视频在线 | 中文在线免费一区三区 | 久久有精品 | 国产91粉嫩白浆在线观看 | 亚洲最大激情中文字幕 | 国产精品久久精品国产 | 在线一区电影 | 国产专区日韩专区 | 免费在线播放视频 | av在线影视| 亚洲欧洲精品一区二区 | 日日碰狠狠躁久久躁综合网 | 成片人卡1卡2卡3手机免费看 | 五月激情电影 | 久久综合电影 | 麻豆成人网| 久久亚洲欧美日韩精品专区 | 天天干天天操天天射 | 狠狠做深爱婷婷综合一区 | 99热这里只有精品国产首页 | 久草在线免费资源站 | 亚洲成人黄色在线 | 在线视频精品 | 亚洲色图美腿丝袜 | 一二三区高清 | 久久精品艹 | www.夜夜操 | 91丨九色丨国产在线 | 色综合亚洲精品激情狠狠 | 中文字幕免费播放 | 天天色综合1| 日日夜夜精品免费视频 | 手机在线黄色网址 | 久久开心激情 | 久久久不卡影院 | 日韩中文在线播放 | 午夜美女福利 | 在线观看久久久久久 | 国产福利精品视频 | 欧美成人免费在线 | 国产精品福利午夜在线观看 | 免费精品在线 | 一区二区三区四区在线免费观看 | 五月天.com | 成人久久亚洲 | 啪啪免费观看网站 | 欧美色图亚洲图片 | 色偷偷88888欧美精品久久 | 欧美日韩中文国产 | 精品国产一区二区三区在线 | av免费黄色 | 91桃色国产在线播放 | 中文在线字幕免费观 | 91在线视频观看免费 | 天天插天天操天天干 | 成人小视频在线播放 | 成人在线免费av | 狠狠干天天操 | 九九九九精品九九九九 | 日韩在线视频看看 | 欧美日韩一区二区三区视频 | 国产精品久久久久三级 | 天天干天天操天天拍 | 97超碰影视 | 久久成人精品电影 | 日韩欧美一区二区三区在线 | 美女久久久 | 久久国内精品 | 亚洲激情在线观看 | 天天爱天天干天天爽 | 成人三级网站在线观看 | 亚洲精品自拍视频在线观看 | 99 精品 在线 | 色天天中文 | 九月婷婷色 | aⅴ视频在线 | 日韩av一区在线观看 | 日韩精品视频在线观看网址 | 1024在线看片 | 国产毛片久久久 | 国产精品久久久一区二区三区网站 | 国产精品久久久久一区二区 | 日韩一区精品 | 91精品国产自产在线观看永久 | 人人澡人人爱 | 97超碰总站| 国产精品一区二区三区久久 | 国产一区免费观看 | 亚洲精品人人 | 久久精品中文字幕免费mv | 精品无人国产偷自产在线 | 成人欧美一区二区三区在线观看 | 久久综合9988久久爱 | 美女久久久久久久 | 欧美日韩精品免费观看 | 免费看污片| av片子在线观看 | 中文字幕在线观看视频免费 | 国产区 在线 | 最近免费中文字幕大全高清10 | 中文字幕在线观看免费高清电影 | 欧美成人xxxxx| 99视频在线免费观看 | 在线观看 国产 | 天天插日日操 | 国产又粗又硬又爽视频 | 久久精久久精 | 91看片一区二区三区 | 欧美国产精品一区二区 | 亚洲欧美日本国产 | 欧美久久电影 | 久草视频网 | 丁香六月国产 | 欧美一区在线看 | 在线视频日韩一区 | 黄色网址av| 97超碰总站 | 日韩毛片在线播放 | 成人蜜桃视频 | av在线电影网站 | 色资源网免费观看视频 | 一区二区三区视频在线 | 一区二区三区视频在线 | 国产午夜亚洲精品 | 四虎欧美 | av免费在线免费观看 | 亚洲成人频道 | 播五月婷婷 | 99久久精品久久久久久清纯 | 国产探花视频在线播放 | av色影院 | 午夜精品一区二区三区免费视频 | 国产五月婷 | 美女黄频免费 | a天堂一码二码专区 | 天天操人| 国产日韩欧美在线影视 | 久久五月天色综合 | 色a网 | 色婷婷六月 | 999久久a精品合区久久久 | 成人国产精品久久久久久亚洲 | 色网站在线观看 | av电影在线观看完整版一区二区 | 免费视频91蜜桃 | 精品一区精品二区高清 | 精品国产精品国产偷麻豆 | 久久久麻豆视频 | 偷拍精品一区二区三区 | 伊人婷婷久久 | 久久久男人的天堂 | 国产精品久久久777 成人手机在线视频 | 97在线视频免费观看 | 国产黄色在线看 | 国产精品av在线免费观看 | www.人人干 | 久久在线视频精品 | 天堂网一区二区三区 | 亚洲日本一区二区在线 | 最新av电影网站 | 91在线看片| 在线免费色 | 午夜黄网 | 777xxx欧美 | 在线观看911视频 | 综合在线亚洲 | 五月激情久久 | a国产精品 | 国产精品九九热 | 久久免费视频国产 | 在线免费看片 | 久久国产精品免费一区二区三区 | av日韩不卡| www久久99| 九九视频一区 | 51久久成人国产精品麻豆 | 天天干天天上 | 天天玩天天操天天射 | 97视频在线观看播放 | 欧美日韩三级在线观看 | 青青五月天 | av电影一区二区三区 | 色资源网在线观看 | 天天射天天爱天天干 | 色综合久久五月天 | 91久久国产露脸精品国产闺蜜 | 亚洲三级视频 | 国产精品成人自产拍在线观看 | 97超碰中文 | 91欧美日韩国产 | 亚洲国产免费看 | 中文永久免费观看 | 日韩精品黄 | 在线免费av播放 | 啪啪凸凸 | 日韩最新av在线 | 国产二区视频在线 | 黄污在线看| 亚洲乱码精品 | 国产日韩精品一区二区 | 久久精品毛片基地 | 精品国产伦一区二区三区观看说明 | 精品久久综合 | 色综合咪咪久久网 | www黄| 婷婷色在线 | 综合色在线 | 高清国产一区 | 成年人国产在线观看 | 天天色综合三 | 五月天中文字幕 | 日韩激情免费视频 | 久久久黄色免费网站 | 日韩精品视频在线观看网址 | 色视频一区 | 欧美xxxxx在线视频 | 美女亚洲精品 | 欧美日韩国产xxx | 日韩在线免费电影 | 欧美一二三区播放 | 久久精品在线视频 | 亚洲 中文 欧美 日韩vr 在线 | 欧美激情第十页 | 日本一区二区三区免费看 | 久久99精品久久久久久久久久久久 | 黄色网址在线播放 | 国产一区成人 | 婷婷色 亚洲 | 久久99精品国产99久久 | 国产精品精品 | 97超碰免费| 五月婷婷在线观看视频 | 久久www免费视频 | 在线播放精品一区二区三区 | 久久国产精品99久久久久久丝袜 | 久久国产亚洲 | 欧美夫妻生活视频 | 欧美精品久久久久久久久久白贞 | 伊人色**天天综合婷婷 | 国产高清视频免费在线观看 | 久久精品国产一区二区 | 在线观看黄a | 97超碰资源总站 | 中文字幕日韩一区二区三区不卡 | 91麻豆精品国产91久久久无需广告 | 91人人澡| 99久久久久久久 | 日日精品| 中文字幕乱码亚洲精品一区 | 免费福利视频网 | 97品白浆高清久久久久久 | 在线观看av网站 | 99精品热 | 最近乱久中文字幕 | 精产嫩模国品一二三区 | 国产精品视频一二三 | av视屏在线播放 | 国产精久久久久久妇女av | 天天草天天操 | 亚洲精品综合一区二区 | 久久久精品二区 | 国产午夜精品一区二区三区嫩草 | 日韩欧美一区二区三区在线观看 | 五月婷婷久久丁香 | 91大神视频网站 | 九九精品视频在线观看 | 国产无限资源在线观看 | 亚洲男男gaygay无套 | 69国产精品成人在线播放 | 亚洲精品国产精品国 | 日韩日韩日韩日韩 | 中文字幕在线观看资源 | 久久久久国产精品视频 | 在线色资源 | 免费能看的黄色片 | 亚洲国产精品第一区二区 | 日韩免费观看av | 亚洲影视九九影院在线观看 | 日韩手机在线观看 | 天天在线免费视频 | 精品国产自在精品国产精野外直播 | 欧美激情第一区 | 一区二区三区四区不卡 | 99久久精品久久亚洲精品 | 亚州欧美视频 | 久久久国产精品一区二区三区 | 91网页版在线观看 | 91精品成人| www日日夜夜| 美女久久久久 | 91最新在线视频 | 少妇精69xxtheporn | 国产精品久久久久久久久久直播 | 国产在线精品区 | 日韩午夜小视频 | 毛片a级片| 久久这里只有精品23 | 欧美激情视频一区二区三区 | 国产精品九色 | 国产毛片aaa| 久久久久激情电影 | 国产999视频在线观看 | 久久r精品 | 国产成人久久av977小说 | 91人人在线 | 五月婷婷在线播放 | 国产二区电影 | adn—256中文在线观看 | 天天综合天天做天天综合 | 五月天堂色 | www.久久视频 | 色黄久久久久久 | 超碰com | 精品视频在线视频 | 在线看欧美 | 狠狠操天天操 | 日韩美女一级片 | 欧美性超爽 | 久久久国产成人 | 黄色三级免费观看 | 婷婷电影在线观看 | 精品理论片 | 成年人免费看片 | 黄色成人毛片 | 欧美亚洲国产精品久久高清浪潮 | 精品国产乱码久久久久久三级人 | 日韩高清无线码2023 | 97香蕉久久国产在线观看 | 91视频黄色| 九九久久精品视频 | 夜夜操狠狠干 | 国产亚洲aⅴaaaaaa毛片 | 又爽又黄又刺激的视频 | 中文字幕高清在线 | 中文字幕免费久久 | 天天爽人人爽夜夜爽 | 不卡视频在线看 | 国产超碰在线 | 色综合久久天天 | 91在线看网站 | 久久爱www.| 人人涩| 国精产品满18岁在线 | 久久综合狠狠综合久久狠狠色综合 | 一区二区激情 | 日本h视频在线观看 | 午夜av电影| 久久免费国产视频 | 亚洲狠狠操 | 欧美午夜理伦三级在线观看 | 手机在线小视频 | 久久九九视频 | 国产最顶级的黄色片在线免费观看 | 国产97在线播放 | 精品一区二区三区久久久 | 久久免费视频一区 | av看片在线观看 | 亚洲区视频在线观看 | 中文字幕国产在线 | 97视频在线 | 亚洲va男人天堂 | 中文国产字幕 | 成人毛片在线视频 | 亚洲综合色激情五月 | 国产精品久久久久永久免费观看 | 国产精品久久久久国产a级 激情综合中文娱乐网 | 色天天综合久久久久综合片 | 成人一级黄色片 | 亚洲丝袜一区 | 国产精品亚洲人在线观看 | 欧美一级视频免费看 | 丁香国产视频 | 婷婷国产v亚洲v欧美久久 | 亚洲午夜大片 | 久久精品欧美 | 日韩区欧美久久久无人区 | 久草视频99 | 欧美大片aaa | 五月婷社区| 亚洲精品综合欧美二区变态 | 国产剧情在线一区 | 国产精品久久久999 国产91九色视频 | 国产成人久久久久 | 国产精品视频免费观看 | 日韩精品一区二区不卡 | 色在线亚洲 | 九九九在线观看视频 | 国产精品欧美激情在线观看 | 91九色精品国产 | 黄色亚洲在线 | 色国产在线 | 久久不卡免费视频 | 丁香久久激情 | 亚洲人人射 | 在线精品观看 | 国产国产人免费人成免费视频 | 免费在线观看黄 | 免费在线黄色av | 午夜黄色影院 | 国外av在线 | 蜜桃视频精品 | 91精品视频在线免费观看 | 欧美日韩国产一区二区三区 | 在线观看中文字幕dvd播放 | 欧美做受xxx | 久操视频在线播放 | 久久久精品网 | 成人久久久精品国产乱码一区二区 | 日韩激情中文字幕 | 美女视频黄免费的久久 | 色综合天天天天做夜夜夜夜做 | 久草免费资源 | 国产精品视频999 | 亚洲视频在线看 | 亚洲人成人99网站 | 久久久久久久久久久久久久电影 | 91片黄在线观| 欧美日一级片 | 91视频黄色 | av中文字幕电影 | 亚洲影音先锋 | 国产成人精品一区二区在线 | 色婷婷www | 91精品网站 | 99热国产在线中文 | 99视频精品免费观看, | 激情视频一区 | 二区三区在线观看 | 天天做天天爱天天综合网 | 久久久影院一区二区三区 | 国产精品久久影院 | 久久激情视频 | 午夜精品久久 | 亚洲婷婷综合色高清在线 | 成年性视频| 91麻豆精品国产91久久久使用方法 | 97视频免费 | 天堂av在线网 | 午夜国产影院 | 欧美做受高潮1 | 最新av在线播放 | 欧美性生活一级片 | 亚洲黄色av | 狠狠色丁香婷婷综合久小说久 | 成人国产一区 | 日韩一区二区三区免费电影 | 在线观看91精品国产网站 | 超碰日韩 | 欧美激情综合色 | 麻豆国产在线视频 | 中文字幕永久免费 | 亚洲精品tv久久久久久久久久 | 欧美精品一区在线 | 久久精品国产一区二区电影 | 日韩二级毛片 | 国产精品不卡在线观看 | 国产精品永久免费视频 | 国产成人三级在线 | 一级做a爱片性色毛片www | 欧美亚洲精品在线观看 | 国产精品国产三级国产 | 欧美性天天 | 久久久高清免费视频 | 成人久久久精品国产乱码一区二区 | 国产99在线免费 | 欧美日韩国产三级 | 99re在线视频观看 | 欧美黑人xxxx猛性大交 | 精品免费国产一区二区三区四区 | 国产精品久久中文字幕 | 久久久男人的天堂 | 97电影在线观看 | 免费看的黄色小视频 | 不卡中文字幕av | 久草在线视频免赞 | 韩日三级在线 |