必胜高考网_全国高考备考和志愿填报信息平台

必勝高考網 > 計算機類 > 計算機等級 > 資訊 >

全國計算機等級考試四級上機編程修養(下)

時間: 家輝2 資訊

  }else{

  /* 輸出錯誤信息 */

  printf("error ......\n");

  return ( FALSE );

  }

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

  if ( ch < ’0’ || ch > ’9’ ){

  /* 輸出錯誤信息 */

  printf("error ......\n");

  return ( FALSE );

  }

  ......

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

  7、頭文件中的#ifndef

  ——————————

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

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

  #ifndef <標識>

  #define <標識>

  ......

  ......

  #endif

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

  stdio.h

  #ifndef _STDIO_H_

  #define _STDIO_H_

  ......

  #endif

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

  堆內存分配

  —————

  char*

  AllocStrFromHeap(int len)

  {

  char *pstr;

  if ( len <= 0 ) return NULL;

  return ( char* ) malloc( len );

  } 轉貼于 265考試網

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

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

  記得有一個UNIX上的服務應用程序,大約有幾百的C文件編譯而成,運行測試良好,等使用時,每隔三個月系統就是down一次,搞得許多人焦頭爛額,查不出問題所在。只好,每隔兩個月人工手動重啟系統一次。出現這種問題就是Memery Leak在做怪了,在C/C++中這種問題總是會發生,所以你一定要小心。一個Rational的檢測工作——Purify,可以幫你測試你的程序有沒有內存泄漏。

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

  對于malloc和free的操作有以下規則:

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

  2) 盡量在同一層上使用,不要像上面那種,malloc在函數中,而free在函數外。最好在同一調用層上使用這兩個函數。

  3) malloc分配的內存一定要初始化。free后的指針一定要設置為NULL。注:雖然現在的操作系統(如:UNIX和Win2k/NT)都有進程內存跟蹤機制,也就是如果你

  有沒有釋放的內存,操作系統會幫你釋放。但操作系統依然不會釋放你程序中所有產生了Memory Leak的內存,所以,最好還是你自己來做這個工作。(有的時候不知不覺就出現Memory Leak了,而且在幾百萬行的代碼中找無異于海底撈針,Rational有一個工具叫Purify,可能很好的幫你檢查程序中的Memory Leak)

  9、變量的初始化

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

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

  2) 對一些棧上分配的struct或數組進行初始化。(最好也是清零)不過話又說回來了,初始化也會造成系統運行時間有一定的開銷,所以,也不要對所有的變量做初始化,這個也沒有意義。好的程序員知道哪些變量需要初始化,哪些則不需要。

  如:以下這種情況,則不需要。

  char *pstr; /* 一個字符串 */

  pstr = ( char* ) malloc( 50 );

  if ( pstr == NULL ) exit(0);

  strcpy( pstr, "Hello Wrold" );

  strcpy( pstr, "Hello Wrold" );

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

  char **pstr; /* 一個字符串數組 */

  pstr = ( char** ) malloc( 50 );

  if ( pstr == NULL ) exit(0);

  /* 讓數組中的指針都指向NULL */

  memset( pstr, 0, 50*sizeof(char*) );

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

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

  10、h和c文件的使用

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

  H文件和C文件最大的用處就是聲明和實現分開。這個特性應該是公認的了,但我仍然看到有些人喜歡把函數寫在H文件中,這種習慣很不好。(如果是C++話,對于其模板函數,在VC中只有把實現和聲明都寫在一個文件中,因為 VC不支持export關鍵字)。而且,如果在H文件中寫上函數的實現,你還得在makefile中把頭文件的依賴關系也加上去,這個就會讓你的 makefile很不規范。

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

  例如有一個處理錯誤信息的結構:

  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個函數庫(.lib或是.a)所用到,于是他就被鏈接在這5個.lib或.a中,而如果你的一個程序用到了這5個函數庫中的函數,并且這些函數都用到了這個出錯信息數組。那么這份信息將有5個副本存在于你的執行文件中。如果你的這個errmsg很大的話,而且你用到的函數庫更多的話,你的執行文件也會變得很大。

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

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

  [ 備注 ]

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

  生副本導致執行文件尺寸變大,不僅增加了系統裝載時間,也會讓一個程序在內存中占更多的頁面。而對于errmsg這樣數據,一般來說,在系統運行時不會經常用到,所以還是產生的內存換頁也就不算頻繁。權衡之下,還是只有一份errmsg的效率高。即便是像logmsg這樣頻繁使用的的數據,操作系統的內存調度算法會讓這樣的頻繁使用的頁面常駐于內存,所以也就不會出現內存換頁問題了。

55480 主站蜘蛛池模板: 丹佛斯变频器-丹佛斯压力开关-变送器-广州市风华机电设备有限公司 | 萃取箱-萃取槽-PVC萃取箱厂家-混合澄清槽- 杭州南方化工设备 | 济南品牌包装设计公司_济南VI标志设计公司_山东锐尚文化传播 | 石磨面粉机|石磨面粉机械|石磨面粉机组|石磨面粉成套设备-河南成立粮油机械有限公司 | 动环监控_机房环境监控_DCIM_机房漏水检测-斯特纽 | 丝印油墨_水性油墨_环保油墨油漆厂家_37国际化工 | 电脑知识|软件|系统|数据库|服务器|编程开发|网络运营|知识问答|技术教程文章 - 好吧啦网 | NMRV减速机|铝合金减速机|蜗轮蜗杆减速机|NMRV减速机厂家-东莞市台机减速机有限公司 | 警用|治安|保安|不锈钢岗亭-售货亭价格-垃圾分类亭-移动厕所厂家-苏州灿宇建材 | 专业生物有机肥造粒机,粉状有机肥生产线,槽式翻堆机厂家-郑州华之强重工科技有限公司 | 有机废气处理-rto焚烧炉-催化燃烧设备-VOC冷凝回收装置-三梯环境 | 耐高温风管_耐高温软管_食品级软管_吸尘管_钢丝软管_卫生级软管_塑料波纹管-东莞市鑫翔宇软管有限公司 | 安规_综合测试仪,电器安全性能综合测试仪,低压母线槽安规综合测试仪-青岛合众电子有限公司 | 烘干设备-热泵烘干机_广东雄贵能源设备有限公司 | 气弹簧定制-气动杆-可控气弹簧-不锈钢阻尼器-工业气弹簧-可调节气弹簧厂家-常州巨腾气弹簧供应商 | 河南mpp电力管_mpp电力管生产厂家_mpp电力电缆保护管价格 - 河南晨翀实业 | RTO换向阀_VOC高温阀门_加热炉切断阀_双偏心软密封蝶阀_煤气蝶阀_提升阀-湖北霍科德阀门有限公司 | 冰晶石|碱性嫩黄闪蒸干燥机-有机垃圾烘干设备-草酸钙盘式干燥机-常州市宝康干燥 | 环比机械 | Boden齿轮油泵-ketai齿轮泵-yuken油研-无锡新立液压有限公司 | 新中天检测有限公司青岛分公司-山东|菏泽|济南|潍坊|泰安防雷检测验收 | 踏板力计,制动仪,非接触多功能速度仪,逆反射系数测试仪-创宇 | 恒温水槽与水浴锅-上海熙浩实业有限公司 | 阀门智能定位器_电液动执行器_气动执行机构-赫尔法流体技术(北京)有限公司 | atcc网站,sigma试剂价格,肿瘤细胞现货,人结肠癌细胞株购买-南京科佰生物 | 河南凯邦机械制造有限公司| 送料机_高速冲床送料机_NC伺服滚轮送料机厂家-东莞市久谐自动化设备有限公司 | 波纹补偿器_不锈钢波纹补偿器_巩义市润达管道设备制造有限公司 | 单级/双级旋片式真空泵厂家,2xz旋片真空泵-浙江台州求精真空泵有限公司 | 干法制粒机_智能干法制粒机_张家港市开创机械制造有限公司 | 除甲醛公司-甲醛检测-广西雅居环境科技有限公司 | 无菌实验室规划装修设计-一体化实验室承包-北京洁净净化工程建设施工-北京航天科恩实验室装备工程技术有限公司 | 无锡网站建设_企业网站定制-网站制作公司-阿凡达网络 | 浴室柜-浴室镜厂家-YINAISI · 意大利设计师品牌 | 咿耐斯 |-浙江台州市丰源卫浴有限公司 | 聚合氯化铝价格_聚合氯化铝厂家_pac絮凝剂-唐达净水官网 | 应急灯_消防应急灯_应急照明灯_应急灯厂家-大成智慧官网 | 成都茶楼装修公司 - 会所设计/KTV装修 - 成都朗煜装饰公司 | 缠膜机|缠绕包装机|无纺布包装机-济南达伦特机械设备有限公司 | 酒糟烘干机-豆渣烘干机-薯渣烘干机-糟渣烘干设备厂家-焦作市真节能环保设备科技有限公司 | 低气压试验箱_高低温低气压试验箱_低气压实验箱 |林频试验设备品牌 | 湖南印刷厂|长沙印刷公司|画册印刷|挂历印刷|台历印刷|杂志印刷-乐成印刷 |