文件名称:你必须知道的495个C语言问题(PDF)
文件大小:1.37MB
文件格式:PDF
更新时间:2012-10-05 04:12:27
495 C语言
老资源。
目录
1 声明和初始化1
1.1 我如何决定使用那种整数类型? . . . . . . . . . . . . . . . . . . . 1
1.2 64 位机上的64 位类型是什么样的? . . . . . . . . . . . . . . . . 1
1.3 怎样定义和声明全局变量和函数最好? . . . . . . . . . . . . . . . 2
1.4 extern 在函数声明中是什么意思? . . . . . . . . . . . . . . . . . 2
1.5 关键字auto 到底有什么用途? . . . . . . . . . . . . . . . . . . . 2
1.6 我似乎不能成功定义一个链表。我试过typedef struct f char
*item; NODEPTR next; g *NODEPTR; 但是编译器报了错误信
息。难道在C语言中一个结构不能包含指向自己的指针吗? . . . . 3
1.7 怎样建立和理解非常复杂的声明?例如定义一个包含N 个指向返
回指向字符的指针的函数的指针的数组? . . . . . . . . . . . . . . 3
1.8 函数只定义了一次, 调用了一次, 但编译器提示非法重定义了。. . 4
1.9 main() 的正确定义是什么? void main() 正确吗? . . . . . . . . . 4
1.10 对于没有初始化的变量的初始值可以作怎样的假定?如果一个全
局变量初始值为“零”, 它可否作为空指针或浮点零? . . . . . . . 4
1.11 代码int f() f char a[] = "Hello, world!";g 不能编译。. . . . . . . 5
1.12 这样的初始化有什么问题?char *p = malloc(10); 编译器提示“非
法初始式” 云云。. . . . . . . . . . . . . . . . . . . . . . . . . . . 5
1.13 以下的初始化有什么区别?char a[] = "string literal"; char *p =
"string literal"; 当我向p[i] 赋值的时候, 我的程序崩溃了。. . . . 5
1.14 我总算弄清除函数指针的声明方法了, 但怎样才能初始化呢? . . 5
2 结构、联合和枚举7
2.1 声明struct x1 f . . . g; 和typedef struct f . . . g x2; 有什么不同? . 7
2.2 为什么struct x f . . . g; x thestruct; 不对? . . . . . . . . . . . . . 7
2.3 一个结构可以包含指向自己的指针吗? . . . . . . . . . . . . . . . 7
2.4 在C 语言中实现抽象数据类型什么方法最好? . . . . . . . . . . . 7
2.5 在C 中是否有模拟继承等面向对象程序设计特性的好方法? . . . 7
i
目录ii
2.6 我遇到这样声明结构的代码: struct name f int namelen; char
namestr[1];g; 然后又使用一些内存分配技巧使namestr 数组用起
来好像有多个元素。这样合法和可移植吗? . . . . . . . . . . . . 8
2.7 是否有自动比较结构的方法? . . . . . . . . . . . . . . . . . . . . 8
2.8 如何向接受结构参数的函数传入常数值? . . . . . . . . . . . . . . 8
2.9 怎样从/向数据文件读/写结构? . . . . . . . . . . . . . . . . . . . 9
2.10 我的编译器在结构中留下了空洞, 这导致空间浪费而且无法与外
部数据文件进行”二进制” 读写。能否关掉填充, 或者控制结构域
的对齐方式? . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 9
2.11 为什么sizeof 返回的值大于结构的期望值, 是不是尾部有填充? . . 9
2.12 如何确定域在结构中的字节偏移? . . . . . . . . . . . . . . . . . 9
2.13 怎样在运行时用名字访问结构中的域? . . . . . . . . . . . . . . . 10
2.14 程序运行正确, 但退出时却“core dump”了,怎么回事? . . . . . 10
2.15 可以初始化一个联合吗? . . . . . . . . . . . . . . . . . . . . . . . 10
2.16 枚举和一组预处理的#define 有什么不同? . . . . . . . . . . . . 10
2.17 有什么容易的显示枚举值符号的方法? . . . . . . . . . . . . . . . 11
3 表达式13
3.1 为什么这样的代码: a[i] = i++; 不能工作? . . . . . . . . . . . . 13
3.2 使用我的编译器,下面的代码int i=7; printf("%dnn", i++ * i++);
返回49?不管按什么顺序计算, 难道不该打印出56吗? . . . . . . 13
3.3 对于代码int i = 3; i = i++; 不同编译器给出不同的结果, 有的为
3, 有的为4, 哪个是正确的? . . . . . . . . . . . . . . . . . . . . . 14
3.4 这是个巧妙的表达式: a ˆ= b ˆ= a ˆ= b 它不需要临时变量就可
以交换a 和b 的值。. . . . . . . . . . . . . . . . . . . . . . . . . 14
3.5 我可否用括号来强制执行我所需要的计算顺序? . . . . . . . . . . 14
3.6 可是&& 和|| 运算符呢?我看到过类似while((c = getchar()) !=
EOF && c != ’nn’) 的代码⋯⋯ . . . . . . . . . . . . . . . . . . 14
3.7 我怎样才能理解复杂表达式?“序列点” 是什么? . . . . . . . . . 15
3.8 那么, 对于a[i] = i++; 我们不知道a[] 的哪一个分量会被改写,但i
的确会增加1, 对吗? . . . . . . . . . . . . . . . . . . . . . . . . . 15
3.9 ++i 和i++ 有什么区别? . . . . . . . . . . . . . . . . . . . . . . 15
3.10 如果我不使用表达式的值, 我应该用++i 或i++ 来自增一个变量
吗? . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 15
3.11 为什么如下的代码int a = 100, b = 100; long int c = a * b; 不能
工作? . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 15
3.12 我需要根据条件把一个复杂的表达式赋值给两个变量中的一
个。可以用下边这样的代码吗? ((condition) ? a : b) = complicated
expression; . . . . . . . . . . . . . . . . . . . . . . . . . . . 16
目录iii
4 指针17
4.1 我想声明一个指针并为它分配一些空间, 但却不行。这些代码有
什么问题?char *p; *p = malloc(10); . . . . . . . . . . . . . . . . 17
4.2 *p++ 自增p 还是p 所指向的变量? . . . . . . . . . . . . . . . . 17
4.3 我有一个char * 型指针正巧指向一些int 型变量, 我想跳过它们。
为什么如下的代码((int *)p)++; 不行? . . . . . . . . . . . . . . 17
4.4 我有个函数,它应该接受并初始化一个指针void f(int *ip) f static
int dummy = 5; ip = &dummy;g 但是当我如下调用时: int *ip;
f(ip); 调用者的指针却没有任何变化。. . . . . . . . . . . . . . . 18
4.5 我能否用void** 指针作为参数, 使函数按引用接受一般指针? . . 18
4.6 我有一个函数extern int f(int *); 它接受指向int 型的指针。我怎
样用引用方式传入一个常数?下面这样的调用f(&5); 似乎不行。. 18
4.7 C 有“按引用传递” 吗? . . . . . . . . . . . . . . . . . . . . . . . 18
4.8 我看到了用指针调用函数的不同语法形式。到底怎么回事? . . . 19
4.9 我怎样把一个int 变量转换为char * 型?我试了类型转换, 但是不
行。. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 19
5 空(null) 指针21
5.1 臭名昭著的空指针到底是什么? . . . . . . . . . . . . . . . . . . . 21
5.2 怎样在程序里获得一个空指针? . . . . . . . . . . . . . . . . . . . 21
5.3 用缩写的指针比较“if(p)” 检查空指针是否可靠?如果空指针的内
部表达不是0 会怎么样? . . . . . . . . . . . . . . . . . . . . . . . 22
5.4 NULL 是什么, 它是怎么定义的? . . . . . . . . . . . . . . . . . . 23
5.5 在使用非全零作为空指针内部表达的机器上, NULL 是如何定义
的? . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 23
5.6 如果NULL 定义成#define NULL ((char *)0) 难道不就可以向函
数传入不加转换的NULL 了吗? . . . . . . . . . . . . . . . . . . 23
5.7 如果NULL 和0 作为空指针常数是等价的, 那我到底该用哪一个
呢? . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 24
5.8 但是如果NULL 的值改变了, 比如在使用非零内部空指针的机器
上, 难道用NULL (而不是0) 不是更好吗? . . . . . . . . . . . . . 24
5.9 用预定义宏#define Nullptr(type) (type *)0 帮助创建正确类型的
空指针。. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 24
5.10 这有点奇怪。NULL 可以确保是0, 但空(null) 指针却不一定? . . 24
5.11 为什么有那么多关于空指针的疑惑?为什么这些问题如此经常地
出现? . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 25
5.12 我很困惑。我就是不能理解这些空指针一类的东西。. . . . . . . 25
5.13 考虑到有关空指针的所有这些困惑, 难道把要求它们内部表达都
必须为0 不是更简单吗? . . . . . . . . . . . . . . . . . . . . . . . 26
5.14 说真的, 真有机器用非零空指针吗, 或者不同类型用不同的表达? 26
目录iv
5.15 运行时的“空指针赋值” 错误是什么意思? . . . . . . . . . . . . . 26
6 数组和指针27
6.1 我在一个源文件中定义了char a[6], 在另一个中声明了extern
char *a 。为什么不行? . . . . . . . . . . . . . . . . . . . . . . . 27
6.2 可是我听说char a[ ] 和char *a 是一样的。. . . . . . . . . . . . . 27
6.3 那么, 在C 语言中“指针和数组等价” 到底是什么意思? . . . . . 28
6.4 那么为什么作为函数形参的数组和指针申明可以互换呢? . . . . . 28
6.5 如果你不能给它赋值, 那么数组如何能成为左值呢? . . . . . . . . 29
6.6 现实地讲, 数组和指针地区别是什么? . . . . . . . . . . . . . . . 29
6.7 有人跟我讲, 数组不过是常指针。. . . . . . . . . . . . . . . . . . 29
6.8 我遇到一些“搞笑” 的代码, 包含5["abcdef"] 这样的“表达式”。
这为什么是合法的C 表达式呢? . . . . . . . . . . . . . . . . . . 29
6.9 既然数组引用会蜕化为指针, 如果arr 是数组, 那么arr 和&arr 又
有什么区别呢? . . . . . . . . . . . . . . . . . . . . . . . . . . . . 30
6.10 我如何声明一个数组指针? . . . . . . . . . . . . . . . . . . . . . 30
6.11 我如何在运行期设定数组的大小?我怎样才能避免固定大小的数
组? . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 30
6.12 我如何声明大小和传入的数组一样的局部数组? . . . . . . . . . . 30
6.13 我该如何动态分配多维数组? . . . . . . . . . . . . . . . . . . . . 31
6.14 有个灵巧的窍门: 如果我这样写int realarray[10]; int *array =
&realarray[-1]; 我就可以把“array” 当作下标从1 开始的数组。. . 32
6.15 当我向一个接受指针的指针的函数传入二维数组的时候, 编译器
报错了。. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 32
6.16 我怎样编写接受编译时宽度未知的二维数组的函数? . . . . . . . 32
6.17 我怎样在函数参数传递时混用静态和动态多维数组? . . . . . . . 33
6.18 当数组是函数的参数时, 为什么sizeof 不能正确报告数组的大小? 34
7 内存分配35
7.1 为什么这段代码不行?char *answer; printf("Type something:nn");
gets(answer); printf("You typed n"%sn"nn", answer); . . . . . . . 35
7.2 我的strcat() 不行.我试了char *s1 = "Hello, "; char *s2 = "world!";
char *s3 = strcat(s1, s2); 但是我得到了奇怪的结果。. . . . . . . 35
7.3 但是strcat 的手册页说它接受两个char * 型参数。我怎么知道
(空间) 分配的事情呢? . . . . . . . . . . . . . . . . . . . . . . . . 36
7.4 我刚才试了这样的代码char *p; strcpy(p, "abc"); 而它运行正
常?怎么回事?为什么它没有崩溃? . . . . . . . . . . . . . . . . 36
7.5 一个指针变量分配多少内存? . . . . . . . . . . . . . . . . . . . . 36
7.6 我有个函数, 本该返回一个字符串, 但当它返回调用者的时候, 返
回串却是垃圾信息。. . . . . . . . . . . . . . . . . . . . . . . . . 36
目录v
7.7 那么返回字符串或其它集合的争取方法是什么呢? . . . . . . . . 37
7.8 为什么在调用malloc() 时, 我得到“警告: 整数赋向指针需要类型
转换”? . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 37
7.9 为什么有些代码小心地把malloc 返回的值转换为分配的指针类型。37
7.10 在调用malloc() 的时候, 错误“不能把void * 转换为int *” 是什
么意思? . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 37
7.11 我见到了这样的代码char *p = malloc(strlen(s) + 1); strcpy(p,
s); 难道不应该是malloc((strlen(s) + 1) * sizeof(char))? . . . . . 37
7.12 我如何动态分配数组? . . . . . . . . . . . . . . . . . . . . . . . . 38
7.13 我听说有的操作系统程序使用的时候才真正分配malloc 申请的内
存。这合法吗? . . . . . . . . . . . . . . . . . . . . . . . . . . . . 38
7.14 我用一行这样的代码分配一个巨大的数组, 用于数字运算: double
*array = malloc(300 * 300 * sizeof( double )); malloc() 并没有返
回null, 但是程序运行得有些奇怪, 好像改写了某些内存, 或者
malloc() 并没有分配我申请的那么多内存, 云云。. . . . . . . . . 38
7.15 我的PC 有8 兆内存。为什么我只能分配640K 左右的内存? . . 38
7.16 我的程序总是崩溃, 显然在malloc 内部的某个地方。但是我看不
出哪里有问题。是malloc() 有bug 吗? . . . . . . . . . . . . . . . 38
7.17 动态分配的内存一旦释放之后你就不能再使用, 是吧? . . . . . . 38
7.18 为什么在调用free() 之后指针没有变空?使用(赋值, 比较) 释放
之后的指针有多么不安全? . . . . . . . . . . . . . . . . . . . . . 39
7.19 当我malloc() 为一个函数的局部指针分配内存时, 我还需要用
free() 明确的释放吗? . . . . . . . . . . . . . . . . . . . . . . . . 39
7.20 我在分配一些结构, 它们包含指向其它动态分配的对象的指针。
我在释放结构的时候, 还需要释放每一个下级指针吗? . . . . . . 39
7.21 我必须在程序退出之前释放分配的所有内存吗? . . . . . . . . . . 40
7.22 我有个程序分配了大量的内存, 然后又释放了。但是从操作系统
看, 内存的占用率却并没有回去。. . . . . . . . . . . . . . . . . . 40
7.23 free() 怎么知道有多少字节需要释放? . . . . . . . . . . . . . . . 40
7.24 那么我能否查询malloc 包, 可分配的最大块是多大? . . . . . . . 40
7.25 向realloc() 的第一个参数传入空指针合法吗?你为什么要这样做? 40
7.26 calloc() 和malloc() 有什么区别?利用calloc 的零填充功能安全
吗?free() 可以释放calloc() 分配的内存吗, 还是需要一个cfree()? 40
7.27 alloca() 是什么?为什么不提倡使用它? . . . . . . . . . . . . . . 41
8 字符和字符串43
8.1 为什么strcat(string, ’!’); 不行? . . . . . . . . . . . . . . . . . . 43
8.2 我在检查一个字符串是否跟某个值匹配。为什么这样不行?char
*string; . . . if(string == "value") f /* string matches ”value” */
. . . g . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 43
目录vi
8.3 如果我可以写char a[] = "Hello, world!"; 为什么我不能写char
a[14]; a = "Hello, world!"; . . . . . . . . . . . . . . . . . . . . . . 43
8.4 我怎么得到对应字符的数字(字符集) 值, 或者相反? . . . . . . . 44
8.5 我认为我的编译器有问题: 我注意到sizeof(’a’) 是2 而不是1 (即,
不是sizeof(char))。. . . . . . . . . . . . . . . . . . . . . . . . . . 44
9 布尔表达式和变量45
9.1 C 语言中布尔值的候选类型是什么?为什么它不是一个标准类
型?我应该用#define 或enum 定义true 和false 值吗? . . . . . 45
9.2 因为在C 语言中所有的非零值都被看作“真”, 是不是把TRUE 定
义为1 很危险?如果某个内置的函数或关系操作符“返回” 不是1
的其它值怎么办? . . . . . . . . . . . . . . . . . . . . . . . . . . 45
9.3 当p 是指针时, if(p) 是合法的表达式吗? . . . . . . . . . . . . . 46
10 C 预处理器47
10.1 这些机巧的预处理宏: #define begin f #define end g 你觉得怎么
样? . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 47
10.2 怎么写一个一般用途的宏交换两个值? . . . . . . . . . . . . . . . 47
10.3 书写多语句宏的最好方法是什么? . . . . . . . . . . . . . . . . . 47
10.4 我第一次把一个程序分成多个源文件, 我不知道该把什么放到.c
文件, 把什么放到.h 文件。(“.h” 到底是什么意思?) . . . . . . . 48
10.5 一个头文件可以包含另一头文件吗? . . . . . . . . . . . . . . . . 48
10.6 #include <> 和#include "" 有什么区别? . . . . . . . . . . . . 48
10.7 完整的头文件搜索规则是怎样的? . . . . . . . . . . . . . . . . . 49
10.8 我在文件的第一个声明就遇到奇怪的语法错误, 但是看上去没什
么问题。. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 49
10.9 我包含了我使用的库函数的正确头文件, 可是连接器还是说它没
有定义。. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 49
10.10 我在编译一个程序, 看起来我好像缺少需要的一个或多个头文
件。谁能发给我一份? . . . . . . . . . . . . . . . . . . . . . . . . 49
10.11 我怎样构造比较字符串的#if 预处理表达式? . . . . . . . . . . . 49
10.12 sizeof 操作符可以用于#if 预编译指令中吗? . . . . . . . . . . . . 50
10.13 我可以在#include 行里使用#ifdef 来定义两个不同的东西吗? . 50
10.14 对typdef 的类型定义有没有类似#ifdef的东西? . . . . . . . . . 50
10.15 我如何用#if 表达式来判断机器是高字节在前还是低字节在前? . 50
10.16 我得到了一些代码, 里边有太多的#ifdef。我不想使用预处理器
把所有的#include 和#ifdef 都扩展开, 有什么办法只保留一种条
件的代码呢? . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 50
10.17 如何列出所有的预定义标识符? . . . . . . . . . . . . . . . . . . . 50
目录vii
10.18 我有些旧代码, 试图用这样的宏来构造标识符#define Paste(a, b)
a/**/b 但是现在不行了。. . . . . . . . . . . . . . . . . . . . . . 51
10.19 为什么宏#define TRACE(n) printf("TRACE: %dnn", n) 报出警
告“用字符串常量代替宏”?它似乎应该把TRACE(count); 扩展
为printf("TRACE: %dncount", count); . . . . . . . . . . . . . . 51
10.20 使用# 操作符时, 我在字符串常量内使用宏参数有问题。. . . . . 51
10.21 我想用预处理做某件事情, 但却不知道如何下手。. . . . . . . . . 51
10.22 怎样写参数个数可变的宏? . . . . . . . . . . . . . . . . . . . . . 51
11 ANSI/ISO 标准C 53
11.1 什么是“ANSI C 标准”? . . . . . . . . . . . . . . . . . . . . . . . 53
11.2 我如何得到一份标准的副本? . . . . . . . . . . . . . . . . . . . . 53
11.3 我在哪里可以找到标准的更新? . . . . . . . . . . . . . . . . . . . 54
11.4 很多ANSI 编译器在遇到以下代码时都会警告类型不匹配。
extern int func(float); int func(x) float x; f . . . . . . . . . . . . . 54
11.5 能否混用旧式的和新型的函数语法? . . . . . . . . . . . . . . . . 55
11.6 为什么声明extern int f(struct x *p); 报出了一个奇怪的警告信
息“结构x 在参数列表中声明”? . . . . . . . . . . . . . . . . . . 55
11.7 我不明白为什么我不能象这样在初始化和数组维度中使用常量:
const int n = 5; int a[n]; . . . . . . . . . . . . . . . . . . . . . . . 55
11.8 既然不能修改字符串常量, 为什么不把它们定义为字符常量的数
组? . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 55
11.9 “const char *p” 和“char * const p” 有何区别? . . . . . . . . . . 56
11.10 为什么我不能向接受const char ** 的函数传入char **? . . . . . 56
11.11 怎样正确声明main()? . . . . . . . . . . . . . . . . . . . . . . . . 56
11.12 我能否把main() 定义为void, 以避免扰人的“main无返回值” 警
告? . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 56
11.13 可main() 的第三个参数envp 是怎么回事? . . . . . . . . . . . . 57
11.14 我觉得把main() 声明为void 不会失败, 因为我调用了exit() 而不
是return , 况且我的操作系统也忽略了程序的退出/返回状态。. . 57
11.15 那么到底会出什么问题?真的有什么系统不支持void main() 吗? 57
11.16 我一直用的那本书《熟练傻瓜C语言》总是使用void main()。. . 57
11.17 从main() 中, exit(status) 和返回同样的status 真的等价吗? . . . 57
11.18 我试图用ANSI “字符串化” 预处理操作符# 向信息中插入符号
常量的值, 但它字符串化的总是宏的名字而不是它的值。. . . . . 58
11.19 警告信息“warning: macro replacement within a string literal” 是
什么意思? . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 58
11.20 在我用#ifdef 去掉的代码里出现了奇怪的语法错误。. . . . . . . 58
11.21 #pragma 是什么, 有什么用? . . . . . . . . . . . . . . . . . . . . 59
11.22 “#pragma once” 是什么意思?我在一些头文件中看到了它。. . 59
11.23 a[3] = "abc"; 合法吗?它是什么意思? . . . . . . . . . . . . . . . 59
11.24 为什么我不能对void* 指针进行运算? . . . . . . . . . . . . . . . 59
11.25 memcpy() 和memmove() 有什么区别? . . . . . . . . . . . . . . 59
11.26 malloc(0) 有什么用?返回一个控指针还是指向0 字节的指针? . 59
11.27 为什么ANSI 标准规定了外部标示符的长度和大小写限制? . . . 60
11.28 我的编译对最简单的测试程序报出了一大堆的语法错误。. . . . . 60
11.29 为什么有些ASNI/ISO 标准库函数未定义?我明明使用的就是
ANSI 编译器。. . . . . . . . . . . . . . . . . . . . . . . . . . . . 60
11.30 谁有把旧的C 程序转化为ANSI C 或相反的工具, 或者自动生成
原型的工具? . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 60
11.31 为什么声称兼容ANSI 的Frobozz Magic C 编译器不能编译这些
代码?我知道这些代码是ANSI 的, 因为gcc 可以编译。. . . . . 60
11.32 人们好像有些在意实现定义(implementation-defin-ed)、未明确
(unspecified) 和无定义(undefined) 行为的区别。它们的区别到底
在哪里? . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 61
11.33 一个程序的“合法”, “有效” 或“符合” 到底是什么意思? . . . . . 61
11.34 我很吃惊, ANSI 标准竟然有那么多没有定义的东西。标准的唯一
任务不就是让这些东西标准化吗? . . . . . . . . . . . . . . . . . 61
11.35 有人说i = i++ 的行为是未定义的, 但是我刚在一个兼容ANSI 的
编译器上测试, 得到了我希望的结果。. . . . . . . . . . . . . . . 62
12 标准输入输出库63
12.1 这样的代码有什么问题?char c; while((c = getchar()) != EOF) ... 63
12.2 我有个读取直到EOF 的简单程序, 但是我如何才能在键盘上输入
那个“EOF” 呢? . . . . . . . . . . . . . . . . . . . . . . . . . . . 63
12.3 为什么这些代码while(!feof(infp)) f fgets(buf, MAXLINE, infp);
fputs(buf, outfp); g 把最后一行复制了两遍? . . . . . . . . . . . 63
12.4 我的程序的屏幕提示和中间输出有时显示在屏幕上, 尤其是当我
用管道向另一个程序输出的时候。. . . . . . . . . . . . . . . . . 63
12.5 我怎样不等待回车键一次输入一个字符? . . . . . . . . . . . . . . 64
12.6 我如何在printf 的格式串中输出一个’%’?我试过n%, 但是不行。64
12.7 有人告诉我在printf 中使用%lf 不正确。那么, 如果scanf() 需要
%lf, 怎么可以用在printf() 中用%f 输出双精度数呢? . . . . . . . 64
12.8 对于size t 那样的类型定义, 当我不知道它到底是long 还是其它
类型的时候, 我应该使用什么样的printf 格式呢? . . . . . . . . . 64
12.9 我如何用printf 实现可变的域宽度?就是说, 我想在运行时确定
宽度而不是使用%8d? . . . . . . . . . . . . . . . . . . . . . . . . 64
12.10 如何输出在千位上用逗号隔开的数字?金额数字呢? . . . . . . . 65
12.11 为什么scanf("%d", i) 调用不行? . . . . . . . . . . . . . . . . . . 65
12.12 为什么char s[30]; scanf("%s", s); 不用& 也可以? . . . . . . . . 65
目录ix
12.13 为什么这些代码double d; scanf("%f", &d); 不行? . . . . . . . . 65
12.14 怎样在scanf() 格式串中指定可变的宽度? . . . . . . . . . . . . . 65
12.15 当我用“%dnn” 调用scanf 从键盘读取数字的时候, 好像要多输入
一行函数才返回。. . . . . . . . . . . . . . . . . . . . . . . . . . 65
12.16 我用scanf %d 读取一个数字, 然后再用gets() 读取字符串, 但是
编译器好像跳过了gets() 调用! . . . . . . . . . . . . . . . . . . . 66
12.17 我发现如果坚持检查返回值以确保用户输入的是我期待的数值,
则scanf() 的使用会安全很多, 但有的时候好像会陷入无限循环。. 66
12.18 为什么大家都说不要使用scanf()?那我该用什么来代替呢? . . . 66
12.19 我怎样才知道对于任意的sprintf 调用需要多大的目标缓冲区?怎
样才能避免sprintf() 目标缓冲区溢出? . . . . . . . . . . . . . . . 66
12.20 为什么大家都说不要使用gets()? . . . . . . . . . . . . . . . . . . 67
12.21 为什么调用printf() 之后errno 内有ENOTTY? . . . . . . . . . . 67
12.22 fgetops/fsetops 和ftell/fseek 之间有什么区别? fgetops() 和fsetops()
到底有什么用处? . . . . . . . . . . . . . . . . . . . . . . . 68
12.23 如何清除多余的输入, 以防止在下一个提示符下读入?fflush(stdin)
可以吗? . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 68
12.24 既然fflush() 不能, 那么怎样才能清除输入呢? . . . . . . . . . . . 68
12.25 对某些路径文件名调用fopen() 总是失败。. . . . . . . . . . . . . 68
12.26 我想用“r+” 打开一个文件, 读出一个字符串, 修改之后再写入, 从
而就地更新一个文件。可是这样不行。. . . . . . . . . . . . . . . 69
12.27 怎样在程序里把stdin 或stdout 重定向到文件? . . . . . . . . . . 69
12.28 一旦使用freopen() 之后, 怎样才能恢复原来的stdout (或stdin)? 69
12.29 怎样同时向两个地方输出, 如同时输出到屏幕和文件? . . . . . . 69
12.30 怎样正确的读取二进制文件?我有时看到0x0a 和0x0d 混淆了,
而且如果数据中包含0x1a 的话, 我好像会提前遇到EOF。. . . . 70
13 库函数71
13.1 怎样把数字转为字符串(与atoi 相反)?有itoa() 函数吗? . . . . 71
13.2 为什么strncpy() 不能总在目标串放上终止符’n0’? . . . . . . . 71
13.3 为什么有些版本的toupper() 对大写字符会有奇怪的反应?为什
么有的代码在调用toupper() 前先调用tolower()? . . . . . . . . . 71
13.4 怎样把字符串分隔成用空白作间隔符的段?怎样实现类似传递给
main() 的argc 和argv? . . . . . . . . . . . . . . . . . . . . . . . 72
13.5 我需要一些处理正则表达式或通配符匹配的代码。. . . . . . . . 72
13.6 我想用strcmp() 作为比较函数, 调用qsort() 对一个字符串数组排
序, 但是不行。. . . . . . . . . . . . . . . . . . . . . . . . . . . . 72
13.7 我想用qsort() 对一个结构数组排序。我的比较函数接受结构指
针, 但是编译器认为这个函数对于qsort() 是错误类型。我要怎样
转换这个函数指针才能避免这样的警告? . . . . . . . . . . . . . . 73
13.8 怎样对一个链表排序? . . . . . . . . . . . . . . . . . . . . . . . . 73
13.9 怎样对多于内存的数据排序? . . . . . . . . . . . . . . . . . . . . 73
13.10 怎样在C 程序中取得当前日期或时间? . . . . . . . . . . . . . . . 73
13.11 我知道库函数localtime() 可以把time t 转换成结构struct tm, 而
ctime() 可以把time t 转换成为可打印的字符串。怎样才能进行
反向操作, 把struct tm 或一个字符串转换成time t? . . . . . . . 74
13.12 怎样在日期上加N 天?怎样取得两个日期的时间间隔? . . . . . . 74
13.13 我需要一个随机数生成器。. . . . . . . . . . . . . . . . . . . . . 75
13.14 怎样获得在一定范围内的随机数? . . . . . . . . . . . . . . . . . 75
13.15 每次执行程序, rand() 都返回相同顺序的数字。. . . . . . . . . . 75
13.16 我需要随机的真/假值, 所以我用直接用rand() % 2, 可是我得到
交替的0, 1, 0, 1, 0 ⋯⋯ . . . . . . . . . . . . . . . . . . . . . . . 76
13.17 怎样产生标准分布或高斯分布的随机数? . . . . . . . . . . . . . . 76
13.18 我不断得到库函数未定义错误, 但是我已经#inlude 了所有用到
的头文件了。. . . . . . . . . . . . . . . . . . . . . . . . . . . . . 77
13.19 虽然我在连接时明确地指定了正确的函数库, 我还是得到库函数
未定义错误。. . . . . . . . . . . . . . . . . . . . . . . . . . . . . 77
13.20 连接器说end 未定义代表什么意思? . . . . . . . . . . . . . . . . 77
13.21 我的编译器提示printf 未定义!这怎么可能? . . . . . . . . . . . 77
14 浮点运算79
14.1 一个float 变量赋值为3.1 时, 为什么printf 输出的值为3.0999999? 79
14.2 执行一些开方根运算, 可是得到一些疯狂的数字。. . . . . . . . . 79
14.3 做一些简单的三角函数运算, 也引用了#include