1、使用C语言的一半价值在于使用其标准库函数。当然,灵活的for循环以及数组和指针之间的相似性也是C语言的重要价值。在解决实际问题时,能方便地操作字符串和文件等对象是最重要的,有些语言能出色地完成其中的一部分工作,另一些语言能出色地完成其中的另一部分工作,然而,没有几种语言能象C语言那样能出色地完成全部工作。 c标准库中还缺少很多函数,例如投有图形函数,甚至没有全屏幕文本操作函数,signal机制也相当弱(见1210),并且根本没有对多任务或使用常规内存以外的内存提供支持。尽管C标准库存在上述缺陷,但它毕竟为所有的程序都提供了一套基本功能,不管这些程序是运行在多任务、多窗口的环境下,还是运行在简单
2、的终端上,或者是运行在一台昂贵的烤面包机上。C标准库中所缺的函数可以从其它途径获得,例如编译程序开发商和第三方的函数库都会提供一些函数,这些函数都是事实上的标准函数。然而,标准库中的函数已经为程序设计提供了一个非常坚实的基础。 121 为什么应该使用标准库函数而不要自己编写函数? 标准库函数有三点好处:准确性、高效性和可移植性。 准确性:编译程序的开发商通常会保证标准库函数的准确性。更重要的是。至少开发商做了全面的检测来证实其准确性,这比你所能做到的更加全面(有些昂贵的测试工具能使这项工作更加容易)。 高效性:优秀的C程序员会大量使用标准库函数,而内行的编译程序开发商也知道这一点。如果开发商能
3、提供一套出色的标准库函数,他就会在竞争中占优势。当对相互竞争的编译程序的效率进行比较时,一套出色的标准库函数将起到决定性的作用。因此,开发商比你更有动力,并且有更多的时间,去开发一套高效的标准库函数。 可移植性:在软件要求不断变化的情况下,标准库函数在任何计算机上,对任何编译程序都具有同样的功能,并且表达同样的含义,因此它们是C程序员屈指可数的几种依靠之一。 有趣的是,你很难找到一项关于标准库函数的最标准的信息。对于每一个函数,都需要有一个(在极少数情况下需要两个)保证能将该函数的原型提供给你的头文件(在调用任何一个函数时,都应该包含其原型,见82)。有趣的是什么呢?这个头文件可能并不是真正包
4、含该函数原型的文件,在有些(非常糟糕!)情况下,甚至由编译程序手册推荐的头文件都不一定正确。对于宏定义,typedef和全局变量,同样会发生这种情况。 为了找到“正确的”头文件,你可以在一份ANSIISO c标准的拷贝中查阅相应的函数。如果你手头没有这样一份拷贝,你可以使用表122。 请参见: 82为什么要使用函数原型? 122 为了定义我要使用的标准库函数,我需要使用哪些头文件? 122 为了定义我要使用的标准库函数,我需要使用哪些头文件? 你需要使用ANSIISO标准规定的你应该使用的那些头文件,见表122。 有趣的是,这些文件并不一定定义你要使用的函数。例如,如果你要使用宏EDOM,你的
5、编译程序保证你能通过包含(errnoh)得到这个宏,而(errnoh)可能定义了宏EDOM,也可能只包含定义这个宏的头文件。更糟的是,编译程序的下一个版本可能会在另一个地方定义宏EDOM。 因此,你不用去寻找真正定义一个函数的头文件并使用这个文件,而应该使用那个被假定为定义了该函数的头文件,这样做是肯定可行的。 有几个名字在多个头文件中被定义:NULL,size_t和wchar_t。如果你需要其中一个名字的定义,可以使用任意一个定义了该名字的头文件(stddefh是一个较好的选择,它不仅小,而且包含了常用的宏定义和类型定义)。 表122标准库函数的头文件- 函数 头文件-abort stdli
6、b. habs stdlib. hacos math. hasctime time. hasin math. hassert assert.hatan math. hatan2 math. hatexit stdlib. hatof stdlib. hatoi stdlib. hatol stdlib. hbsearch stdlib. hBUFSIZ stdio. hcalloc stdlib. hceil math. hclearerr stdio. hclock time. hCLOCKS-PER-SEC time. hclock_t time. hcos math. hcosh mat
7、h. hctime time. hdifftime time. hdiv stdlib. hdiv_t stdlib. hEDOM errno. hEOF stdio. hERANGE errno. herrno errno. hexit stdlib. hEXIT_FAILURE stdlib. hEXIT_SUCCESS stdlib. hexp math. hfabs math. hfclose stdio. hfeof stdio.hferror stdio.hfflush stdio. hfgetc stdio.hfgetpos stdio. hfgets stdio.hFILE s
8、tdio. hFILENAME-MAX stdio. hfloor math. hfmod math. hfopen stdio. hFOPEN_MAX stdio. hfpos_t stdio. hfpnntf stdio. hfputc stdio.hfputs stdio. hhead stdio. hfree stdlib. hfreopen stdio. hfrexp math. hfscanf stdio. hfseek stdio. hfsetpos stdio. hftell stdio. hfwrite stdio. hgetc stdio.hgetchar stdio. h
9、getenv stdlib. hgets stdio.hgmtime time. hHUGE-VAL math.h_IOFBF stdio. h_IOLBF stdio. h_IONBF stdio. hisalnum ctype. hisalpha ctype. hiscntrl ctype. hisdigit ctype. hisgraph ctype. hislower ctype. hisprint ctype. hispunct ctype. hisspace ctype. hisupper ctype. hisxdigit ctype. hjmp_buf setjmp. hlabs
10、 stdlib. hLC_ALL locale. hLC_COLLATE locale. hLC_CTYPE locale. hLC_MONETARY locale. hLC_NUMERIC locale. hLC_TIME locale. hstruct lconv locale. hldexp math. hldiv stdlib. hldiv_t stdlib. hlocaleconv locale. hlocaltime time. hlog math. hlog10 math. hlongjmp setjmp. hL_tmpnam stdio. hmalloc stdlib. hmb
11、len stdlib. hmbstowcs stdlib. hmbtowc stdlib. hMB_CUR_MAX stdlib. hmemchr string. hmemcmp string. hmemcpy string. hmemmove string. hmemset string. hmktime time. hmodf math. hNDEBUG assert. hNULL locale. h.stddef. h.stdio. h.stdlib. h.string. h.time. hoffsetof stddef. hperror stdio.hpow math. hprintf
12、 stdio.hptrdiff_t stddef. hputc stdio. hputchar stdio. hputs stdio. hqsort stdlib. hraise signal. hrand stdlib. hRAND_MAX stdlib. hrealloc stdlib. hremove stdio. hrename stdio. hrewind stdio. hscanf stdio.hSEEK_CUR stdio. hSEEK_END stdio. hSEEK_SET stdio. hsetbuf stdio. hsetjmp setjmp. hsetlocale lo
13、cale. hsetvbuf stdio. hSIGABRT signal. hSIGFPE signal. hSIGILL signal. hSIGINT signal. hsignal signal. hSIGSEGV signal. hSIGTERM signal. hsig_atomic_t signal. hSIG_DFL signal. hSIG_ERR signal. hSIG_IGN signal. hsin math. hsinh math. hsize_t stddef. h.stdlib. h.string. hsprintf stdio. hsqrt math. hsr
14、and stdlib. hsscanf stdio. hstderr stdio.hstdin stdio. hstdout stdio. hstrcat string. hstrchr string. hstrcmp string. hstrcoll string. hstrcpy string. hstrcspn string. hstrerror string.hstrftime time. hstrlen string. hstrncat string. hstrncmp string. hstrncpy string. hstrpbrk string. hstrrchr string
15、. hstrspn string. hstrstr string. hstrtod stdlib. hstrtok string. hstrtol stdlib. hstrtoul stdlib. hstrxfrm string. hsystem stblib. htan math. htanh math. htime time. htime_t time. hstruct tm time. htmpfile stdio. htmpnam stdio. hTMP_MAX stdio. htolower ctype. htoupper ctype. hungetc stdio. hva_arg
16、stdarg. hva_end stdarg. hvalist stdarg. hva_ start stdarg. hvfprintf stdio. hvprintf stdio. hvsprintf stdio. hwchar_t stddef. h. stdlib. hwcstombs stdlib. hwctomb stdlib. h- 请参见: 512 #include(file和#include“file”有什么不同? 121 为什么应该使用标准库函数而不要自己编写函数? 123 怎样编写参数数目可变的函数? 你可以利用(stdargh)头文件,它所定义的一些宏可以让你处理数目可变
17、的参数。 注意:这些宏以前包含在名为(varargsh)或类似的一个头文件中。你的编译程序中可能还有这样一个文件,也可能没有;即使现在有,下一个版本中可能就没有了。因此,还是使用(stadrgh)为好。 如果对传递给c函数的参数不加约束,就没有一种可移植的方式让c函数知道它的参数的数目和类型。如果一个c函数的参数数目不定(或类型不定),就需要引入某种规则来约束它的参数。例如,printf()函数的第一个参数是一个字符串,它将指示其后都是一些什么样的参数:printf( Hello, world! n ); /* no more arguments */printf(%sn , Hello, w
18、orld!); /* one more string argument */printf(%s, %sn , Hello , world!); /* two more string arguments */printf(%s, %dn, Hello, 42); /* one string, one int */ 例123给出了一个简单的类似printf()的函数,它的第一个参数是格式字符串,根据该字符串可以确定其余参数的数目和类型。与真正的printf()函数一样,如果格式字符串和其余参数不匹配,那么结果是没有定义的,你无法知道程序此后将做些什么(但很可能是一些糟糕的事情)。 例123一个简单
19、的类似printf()的函数# include # include # include # include static char *int2str (int n) int minus = (n 0); if (minus) *- - p = -; return p;/* * This is a simple printf-like function that handles only * the format specifiers %, %s, and %d. */voidsimplePrintf(const char * format, . . . ) va_list ap; / * ap
20、 is our argument pointer. * / int i; char * s ; /* * Initialize ap to start with the argument * after format */ va_start(ap, format); for (; * format; format + + ) if (* format !=%) putcharC * format); continue; switch ( * +format) case s : / * Get next argument (a char * ) * / s = va_arg(ap, char *
21、 ); fputs(s, stdout); break; case d:/ * Get next argument (an int) * / i = va_arg(ap, int); s = int2str(i) ; fputs(s, stdout) ; break s case 0 : format-; breaks default :putchar ( * format) ; break; / * Clean up varying arguments before returning * / va_end(ap); void main() simplePrintK The %s tax r
22、ate is %d%. n , sales, 6); 请参见: 122为了定义我要使用的标准库函数,我需要使用哪些头文件? 124 独立(freestanding)环境和宿主(hosted)环境之间有什么区别? 并不是所有的C程序员都在编写数据库管理系统和字处理软件,有些C程序员要为嵌入式系统(embedded system)编写代码,例如防抱死刹车系统和智能型的烤面包机。嵌入式系统可以不要任何类型的文件系统,也可以基本上不要操作系统。ANSI1SO标准称这样的系统为“独立(freestanding)”系统,并且不要求它们提供除语言本身以外的任何东西。与此相反的情况是程序运行在RC机、大型机或
23、者介于两者之间的计算机上,这被称为“宿主(hosted)”环境。 即使是开发独立环境的程序员也应该重视标准库:其一,独立环境往往以与标准兼容的方式提供某种功能(例如求平方根函数,重新设计该函数显然很麻烦,因而毫无意义);其二,在将嵌入式程序植入烤面包机这样的环境之前,通常要先在PC机上测试该程序,而使用标准库函数能增加可同时在测试环境和实际环境中使用的代码的总量。 请参见: 121为什么应该使用标准库函数而不要自己编写函数?125 对字符串进行操作的标准库函数有哪些? 简单的回答是:(stringh)中的函数。 C语言没有固有的字符串类型,但c程序可以用以NUL(O)字符结束的字符数组来代替字符串。 C程序(以及c程序员)应该保证数组足够大,以容纳所有将要存入的内容。这一点可以通过以下三种方法来实现: (1)分配大量的空间,并假定它足够大,不考虑它不够大时将产生的问题(这种方法效率高,但在空间不足时会产生严重的问题); (2)总是分配并重新分配所需大小的空间(如果使用realloc()函数,这种方法的效率不会太低;这种方法需要使用大量代码,并且会耗费大量运行时间); (3)分配应该足够的空间,并禁止