文件名称:C语言FAQ 常见问题列表
文件大小:631KB
文件格式:RAR
更新时间:2013-11-17 10:28:09
C语言 常见问题 FAQ
previous up contents next
C 语言常见问题集
原著:Steve Summit
翻译:朱群英, 孙 云
修订版 0.9.4, 2005年6月23日
版权所有 © 2005
* 目录
* 1. 前言
* 2. 声明和初始化
o 2.1 我如何决定使用那种整数类型?
o 2.2 64 位机上的 64 位类型是什么样的?
o 2.3 怎样定义和声明全局变量和函数最好?
o 2.4 extern 在函数声明中是什么意思?
o 2.5 关键字 auto 到底有什么用途?
o 2.6 我似乎不能成功定义一个链表。我试过 typedef struct { char *item; NODEPTR next; } *NODEPTR; 但是编译器报了错误信息。难道在C语言中一个结构不能包含指向自己的指针吗?
o 2.7 怎样建立和理解非常复杂的声明?例如定义一个包含 N 个指向返回指向字符的指针的函数的指针的数组?
o 2.8 函数只定义了一次, 调用了一次, 但编译器提示非法重定义了。
o 2.9 main() 的正确定义是什么? void main() 正确吗?
o 2.10 对于没有初始化的变量的初始值可以作怎样的假定?如果一个全局变量初始值为 ``零", 它可否作为空指针或浮点零?
o 2.11 代码 int f() { char a[] = "Hello, world!";} 不能编译。
o 2.12 这样的初始化有什么问题?char *p = malloc(10); 编译器提示 ``非法初始式" 云云。
o 2.13 以下的初始化有什么区别?char a[] = "string literal"; char *p = "string literal"; 当我向 p[i] 赋值的时候, 我的程序崩溃了。
o 2.14 我总算弄清除函数指针的声明方法了, 但怎样才能初始化呢?
* 3. 结构、联合和枚举
o 3.1 声明 struct x1 { ...}; 和 typedef struct { ...} x2; 有什么不同?
o 3.2 为什么 struct x { ...}; x thestruct; 不对?
o 3.3 一个结构可以包含指向自己的指针吗?
o 3.4 在 C 语言中实现抽象数据类型什么方法最好?
o 3.5 在 C 中是否有模拟继承等面向对象程序设计特性的好方法?
o 3.6 我遇到这样声明结构的代码: struct name { int namelen; char namestr[1];}; 然后又使用一些内存分配技巧使 namestr 数组用起来好像有多个元素。这样合法和可移植吗?
o 3.7 是否有自动比较结构的方法?
o 3.8 如何向接受结构参数的函数传入常数值?
o 3.9 怎样从/向数据文件读/写结构?
o 3.10 我的编译器在结构中留下了空洞, 这导致空间浪费而且无法与外部数据文件进行 "二进制" 读写。能否关掉填充, 或者控制结构域的对齐方式?
o 3.11 为什么 sizeof 返回的值大于结构的期望值, 是不是尾部有填充?
o 3.12 如何确定域在结构中的字节偏移?
o 3.13 怎样在运行时用名字访问结构中的域?
o 3.14 程序运行正确, 但退出时却 ``core dump''了,怎么回事?
o 3.15 可以初始化一个联合吗?
o 3.16 枚举和一组预处理的 #define 有什么不同?
o 3.17 有什么容易的显示枚举值符号的方法?
* 4. 表达式
o 4.1 为什么这样的代码: a[i] = i++; 不能工作?
o 4.2 使用我的编译器,下面的代码 int i=7; printf("%d\n", i++ * i++); 返回 49?不管按什么顺序计算, 难道不该打印出56吗?
o 4.3 对于代码 int i = 3; i = i++; 不同编译器给出不同的结果, 有的为 3, 有的为 4, 哪个是正确的?
o 4.4 这是个巧妙的表达式: a ^= b ^= a ^= b 它不需要临时变量就可以交换 a 和 b 的值。
o 4.5 我可否用括号来强制执行我所需要的计算顺序?
o 4.6 可是 && 和 || 运算符呢?我看到过类似 while((c = getchar()) != EOF && c != '\n') 的代码 ……
o 4.7 我怎样才能理解复杂表达式?``序列点" 是什么?
o 4.8 那么, 对于 a[i] = i++; 我们不知道 a[] 的哪一个分量会被改写,但 i 的确会增加 1, 对吗?
o 4.9 ++i 和 i++ 有什么区别?
o 4.10 如果我不使用表达式的值, 我应该用 ++i 或 i++ 来自增一个变量吗?
o 4.11 为什么如下的代码 int a = 100, b = 100; long int c = a * b; 不能工作?
o 4.12 我需要根据条件把一个复杂的表达式赋值给两个变量中的一个。可以用下边这样的代码吗? ((condition) ? a : b) = complicated_expression;
* 5. 指针
o 5.1 我想声明一个指针并为它分配一些空间, 但却不行。这些代码有什么问题? char *p; *p = malloc(10);
o 5.2 *p++ 自增 p 还是 p 所指向的变量?
o 5.3 我有一个 char * 型指针正巧指向一些 int 型变量, 我想跳过它们。为什么如下的代码 ((int *)p)++; 不行?
o 5.4 我有个函数,它应该接受并初始化一个指针 void f(int *ip) { static int dummy = 5; ip = &dummy;} 但是当我如下调用时: int *ip; f(ip); 调用者的指针却没有任何变化。
o 5.5 我能否用 void** 指针作为参数, 使函数按引用接受一般指针?
o 5.6 我有一个函数 extern int f(int *); 它接受指向 int 型的指针。我怎样用引用方式传入一个常数?下面这样的调用 f(&5); 似乎不行。
o 5.7 C 有 ``按引用传递" 吗?
o 5.8 我看到了用指针调用函数的不同语法形式。到底怎么回事?
o 5.9 我怎样把一个 int 变量转换为 char * 型?我试了类型转换, 但是不行。
* 6. 空 (null) 指针
o 6.1 臭名昭著的空指针到底是什么?
o 6.2 怎样在程序里获得一个空指针?
o 6.3 用缩写的指针比较 ``if(p)" 检查空指针是否可靠?如果空指针的内部表达不是 0 会怎么样?
o 6.4 NULL 是什么, 它是怎么定义的?
o 6.5 在使用非全零作为空指针内部表达的机器上, NULL 是如何定义的?
o 6.6 如果 NULL 定义成 #define NULL ((char *)0) 难道不就可以向函数传入不加转换的 NULL 了吗?
o 6.7 如果 NULL 和 0 作为空指针常数是等价的, 那我到底该用哪一个呢?
o 6.8 但是如果 NULL 的值改变了, 比如在使用非零内部空指针的机器上, 难道用 NULL (而不是 0) 不是更好吗?
o 6.9 用预定义宏 #define Nullptr(type) (type *)0 帮助创建正确类型的空指针。
o 6.10 这有点奇怪。NULL 可以确保是 0, 但空 (null) 指针却不一定?
o 6.11 为什么有那么多关于空指针的疑惑?为什么这些问题如此经常地出现?
o 6.12 我很困惑。我就是不能理解这些空指针一类的东西。
o 6.13 考虑到有关空指针的所有这些困惑, 难道把要求它们内部表达都必须为 0 不是更简单吗?
o 6.14 说真的, 真有机器用非零空指针吗, 或者不同类型用不同的表达?
o 6.15 运行时的 ``空指针赋值" 错误是什么意思?
* 7. 数组和指针
o 7.1 我在一个源文件中定义了 char a[6], 在另一个中声明了 extern char *a 。为什么不行 ?
o 7.2 可是我听说 char a[ ] 和 char *a 是一样的。
o 7.3 那么, 在 C 语言中 ``指针和数组等价" 到底是什么意思 ?
o 7.4 那么为什么作为函数形参的数组和指针申明可以互换呢 ?
o 7.5 如果你不能给它赋值, 那么数组如何能成为左值呢 ?
o 7.6 现实地讲, 数组和指针地区别是什么 ?
o 7.7 有人跟我讲, 数组不过是常指针。
o 7.8 我遇到一些 ``搞笑" 的代码, 包含 5["abcdef"] 这样的 ``表达式"。 这为什么是合法的 C 表达式呢 ?
o 7.9 既然数组引用会蜕化为指针, 如果 arr 是数组, 那么 arr 和 &arr 又有什么区别呢 ?
o 7.10 我如何声明一个数组指针 ?
o 7.11 我如何在运行期设定数组的大小 ? 我怎样才能避免固定大小的数组 ?
o 7.12 我如何声明大小和传入的数组一样的局部数组 ?
o 7.13 我该如何动态分配多维数组 ?
o 7.14 有个灵巧的窍门: 如果我这样写 int realarray[10]; int *array = &realarray[-1]; 我就可以把 ``array" 当作下标从 1 开始的数组。
o 7.15 当我向一个接受指针的指针的函数传入二维数组的时候, 编译器报错了。
o 7.16 我怎样编写接受编译时宽度未知的二维数组的函数 ?
o 7.17 我怎样在函数参数传递时混用静态和动态多维数组 ?
o 7.18 当数组是函数的参数时, 为什么 sizeof 不能正确报告数组的大小 ?
* 8. 内存分配
o 8.1 为什么这段代码不行?char *answer; printf("Type something:\n"); gets(answer); printf("You typed \"%s\"\n", answer);
o 8.2 我的 strcat() 不行.我试了 char *s1 = "Hello, "; char *s2 = "world!"; char *s3 = strcat(s1, s2); 但是我得到了奇怪的结果。
o 8.3 但是 strcat 的手册页说它接受两个 char * 型参数。我怎么知道 (空间) 分配的事情呢?
o 8.4 我刚才试了这样的代码 char *p; strcpy(p, "abc"); 而它运行正常?怎么回事?为什么它没有崩溃?
o 8.5 一个指针变量分配多少内存?
o 8.6 我有个函数, 本该返回一个字符串, 但当它返回调用者的时候, 返回串却是垃圾信息。
o 8.7 那么返回字符串或其它集合的争取方法是什么呢?
o 8.8 为什么在调用 malloc() 时, 我得到 ``警告: 整数赋向指针需要类型转换"?
o 8.9 为什么有些代码小心地把 malloc 返回的值转换为分配的指针类型。
o 8.10 在调用 malloc() 的时候, 错误 ``不能把 void * 转换为 int *" 是什么意思?
o 8.11 我见到了这样的代码 char *p = malloc(strlen(s) + 1); strcpy(p, s); 难道不应该是 malloc((strlen(s) + 1) * sizeof(char))?
o 8.12 我如何动态分配数组?
o 8.13 我听说有的操作系统程序使用的时候才真正分配 malloc 申请的内存。这合法吗?
o 8.14 我用一行这样的代码分配一个巨大的数组, 用于数字运算: double *array = malloc(300 * 300 * sizeof( double )); malloc() 并没有返回 null, 但是程序运行得有些奇怪, 好像改写了某些内存, 或者 malloc() 并没有分配我申请的那么多内存, 云云。
o 8.15 我的 PC 有 8 兆内存。为什么我只能分配 640K 左右的内存?
o 8.16 我的程序总是崩溃, 显然在 malloc 内部的某个地方。 但是我看不出哪里有问题。是 malloc() 有 bug 吗?
o 8.17 动态分配的内存一旦释放之后你就不能再使用, 是吧?
o 8.18 为什么在调用 free() 之后指针没有变空?使用 (赋值, 比较) 释放之后的指针有多么不安全?
o 8.19 当我 malloc() 为一个函数的局部指针分配内存时, 我还需要用 free() 明确的释放吗?
o 8.20 我在分配一些结构, 它们包含指向其它动态分配的对象的指针。我在释放结构的时候, 还需要释放每一个下级指针吗?
o 8.21 我必须在程序退出之前释放分配的所有内存吗?
o 8.22 我有个程序分配了大量的内存, 然后又释放了。但是从操作系统看, 内存的占用率却并没有回去。
o 8.23 free() 怎么知道有多少字节需要释放?
o 8.24 那么我能否查询 malloc 包, 可分配的最大块是多大?
o 8.25 向 realloc() 的第一个参数传入空指针合法吗?你为什么要这样做?
o 8.26 calloc() 和 malloc() 有什么区别?利用 calloc 的零填充功能安全吗? free() 可以释放 calloc() 分配的内存吗, 还是需要一个 cfree()?
o 8.27 alloca() 是什么?为什么不提倡使用它?
* 9. 字符和字符串
o 9.1 为什么 strcat(string, '!'); 不行?
o 9.2 我在检查一个字符串是否跟某个值匹配。 为什么这样不行? char *string; ... if(string == "value") { /* string matches "value" */ ... }
o 9.3 如果我可以写 char a[] = "Hello, world!"; 为什么我不能写 char a[14]; a = "Hello, world!";
o 9.4 我怎么得到对应字符的数字 (字符集) 值, 或者相反?
o 9.5 我认为我的编译器有问题: 我注意到 sizeof('a') 是 2 而不是 1 (即, 不是 sizeof(char))。
* 10. 布尔表达式和变量
o 10.1 C 语言中布尔值的候选类型是什么?为什么它不是一个标准类型?我应该用 #define 或 enum 定义 true 和 false 值吗?
o 10.2 因为在 C 语言中所有的非零值都被看作 ``真", 是不是把 TRUE 定义为 1 很危险?如果某个内置的函数或关系操作符 ``返回" 不是 1 的其它值怎么办?
o 10.3 当 p 是指针时, if(p) 是合法的表达式吗?
* 11. C 预处理器
o 11.1 这些机巧的预处理宏: #define begin { #define end } 你觉得怎么样?
o 11.2 怎么写一个一般用途的宏交换两个值?
o 11.3 书写多语句宏的最好方法是什么?
o 11.4 我第一次把一个程序分成多个源文件, 我不知道该把什么放到 .c 文件, 把什么放到 .h 文件。(``.h" 到底是什么意思?)
o 11.5 一个头文件可以包含另一头文件吗?
o 11.6 #include <> 和 #include 有什么区别?
o 11.7 完整的头文件搜索规则是怎样的?
o 11.8 我在文件的第一个声明就遇到奇怪的语法错误, 但是看上去没什么问题。
o 11.9 我包含了我使用的库函数的正确头文件, 可是连接器还是说它没有定义。
o 11.10 我在编译一个程序, 看起来我好像缺少需要的一个或多个头文件。谁能发给我一份?
o 11.11 我怎样构造比较字符串的 #if 预处理表达式?
o 11.12 sizeof 操作符可以用于 #if 预编译指令中吗?
o 11.13 我可以在 #include 行里使用 #ifdef 来定义两个不同的东西吗?
o 11.14 对 typdef 的类型定义有没有类似 #ifdef的东西?
o 11.15 我如何用 #if 表达式来判断机器是高字节在前还是低字节在前?
o 11.16 我得到了一些代码, 里边有太多的 #ifdef。 我不想使用预处理器把所有的 #include 和 #ifdef 都扩展开, 有什么办法只保留一种条件的代码呢?
o 11.17 如何列出所有的预定义标识符?
o 11.18 我有些旧代码, 试图用这样的宏来构造标识符 #define Paste(a, b) a/**/b 但是现在不行了。
o 11.19 为什么宏 #define TRACE(n) printf("TRACE: %d\n", n) 报出警告 ``用字符串常量代替宏"?它似乎应该把 TRACE(count); 扩展为 printf("TRACE: %d\count", count);
o 11.20 使用 # 操作符时, 我在字符串常量内使用宏参数有问题。
o 11.21 我想用预处理做某件事情, 但却不知道如何下手。
o 11.22 怎样写参数个数可变的宏?
* 12. ANSI/ISO 标准 C
o 12.1 什么是 ``ANSI C 标准"?
o 12.2 我如何得到一份标准的副本?
o 12.3 我在哪里可以找到标准的更新?
o 12.4 很多 ANSI 编译器在遇到以下代码时都会警告类型不匹配。 extern int func(float); int func(x) float x; { ...
o 12.5 能否混用旧式的和新型的函数语法?
o 12.6 为什么声明 extern int f(struct x *p); 报出了一个奇怪的警告信息 ``结构 x 在参数列表中声明"?
o 12.7 我不明白为什么我不能象这样在初始化和数组维度中使用常量: const int n = 5; int a[n];
o 12.8 既然不能修改字符串常量, 为什么不把它们定义为字符常量的数组?
o 12.9 ``const char *p" 和 ``char * const p" 有何区别?
o 12.10 为什么我不能向接受 const char ** 的函数传入 char **?
o 12.11 怎样正确声明 main()?
o 12.12 我能否把 main() 定义为 void, 以避免扰人的 ``main无返回值" 警告?
o 12.13 可 main() 的第三个参数 envp 是怎么回事?
o 12.14 我觉得把 main() 声明为 void 不会失败, 因为我调用了 exit() 而不是 return , 况且我的操作系统也忽略了程序的退出/返回状态。
o 12.15 那么到底会出什么问题?真的有什么系统不支持 void main() 吗?
o 12.16 我一直用的那本书《熟练傻瓜C语言》总是使用 void main()。
o 12.17 从 main() 中, exit(status) 和返回同样的 status 真的等价吗?
o 12.18 我试图用 ANSI ``字符串化" 预处理操作符 # 向信息中插入符号常量的值, 但它字符串化的总是宏的名字而不是它的值。
o 12.19 警告信息 ``warning: macro replacement within a string literal" 是什么意思?
o 12.20 在我用 #ifdef 去掉的代码里出现了奇怪的语法错误。
o 12.21 #pragma 是什么, 有什么用?
o 12.22 ``#pragma once" 是什么意思?我在一些头文件中看到了它。
o 12.23 a[3] = "abc"; 合法吗?它是什么意思?
o 12.24 为什么我不能对 void* 指针进行运算?
o 12.25 memcpy() 和 memmove() 有什么区别?
o 12.26 malloc(0) 有什么用?返回一个控指针还是指向 0 字节的指针?
o 12.27 为什么 ANSI 标准规定了外部标示符的长度和大小写限制?
o 12.28 我的编译对最简单的测试程序报出了一大堆的语法错误。
o 12.29 为什么有些 ASNI/ISO 标准库函数未定义?我明明使用的就是 ANSI 编译器。
o 12.30 谁有把旧的 C 程序转化为 ANSI C 或相反的工具, 或者自动生成原型的工具?
o 12.31 为什么声称兼容 ANSI 的 Frobozz Magic C 编译器不能编译这些代码?我知道这些代码是 ANSI 的, 因为 gcc 可以编译。
o 12.32 人们好像有些在意实现定义 (implementation-defin-ed)、未明确 (unspecified) 和无定义 (undefined) 行为的区别。它们的区别到底在哪里?
o 12.33 一个程序的 ``合法", ``有效" 或 ``符合" 到底是什么意思?
o 12.34 我很吃惊, ANSI 标准竟然有那么多没有定义的东西。标准的唯一任务不就是让这些东西标准化吗?
o 12.35 有人说 i = i++ 的行为是未定义的, 但是我刚在一个兼容 ANSI 的编译器上测试, 得到了我希望的结果。
* 13. 标准输入输出库
o 13.1 这样的代码有什么问题? char c; while((c = getchar()) != EOF) ...
o 13.2 我有个读取直到 EOF 的简单程序, 但是我如何才能在键盘上输入那个 ``EOF" 呢?
o 13.3 为什么这些代码 while(!feof(infp)) { fgets(buf, MAXLINE, infp); fputs(buf, outfp); } 把最后一行复制了两遍?
o 13.4 我的程序的屏幕提示和中间输出有时显示在屏幕上, 尤其是当我用管道向另一个程序输出的时候。
o 13.5 我怎样不等待回车键一次输入一个字符?
o 13.6 我如何在 printf 的格式串中输出一个 '%'?我试过 \%, 但是不行。
o 13.7 有人告诉我在 printf 中使用 %lf 不正确。那么, 如果 scanf() 需要 %lf, 怎么可以用在 printf() 中用 %f 输出双精度数呢?
o 13.8 对于 size_t 那样的类型定义, 当我不知道它到底是 long 还是其它类型的时候, 我应该使用什么样的 printf 格式呢?
o 13.9 我如何用 printf 实现可变的域宽度?就是说, 我想在运行时确定宽度而不是使用 %8d?
o 13.10 如何输出在千位上用逗号隔开的数字?金额数字呢?
o 13.11 为什么 scanf("%d", i) 调用不行?
o 13.12 为什么 char s[30]; scanf("%s", s); 不用 & 也可以?
o 13.13 为什么这些代码 double d; scanf("%f", &d); 不行?
o 13.14 怎样在 scanf() 格式串中指定可变的宽度?
o 13.15 当我用 ``%d\n" 调用 scanf 从键盘读取数字的时候, 好像要多输入一行函数才返回。
o 13.16 我用 scanf %d 读取一个数字, 然后再用 gets() 读取字符串, 但是编译器好像跳过了 gets() 调用!
o 13.17 我发现如果坚持检查返回值以确保用户输入的是我期待的数值, 则 scanf() 的使用会安全很多, 但有的时候好像会陷入无限循环。
o 13.18 为什么大家都说不要使用 scanf()?那我该用什么来代替呢?
o 13.19 我怎样才知道对于任意的 sprintf 调用需要多大的目标缓冲区?怎样才能避免 sprintf() 目标缓冲区溢出?
o 13.20 为什么大家都说不要使用 gets()?
o 13.21 为什么调用 printf() 之后 errno 内有 ENOTTY?
o 13.22 fgetops/fsetops 和 ftell/fseek 之间有什么区别? fgetops() 和 fsetops() 到底有什么用处?
o 13.23 如何清除多余的输入, 以防止在下一个提示符下读入? fflush(stdin) 可以吗?
o 13.24 既然 fflush() 不能, 那么怎样才能清除输入呢?
o 13.25 对某些路径文件名调用 fopen() 总是失败。
o 13.26 我想用 ``r+" 打开一个文件, 读出一个字符串, 修改之后再写入, 从而就地更新一个文件。可是这样不行。
o 13.27 怎样在程序里把 stdin 或 stdout 重定向到文件?
o 13.28 一旦使用 freopen() 之后, 怎样才能恢复原来的 stdout (或 stdin)?
o 13.29 怎样同时向两个地方输出, 如同时输出到屏幕和文件?
o 13.30 怎样正确的读取二进制文件?我有时看到 0x0a 和 0x0d 混淆了, 而且如果数据中包含 0x1a 的话, 我好像会提前遇到 EOF。
* 14. 库函数
o 14.1 怎样把数字转为字符串 (与 atoi 相反)?有 itoa() 函数吗?
o 14.2 为什么 strncpy() 不能总在目标串放上终止符 '\0'?
o 14.3 为什么有些版本的 toupper() 对大写字符会有奇怪的反应?为什么有的代码在调用 toupper() 前先调用 tolower()?
o 14.4 怎样把字符串分隔成用空白作间隔符的段?怎样实现类似传递给 main() 的 argc 和 argv?
o 14.5 我需要一些处理正则表达式或通配符匹配的代码。
o 14.6 我想用 strcmp() 作为比较函数, 调用 qsort() 对一个字符串数组排序, 但是不行。
o 14.7 我想用 qsort() 对一个结构数组排序。我的比较函数接受结构指针, 但是编译器认为这个函数对于 qsort() 是错误类型。我要怎样转换这个函数指针才能避免这样的警告?
o 14.8 怎样对一个链表排序?
o 14.9 怎样对多于内存的数据排序?
o 14.10 怎样在 C 程序中取得当前日期或时间?
o 14.11 我知道库函数 localtime() 可以把 time_t 转换成结构 struct tm, 而 ctime() 可以把 time_t 转换成为可打印的字符串。怎样才能进行反向操作, 把 struct tm 或一个字符串转换成 time_t?
o 14.12 怎样在日期上加 N 天?怎样取得两个日期的时间间隔?
o 14.13 我需要一个随机数生成器。
o 14.14 怎样获得在一定范围内的随机数?
o 14.15 每次执行程序, rand() 都返回相同顺序的数字。
o 14.16 我需要随机的真/假值, 所以我用直接用 rand() % 2, 可是我得到交替的 0, 1, 0, 1, 0 ……
o 14.17 怎样产生标准分布或高斯分布的随机数?
o 14.18 我不断得到库函数未定义错误, 但是我已经 #inlude 了所有用到的头文件了。
o 14.19 虽然我在连接时明确地指定了正确的函数库, 我还是得到库函数未定义错误。
o 14.20 连接器说 _end 未定义代表什么意思?
o 14.21 我的编译器提示 printf 未定义!这怎么可能?
* 15. 浮点运算
o 15.1 一个 float 变量赋值为 3.1 时, 为什么 printf 输出的值为 3.0999999?
o 15.2 执行一些开方根运算, 可是得到一些疯狂的数字。
o 15.3 做一些简单的三角函数运算, 也引用了 #include