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。有趣的是,这些文件并不一定定义你要使用的函数。例如
5、,如果你要使用宏 EDOM,你的编译程序保证你能通过包含(errnoh) 得到这个宏,而(errnoh)可能定义了宏EDOM,也可能只包含定义这个宏的头文件。更糟的是,编译程序的下一个版本可能会在另一个地方定义宏 EDOM。因此,你不用去寻找真正定义一个函数的头文件并使用这个文件,而应该使用那个被假定为定义了该函数的头文件,这样做是肯定可行的。有几个名字在多个头文件中被定义:NULL,size_t 和 wchar_t。如果你需要其中一个名字的定义,可以使用任意一个定义了该名字的头文件(stddefh是一个较好的选择,它不仅小,而且包含了常用的宏定义和类型定义)。表 122 标准库函数的头文件-
6、函数 头文件-abort stdlib. 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. hc
7、os math. hcosh math. 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. hf
8、gets stdio.hFILE stdio. 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
9、.hgetchar stdio. hgetenv 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
10、_buf setjmp. hlabs 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. h
11、malloc stdlib. hmblen 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.
12、hpow math. hprintf 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 set
13、jmp. hsetlocale locale. 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 stdi
14、o. hsqrt math. hsrand 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 strin
15、g. hstrrchr string. 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. hunge
16、tc stdio. hva_arg 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)或类似的一个头文件中。你的编译程序中可能还有这样一个文件,也可能没有;即使现在有,下一个版本中可能就没有了。因此,还是使用(stadrg h)为好。如果对传递给 c 函数的参数不加约束,就没有一种可移植的方式让 c 函数知道它的参数的数目和类型。如果一个 c 函数的参数数目不定(或类型不定 ),就需要引入某种规则来约束它的参数。例如,printf()函数的第一个参数是一个字符串,它将指示其后都是一些什么样的参数:printf(“ Hello, world! n“ ); /* no more arguments *
18、/printf(“%sn“ , “Hello, world!“); /* 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()函数一样,如果格式字符串和其余参数不匹配,那么结果是没有定义的
19、,你无法知道程序此后将做些什么(但很可能是一些糟糕的事情)。例 123 一个简单的类似 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 * f
20、ormat, . . . )va_list ap; / * ap 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
21、 = va_arg(ap, char * );fputs(s, stdout);break;case d:/ * Get next argument (an int) * /i = va_arg(ap, int);s = int2str(i) ;fputs(s, stdout) ;break scase 0 : format-;breaksdefault :putchar ( * format) ;break;/ * Clean up varying arguments before returning * /va_end(ap);voidmain()simplePrintK “The %s
22、tax rate is %d%. n“ ,“sales“, 6);请参见:122 为了定义我要使用的标准库函数,我需要使用哪些头文件?124 独立(free standing)环境和宿主(hosted)环境之间有什么区别 ?并不是所有的 C 程序员都在编写数据库管理系统和字处理软件,有些 C 程序员要为嵌入式系统(embedded system)编写代码,例如防抱死刹车系统和智能型的烤面包机。嵌入式系统可以不要任何类型的文件系统,也可以基本上不要操作系统。ANSI1SO 标准称这样的系统为“独立(free standing)”系统,并且不要求它们提供除语言本身以外的任何东西。与此相反的情况是程
23、序运行在 RC 机、大型机或者介于两者之间的计算机上,这被称为“宿主(hosted)”环境。即使是开发独立环境的程序员也应该重视标准库:其一,独立环境往往以与标准兼容的方式提供某种功能(例如求平方根函数,重新设计该函数显然很麻烦,因而毫无意义) ;其二,在将嵌入式程序植入烤面包机这样的环境之前,通常要先在 PC 机上测试该程序,而使用标准库函数能增加可同时在测试环境和实际环境中使用的代码的总量。请参见:121 为什么应该使用标准库函数而不要自己编写函数?125 对字符串进行操作的标准库函数有哪些?简单的回答是:(stringh)中的函数。C 语言没有固有的字符串类型,但 c 程序可以用以 NU
24、L(O )字符结束的字符数组来代替字符串。C 程序(以及 c 程序员) 应该保证数组足够大,以容纳所有将要存入的内容。这一点可以通过以下三种方法来实现:(1)分配大量的空间,并假定它足够大,不考虑它不够大时将产生的问题( 这种方法效率高,但在空间不足时会产生严重的问题);(2)总是分配并重新分配所需大小的空间( 如果使用 realloc()函数,这种方法的效率不会太低;这种方法需要使用大量代码,并且会耗费大量运行时间);(3)分配应该足够的空间,并禁止占用更多的空间( 这种方法既安全又高效,但可能会丢失数据)。注意:C+提供了第 4 种方法:直接定义一种 string 类型。由于种种原因,用
25、C+完成这项工作要比用 C 简单得多。即便如此,用 C+还是显得有点麻烦。幸运的是,尽管定义一个标准的 C+ string 类型并不简单,但这种类型使用起来却非常方便。有两组函数可用于 C 语言的字符串处理。第一组函数(strcpy,strcat,等等) 按第一种或第二种方法工作。这组函数完全按需要拷贝字符串或使用内存,因此最好留出所需的全部空间,否则程序就可能出错。大多数 C 程序员使用第一组函数。第二组函数(strncpy,strncat ,等等)按第三种方法工作。这组函数需要知道应该使用多大的空间,并且永远不会占用更多的空间,因此它们会忽略所有已无法容纳的数据。函数 strncpy()和
26、 strncat()中的参数“n”( 第三个)的意义是不同的:对 strncpy()函数来说,它意味着只能使用 “n”个字符的空间,包括末尾的 NUL 字符。strncpy()函数也恰好只拷贝“ n”个字符。如果第二个参数没有这么多字符,strncpy() 函数会用 NUL 字符填充剩余的空间。如果第二个参数有多于“n”个的字符,那么 strncpy()函数在还没有拷贝到 NUL 字符之前就结束工作了。这意味着,在使用 strncpy()函数时,你应该总是自己在目标字符串的末尾加上 NUL 字符,而不要指望 strncpy()函数为你做这项工作。对 strncat()函数来说,它意味着最多只能
27、拷贝“n”个字符,如果需要还要加上一个NUL 字符。因为你真正知道的是目标字符串能存放多少个字符,所以通常你要用 strlen()函数来计算可以拷贝的字符数。函数 strncpy()和 strncat()之间的区别是“历史性”的( 这是一个技术用语,指的是“它对某些人确实起到了一定的作用,并且它可能是处理问题的正确途径,但为什么正确至今仍然说不清楚”)。例 125a 给出了一个使用 strncpy()和 strncat()函数的程序。注意:你应该去了解一下“string-n”函数,虽然它们使用起来有些困难,但用它们编写的程序兼容性更好,错误更少。如果你愿意的话,可以用函数 strcpy()和
28、strcat()重新编写例 125a 中的程序,并用很长的足以溢出缓冲区的参数运行它。会出现什么现象呢?计算机会挂起吗?你会得到“GeneralProtection Exception”或内存信息转储这样的消息吗?请参见 724 中的讨论。例 125a 使用“string n”函数的一个例子# include # include /*Normally, a constant like MAXBUF would be very large, tohelp ensure that the buffer doesnt overflow. Here, its verysmall, to show how the “string-n“ functions prevent it fromever overflowing.*/# define MAXBUF 16intmain (int argc, char* * argv)char bufMAXBUF;int i;bufMAXBUF - 1 = 0;strncpy(buf, argv0, MAXBUF-1);for (i = 1; iargc; +