C/C++中如何不使用额外的中间变量交换两个变量的值

转载至鲨鱼编程,看完以后开阔了思维:

在编程过程中,我们经常会遇到需要交换两个变量的值的情况。传统的做法是使用一个额外的中间变量作为桥梁,但是有些场合下,我们可能无法或不希望使用额外的变量。本文将讨论在C/C++编程中如何在不使用额外的中间变量的情况下交换两个变量的值。

 
C/C++中如何不使用额外的中间变量交换两个变量的值
1. 问题定义
我们有两个变量a和b,我们希望交换它们的值,即原来a的值赋给b,b的值赋给a。通常的做法是使用一个临时变量temp,如下:

int a = 5;  int b = 10;  int temp;

  temp = a;  a = b;  b = temp;

在上述代码中,我们使用了一个临时变量temp来存储a的值,然后把b的值赋给a,最后再把temp的值(即原来a的值)赋给b。但是,如果我们不能或不希望使用额外的变量,我们该怎么做呢?
 
2. 位操作交换变量值
 
在C/C++中,我们可以使用位操作(bitwise operation)来交换两个变量的值。特别地,我们可以使用异或(XOR)操作:一个数与任何数的异或两次,结果仍然是原来的数。
 
int a = 5;int b = 10;a = a ^ b;b = a ^ b;
 
这里的a其实是a^b,所以b = a^b就变成了b = b^a^b,由于任何数与自身的异或都是0,所以b = 0^a,即b = a,所以b现在的值就是a原来的值a = a ^ b; // 同理,这里的b其实是a原来的值,所以a = a^b就变成了a = a^a^b,即a = 0^b,所以a = b,所以a现在的值就是b原来的值
在这段代码中,我们通过三次异或操作,成功地交换了a和b的值,而没有使用额外的变量。
 
3. 加减法交换变量值
除了位操作,我们还可以使用加减法来交换两个整数变量的值。请看下面的代码:
int a = 5;int b = 10;a = a + b;b = a - b; 
这里的a其实是a+b,所以b = a-b就变成了b = a+b-b,即b = a,所以b现在的值就是a原来的值a = a - b; 同理,这里的b其实是a原来的值,所以a = a-b就变成了a = a+b-a,即a = b,所以a现在的值就是b原来的值
在这段代码中,我们通过两次加法和两次减法,成功地交换了a和b的值,而没有使用额外的变量。
 
4. 注意事项
虽然以上两种方法都可以成功交换两个变量的值,但它们都有一些值得注意的地方。
 
位操作方法只适用于整数,对于浮点数或者其他非整数类型的值,这个方法可能无法正确工作。
 
加减法方法在数值非常大的情况下可能会导致溢出,从而无法正确地交换两个变量的值。
 
因此,在使用这些方法时,我们需要根据具体的情况来选择最合适的方法。
 
5. 结论
在许多编程情况下,我们需要交换两个变量的值。虽然使用临时变量是最常见的方法,但在某些情况下,我们可能需要或希望不使用临时变量。在C/C++中,我们可以使用位操作或加减法来实现这一目标。然而,这些方法并非在所有情况下都有效,我们需要根据具体的情况和需求来选择最合适的方法。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
老资源。 目录 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 , 可是 一直得到编译错误“undefined: sin” (函数sin 未定义)。. . . . . . 79 14.4 浮点计算程序表现奇怪, 在不同的机器上给出不同的结果。. . . . 79 14.5 有什么好的方法来验对浮点数在“足够接近” 情况下的等? . . . 80 14.6 怎样取整数? . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 80 14.7 为什么C 不提供乘幂的运算符? . . . . . . . . . . . . . . . . . . 80 14.8 为什么我机器上的 没有预定义常数M PI? . . . . . . 80 14.9 怎样测试IEEE NaN 以及其它特殊? . . . . . . . . . . . . . . . 81 14.10 在C 如何很好的实现复数? . . . . . . . . . . . . . . . . . . . . 81 14.11 我要寻找一些实现以下功能的程序源代码:快速傅立叶变换 (FFT)、矩阵算术(乘法、倒置等函数)、复数算术。. . . . . . . 81 14.12 Turbo C 的程序崩溃, 显示错误为“floating point formats not linked” (浮点格式未连接)。. . . . . . . . . . . . . . . . . . . . . 81 15 可变参数83 15.1 为什么调用printf() 前, 必须要用#include ? . . . . . 83 15.2 为什么%f 可以在printf() 参数, 同时表示float 和double?他们 难道不是不同类型吗? . . . . . . . . . . . . . . . . . . . . . . . . 83 15.3 为什么当n 为long int, printf("%d", n); 编译时没有匹配警告? 我以为ANSI 函数原型可以防止这样的类型不匹配。. . . . . . . 83 15.4 怎样写一个有可变参数的函数? . . . . . . . . . . . . . . . . . . . 83 15.5 怎样写类似printf() 的函数, 再把参数转传给printf() 去完成大部 分工作? . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 85 15.6 怎样写类似scanf() 的函数, 再把参数转传给scanf() 去完成大部 分工作? . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 85 15.7 怎样知道实际上有多少个参数传入函数? . . . . . . . . . . . . . . 85 15.8 为什么编译器不让我定义一个没有固定参数项的可变参数函数? . 86 15.9 我有个接受float 的可变参函数, 为什么va arg(argp, float) 不工作? 86 15.10 va arg() 不能得到类型为函数指针的参数。. . . . . . . . . . . . . 86 15.11 怎样实现一个可变参数函数, 它把参数再传给另一个可变参数函 数? . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 86 15.12 怎样调用一个参数在执行是才建立的函数? . . . . . . . . . . . . 87 16 奇怪的问题89 16.1 遇到不可理解的不合理语法错误, 似乎大段的程序没有编译。. . 89 16.2 为什么过程调用不工作?编译器似乎直接跳过去了。. . . . . . . 89 16.3 程序在执行用之前就崩溃了, 用调试器单步跟进, 在main() 之前 就死了。. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 89 16.4 程序执行正确, 但退出时崩溃在main() 最后一个语句之后。为什 么会这样? . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 89 16.5 程序在一台机器上执行完美, 但在另一台上却得到怪异的结果。 更奇怪的是, 增加或去除调试的打印语句, 就改变了症状⋯⋯ . . . 90 16.6 为什么代码: char *p = "hello, worl!"; p[0] = ’H’; 会崩溃? . . . 90 16.7 “Segmentation violation”, “Bus error” 和“General protection fault” 意味着什么? . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 91 17 风格93 17.1 什么是C 最好的代码布局风格? . . . . . . . . . . . . . . . . . . 93 17.2 用if(!strcmp(s1, s2)) 比较两个字符串等,是否是个好风格? . . . 93 17.3 为什么有的人用if (0 == x) 而不是if (x == 0)? . . . . . . . . . 93 17.4 原型说明extern int func ((int, int)); , 那些多出来的括号和下 划线代表了什么? . . . . . . . . . . . . . . . . . . . . . . . . . . 94 17.5 为什么有些代码在每次调用printf() 前, 加了类型转换(void)? . . 94 17.6 什么是“匈牙利标志法” (Hungarian Notation)?是否得用? . . 94 17.7 哪里可以找到“印第安山风格指南” (Indian Hill Style Guide) 及 其它编码标准? . . . . . . . . . . . . . . . . . . . . . . . . . . . . 94 17.8 有些人说goto 是邪恶的, 我应该永不用它。那是否太极端了? . . 95 18 工具和资源97 18.1 常用工具列表。. . . . . . . . . . . . . . . . . . . . . . . . . . . . 97 18.2 怎样抓捕棘手的malloc 问题? . . . . . . . . . . . . . . . . . . . . 98 18.3 有什么免费或便宜的编译器可以使用? . . . . . . . . . . . . . . . 98 18.4 刚刚输入完一个程序, 但它表现的很奇怪。你可以发现有什么错 误的地方吗? . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 98 18.5 哪里可以找到兼容ANSI 的lint? . . . . . . . . . . . . . . . . . . 99 18.6 难道ANSI 函数原型说明没有使lint 过时吗? . . . . . . . . . . . 99 18.7 网上有哪些C 的教程或其它资源? . . . . . . . . . . . . . . . . . 99 18.8 哪里可以找到好的源代码实例, 以供研究和学习? . . . . . . . . . 100 18.9 有什么好的学习C 的书?有哪些高级的书和参考? . . . . . . . . 100 18.10 哪里可以找到标准C 函数库的源代码? . . . . . . . . . . . . . . . 101 18.11 是否有一个在线的C 参考指南? . . . . . . . . . . . . . . . . . . 101 18.12 哪里可以得到ANSI/ISO C 标准? . . . . . . . . . . . . . . . . . 101 18.13 我需要分析和评估表达式的代码。. . . . . . . . . . . . . . . . . 101 18.14 哪里可以找到C 的BNF 或YACC 语法? . . . . . . . . . . . . . 101 18.15 谁有C 编译器的测试套件? . . . . . . . . . . . . . . . . . . . . . 102 18.16 哪里有一些有用的源代码片段和例子的收集? . . . . . . . . . . . 102 18.17 我需要执行多精度算术的代码。. . . . . . . . . . . . . . . . . . . 102 18.18 在哪里和怎样取得这些可自由发布的程序? . . . . . . . . . . . . 102 19 系统依赖105 19.1 怎样从键盘直接读入字符而不用等RETURN 键?怎样防止字符 输入时的回显? . . . . . . . . . . . . . . . . . . . . . . . . . . . . 105 19.2 怎样知道有未读的字符, 如果有, 有多少?如果没有字符, 怎样使 读入不阻断? . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 106 19.3 怎样显示一个百分比或“转动的短棒” 的进展表示器? . . . . . . 106 19.4 怎样清屏?怎样输出彩色文本?怎样移动光标到指定位置? . . . 106 19.5 怎样读入方向键, 功能键? . . . . . . . . . . . . . . . . . . . . . . 107 19.6 怎样读入鼠标输入? . . . . . . . . . . . . . . . . . . . . . . . . . 107 19.7 怎样做串口(“comm”) 的输入输出? . . . . . . . . . . . . . . . . 107 19.8 怎样直接输出到打印机? . . . . . . . . . . . . . . . . . . . . . . . 107 19.9 怎样发送控制终端或其它设备的逃逸指令序列? . . . . . . . . . . 108 19.10 怎样直接访问输入输出板? . . . . . . . . . . . . . . . . . . . . . 108 19.11 怎样做图形? . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 108 19.12 怎样显示GIF 和JPEG 图象? . . . . . . . . . . . . . . . . . . . 108 19.13 怎样检验一个文件是否存在? . . . . . . . . . . . . . . . . . . . . 108 19.14 怎样在读入文件前, 知道文件大小? . . . . . . . . . . . . . . . . . 109 19.15 怎样得到文件的修改日期和时间? . . . . . . . . . . . . . . . . . 109 19.16 怎样缩短一个文件而不用清除或重写? . . . . . . . . . . . . . . . 109 19.17 怎样在文件插入或删除一行(或记录)? . . . . . . . . . . . . . . 109 19.18 怎样从一个打开的流或文件描述符得到文件名? . . . . . . . . . . 110 19.19 怎样删除一个文件? . . . . . . . . . . . . . . . . . . . . . . . . . 110 19.20 怎样复制一个文件? . . . . . . . . . . . . . . . . . . . . . . . . . 110 19.21 为什么用了详尽的路径还不能打开文件? fopen("c:n newdir nfile.dat", "r") 返回错误。. . . . . . . . . . . . . . . . . . . . . . 110 19.22 fopen() 不让我打开文件: "$HOME/.profile" 和"˜/ .myrcfile"。. 111 19.23 怎样制止MS-DOS 下令人担忧的“Abort, Retry, Ignore?” 信息? 111 19.24 遇到“Too many open files (打开文件太多)” 的错误, 怎样增加同 时打开文件的允许数目? . . . . . . . . . . . . . . . . . . . . . . . 111 19.25 怎样在C 读入目录? . . . . . . . . . . . . . . . . . . . . . . . . 111 19.26 怎样找出系统还有多少内存可用? . . . . . . . . . . . . . . . . . 111 19.27 怎样分配大于64K 的数组或结构? . . . . . . . . . . . . . . . . . 111 19.28 错误信息“DGROUP data allocation exceeds 64K (DGROUP 数 据分配内存超过64K)” 说明什么?我应该怎么做?我以为使用了 大内存模型, 那我就可以使用多于64K 的数据! . . . . . . . . . . 112 19.29 怎样访问位于某的特定地址的内存(内存映射的设备或图显内存)? 112 19.30 怎样在一个C 程序调用另一个程序(独立可执行的程序, 或系统 命令)? . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 112 19.31 怎样调用另一个程序或命令, 同时收集它的输出? . . . . . . . . . 113 19.32 怎样才能发现程序自己的执行文件的全路径? . . . . . . . . . . . 113 19.33 怎样找出和执行文件在同一目录的配置文件? . . . . . . . . . . . 113 19.34 一个进程如何改变它的调用者的环境变量? . . . . . . . . . . . . 113 19.35 怎样读入一个对象文件并跳跃到其的地址? . . . . . . . . . . . 114 19.36 怎样实现精度小于秒的延时或记录用户回应的时间? . . . . . . . 114 19.37 怎样抓获或忽略像control-C 这样的键盘断? . . . . . . . . . . 114 19.38 怎样很好地处理浮点异常? . . . . . . . . . . . . . . . . . . . . . 115 19.39 怎样使用socket?网络化?写客户/服务器程序? . . . . . . . . . 115 19.40 怎样调用BIOS 函数?写ISR?创建TSR? . . . . . . . . . . . . 115 19.41 编译程序, 编译器出示“union REGS” 未定义错误信息, 连接器出 示“int86()” 的未定义错误信息。. . . . . . . . . . . . . . . . . . 115 19.42 什么是“near” 和“far” 指针? . . . . . . . . . . . . . . . . . . . . 116 19.43 我不能使用这些非标准、依赖系统的函数, 程序需要兼容ANSI! . 116 20 杂项117 20.1 怎样从一个函数返回多个? . . . . . . . . . . . . . . . . . . . . 117 20.2 怎样访问命令行参数? . . . . . . . . . . . . . . . . . . . . . . . . 117 20.3 怎样写数据文件, 使之可以在不同字大小、字节顺序或浮点格式 的机器上读入? . . . . . . . . . . . . . . . . . . . . . . . . . . . . 117 20.4 怎样调用一个由char * 指针指向函数名的函数? . . . . . . . . . 117 20.5 怎样实现比特数组或集合? . . . . . . . . . . . . . . . . . . . . . 118 20.6 怎样判断机器的字节顺序是高字节在前还是低字节在前? . . . . . 118 20.7 怎样掉换字节? . . . . . . . . . . . . . . . . . . . . . . . . . . . . 118 20.8 怎样转换整数到二进制或十六进制? . . . . . . . . . . . . . . . . 119 20.9 我可以使用二进制常数吗?有printf() 的二进制的格式符吗? . . 119 20.10 什么是计算整数比特为1 的个数的最有效的方法? . . . . . . . 119 20.11 什么是提高程序效率的最好方法? . . . . . . . . . . . . . . . . . 119 20.12 指针真得比数组快吗?函数调用会拖慢程序多少? ++i 比i = i +1 快吗? . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 120 20.13 人们说编译器优化的很好, 我们不在需要为速度而写汇编了, 但我 的编译器连用移位代替i/=2 都做不到。. . . . . . . . . . . . . . 120 20.14 怎样不用临时变量交换两个? . . . . . . . . . . . . . . . . . 120 20.15 是否有根据字符串做切换的方法? . . . . . . . . . . . . . . . . . 121 20.16 是否有使用非常量case 标志的方法(例如范围或任意的表达式)? 121 20.17 return 语句外层的括号是否真的可选择? . . . . . . . . . . . . . . 121 20.18 为什么C 注释不能嵌套?怎样注释掉含有注释的代码?引用字符 串内的注释是否合法? . . . . . . . . . . . . . . . . . . . . . . . . 121 20.19 C 是个伟大的语言还是别的?哪个其它语言可以写象a+++++b 这样的代码? . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 122 20.20 为什么C 没有嵌套函数? . . . . . . . . . . . . . . . . . . . . . . 122 20.21 assert() 是什么?怎样用它? . . . . . . . . . . . . . . . . . . . . . 122 20.22 怎样从C 调用FORTRAN (C++, BASIC, Pascal, Ada, LISP) 的函数?反之亦然? . . . . . . . . . . . . . . . . . . . . . . . . . 122 20.23 有什么程序可以做从Pascal 或Fortran (或LISP, Ada, awk, “老” C) 到C 的转换? . . . . . . . . . . . . . . . . . . . . . . . . . . . 123 20.24 C++ 是C 的超集吗?可以用C++ 编译器来编译C 代码吗? . . 123 20.25 需要用到“近似” 的strcmp, 比较两个字符串的近似度, 并不需要 完全一样。. . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 123 20.26 什么是散列法? . . . . . . . . . . . . . . . . . . . . . . . . . . . . 124 20.27 由一个日期, 怎样知道是星期几? . . . . . . . . . . . . . . . . . . 124 20.28 (year%4 == 0) 是否足够判断润年?2000 年是闰年吗? . . . . . . 124 20.29 一个难题: 怎样写一个输出自己源代码的程序? . . . . . . . . . . 124 20.30 什么是“达夫设备” (Duff’s Device)? . . . . . . . . . . . . . . . . 125 20.31 下届国际C 混乱代码竞赛(IOCCC) 什么时候进行?哪里可以找 到当前和以前的获胜代码? . . . . . . . . . . . . . . . . . . . . . 125 20.32 [K&R1] 提到的关健字entry 是什么? . . . . . . . . . . . . . . . . 126 20.33 C 的名字从何而来? . . . . . . . . . . . . . . . . . . . . . . . . . 126 20.34 “char” 如何发音? . . . . . . . . . . . . . . . . . . . . . . . . . . 126 20.35 “lvalue” 和“rvalue” 代表什么意思? . . . . . . . . . . . . . . . . 126 20.36 哪里可以取得本FAQ (英文版) 的额外副本? . . . . . . . . . . . 126
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 对于代码 in
1.static有什么用途?(请至少说明两种) 1)在函数体,一个被声明为静态的变量在这一函数被调用过程维持其不变。 2) 在模块内(但在函数体外),一个被声明为静态的变量可以被模块内所用函数访问,但不能被模块外其它函数访问。它是一个本地的全局变量。 3) 在模块内,一个被声明为静态的函数只可被这一模块内的其它函数调用。那就是,这个函数被限制在声明它的模块的本地范围内使用 2.引用与指针有什么区别? 1) 引用必须被初始化,指针不必。 2) 引用初始化以后不能被改变,指针可以改变所指的对象。 3) 不存在指向空的引用,但是存在指向空的指针。 3.描述实时系统的基本特性 在特定时间内完成特定的任务,实时性与可靠性。 4.全局变量和局部变量在内存是否有区别?如果有,是什么区别? 全局变量储存在静态数据库,局部变量在堆栈。 5.什么是平衡二叉树? 左右子树都是平衡二叉树 且左右子树的深度差的绝对不大于1。 6.堆栈溢出一般是由什么原因导致的? 没有回收垃圾资源。 7.什么函数不能声明为虚函数? constructor函数不能声明为虚函数。 8.冒泡排序算法的时间复杂度是什么? 时间复杂度是O(n^2)。 9.写出float x 与“零”比较的if语句。 if(x>0.000001&&x<-0.000001) 10.Internet采用哪种网络协议?该协议的主要层次结构? Tcp/Ip协议 主要层次结构为: 应用层/传输层/网络层/数据链路层/物理层。 11.Internet物理地址和IP地址转换采用什么协议? ARP (Address Resolution Protocol)(地址解析協議) 12.IP地址的编码分为哪俩部分? IP地址由两部分组成,网络号和主机号。不过是要和“子网掩码”按位与上之后才能区分哪些是网络位哪些是主机位。 13.用户输入M,N,从1至N开始顺序循环数数,每数到M输出该数,直至全部输出。写出C程序。 循环链表,用取余操作做 14.不能做switch()的参数类型是: switch的参数不能为实型。 1.写出判断ABCD四个表达式的是否正确, 若正确, 写出经过表达式 a的(3分) int a = 4; (A)a += (a++); (B) a += (++a) ;(C) (a++) += a;(D) (++a) += (a++); a = ? 答:C错误,左侧不是一个有效变量,不能赋,可改为(++a) += a; 改后答案依次为9,10,10,11 2.某32位系统下, C++程序,请计算sizeof 的(5分). char str[] = “http://www.ibegroup.com/” char *p = str ; int n = 10; 请计算 sizeof (str ) = ?(1) sizeof ( p ) = ?(2) sizeof ( n ) = ?(3) void Foo ( char str[100]){ 请计算 sizeof( str ) = ?(4) } void *p = malloc( 100 ); 请计算 sizeof ( p ) = ?(5) 答:(1)17 (2)4 (3) 4 (4)4 (5)4 3. 回答下面的问题. (4分) (1).头文件的 ifndef/define/endif 干什么用?预处理 答:防止头文件被重复引用 (2). #i nclude 和 #i nclude “filename.h” 有什么区别? 答:前者用来包含开发环境提供的库头文件,后者用来包含自己编写的头文件。 (3).在C++ 程序调用被 C 编译器编译后的函数,为什么要加 extern “C”声明? 答:函数和变量C++编译后在符号库的名字与C语言的不同,被extern "C"修饰的变 量和函数是按照C语言方式编译和连接的。由于编译后的名字不同,C++程序不能直接调 用C 函数。C++提供了一个C 连接交换指定符号extern“C”来解决这个问题。 (4). switch()不允许的数据类型是? 答:实型 4. 回答下面的问题(6分) (1).Void GetMemory(char **p, int num){ *p = (char *)malloc(num); } void Test(void){ char *str = NULL; GetMemory(&str, 100); strcpy(str, "hello"); printf(str); } 请问运行Test 函数会有什么样的结果? 答:输出“hello” (2). void Test(void){ char *str = (char *) malloc(100); strcpy(str, “hello”); free(str); if(str != NULL){ strcpy(str, “world”); printf(str); } } 请问运行Test 函数会有什么样的结果? 答:输出“world” (3). char *GetMemory(void){ char p[] = "hello world"; return p; } void Test(void){ char *str = NULL; str = GetMemory(); printf(str); } 请问运行Test 函数会有什么样的结果? 答:无效的指针,输出不确定 5. 编写strcat函数(6分) 已知strcat函数的原型是char *strcat (char *strDest, const char *strSrc); 其strDest 是目的字符串,strSrc 是源字符串。 (1)不调用C++/C 的字符串库函数,请编写函数 strcat 答: VC源码: char * __cdecl strcat (char * dst, const char * src) { char * cp = dst; while( *cp ) cp++; /* find end of dst */ while( *cp++ = *src++ ) ; /* Copy src to end of dst */ return( dst ); /* return dst */ } (2)strcat能把strSrc 的内容连接到strDest,为什么还要char * 类型的返回? 答:方便赋给其他变量 6.MFCCString是类型安全类么? 答:不是,其它数据类型转换到CString可以使用CString的成员函数Format来转换 7.C++为什么用模板类。 答:(1)可用来创建动态增长和减小的数据结构 (2)它是类型无关的,因此具有很高的可复用性。 (3)它在编译时而不是运行时检查数据类型,保证了类型安全 (4)它是平台无关的,可移植性 (5)可用于基本数据类型 8.CSingleLock是干什么的。 答:同步多个线程对一个数据类的同时访问 9.NEWTEXTMETRIC 是什么。 答:物理字体结构,用来设置字体的高宽大小 10.程序什么时候应该使用线程,什么时候单线程效率高。 答:1.耗时的操作使用线程,提高应用程序响应 2.并行操作时使用线程,如C/S架构的服务器端并发线程响应用户的请求。 3.多CPU系统使用线程提高CPU利用率 4.改善程序结构。一个既长又复杂的进程可以考虑分为多个线程,成为几个独立或半独 立的运行部分,这样的程序会利于理解和修改。 其他情况都使用单线程。 11.Windows是内核级线程么。 答:见下一题 12.Linux有内核级线程么。 答:线程通常被定义为一个进程代码的不同执行路线。从实现方式上划分,线程有两 种类型:“用户级线程”和“内核级线程”。 用户线程指不需要内核支持而在用户程序 实现的线程,其不依赖于操作系统核心,应用进程利用线程库提供创建、同步、调度 和管理线程的函数来控制用户线程。这种线程甚至在象 DOS 这样的操作系统也可实现 ,但线程的调度需要用户程序完成,这有些类似 Windows 3.x 的协作式多任务。另外一 种则需要内核的参与,由内核完成线程的调度。其依赖于操作系统核心,由内核的内部 需求进行创建和撤销,这两种模型各有其好处和缺点。用户线程不需要额外的内核开支 ,并且用户态线程的实现方式可以被定制或修改以适应特殊应用的要求,但是当一个线 程因 I/O 而处于等待状态时,整个进程就会被调度程序切换为等待状态,其他线程得不 到运行的机会;而内核线程则没有各个限制,有利于发挥多处理器的并发优势,但却占 用了更多的系统开支。 Windows NT和OS/2支持内核线程。Linux 支持内核级的多线程 13.C++什么数据分配在栈或堆,New分配数据是在近堆还是远堆? 答:栈: 存放局部变量,函数调用参数,函数返回,函数返回地址。由系统管理 堆: 程序运行时动态申请,new 和 malloc申请的内存就在堆上 14.使用线程是如何防止出现大的波峰。 答:意思是如何防止同时产生大量的线程,方法是使用线程池,线程池具有可以同时提 高调度效率和限制资源使用的好处,线程池的线程达到最大数时,其他线程就会排队 等候。 15函数模板与类模板有什么区别? 答:函数模板的实例化是由编译程序在处理函数调用时自动完成的,而类模板的实例化 必须由程序员在程序显式地指定。 16一般数据库若出现日志满了,会出现什么情况,是否还能使用? 答:只能执行查询等读操作,不能执行更改,备份等写操作,原因是任何写操作都要记 录日志。也就是说基本上处于不能使用的状态。 17 SQL Server是否支持行级锁,有什么好处? 答:支持,设立封锁机制主要是为了对并发操作进行控制,对干扰进行封锁,保证数据 的一致性和准确性,行级封锁确保在用户取得被更新的行到该行进行更新这段时间内不 被其它用户所修改。因而行级锁即可保证数据的一致性又能提高数据操作的迸发性。 18如果数据库满了会出现什么情况,是否还能使用? 答:见16 19 关于内存对齐的问题以及sizof()的输出 答:编译器自动对齐的原因:为了提高程序的性能,数据结构(尤其是栈)应该尽可能 地在自然边界上对齐。原因在于,为了访问未对齐的内存,处理器需要作两次内存访问 ;然而,对齐的内存访问仅需要一次访问。 20 int i=10, j=10, k=3; k*=i+j; k最后的是? 答:60,此题考察优先级,实际写成: k*=(i+j);,赋运算符优先级最低 21.对数据库的一张表进行操作,同时要对另一张表进行操作,如何实现? 答:将操作多个表的操作放入到事务进行处理 22.TCP/IP 建立连接的过程?(3-way shake) 答:在TCP/IP协议,TCP协议提供可靠的连接服务,采用三次握手建立一个连接。   第一次握手:建立连接时,客户端发送syn包(syn=j)到服务器,并进入SYN_SEND状 态,等待服务器确认; 第二次握手:服务器收到syn包,必须确认客户的SYN(ack=j+1),同时自己也发送一个 SYN包(syn=k),即SYN+ACK包,此时服务器进入SYN_RECV状态;   第三次握手:客户端收到服务器的SYN+ACK包,向服务器发送确认包ACK(ack=k+1) ,此包发送完毕,客户端和服务器进入ESTABLISHED状态,完成三次握手。 23.ICMP是什么协议,处于哪一层? 答:Internet控制报文协议,处于网络层(IP层) 24.触发器怎么工作的? 答:触发器主要是通过事件进行触发而被执行的,当对某一表进行诸如UPDATE、 INSERT 、 DELETE 这些操作时,数据库就会自动执行触发器所定义的SQL 语句,从而确保对数 据的处理必须符合由这些SQL 语句所定义的规则。 25.winsock建立连接的主要实现步骤? 答:服务器端:socker()建立套接字,绑定(bind)并监听(listen),用accept() 等待客户端连接。 客户端:socker()建立套接字,连接(connect)服务器,连接上后使用send()和recv( ),在套接字上写读数据,直至数据交换完毕,closesocket()关闭套接字。 服务器端:accept()发现有客户端连接,建立一个新的套接字,自身重新开始等待连 接。该新产生的套接字使用send()和recv()写读数据,直至数据交换完毕,closesock et()关闭套接字。 26.动态连接库的两种方式? 答:调用一个DLL的函数有两种方法: 1.载入时动态链接(load-time dynamic linking),模块非常明确调用某个导出函数 ,使得他们就像本地函数一样。这需要链接时链接那些函数所在DLL的导入库,导入库向 系统提供了载入DLL时所需的信息及DLL函数定位。 2.运行时动态链接(run-time dynamic linking),运行时可以通过LoadLibrary或Loa dLibraryEx函数载入DLL。DLL载入后,模块可以通过调用GetProcAddress获取DLL函数的 出口地址,然后就可以通过返回的函数指针调用DLL函数了。如此即可避免导入库文件了 。 27.IP组播有那些好处? 答:Internet上产生的许多新的应用,特别是高带宽的多媒体应用,带来了带宽的急剧 消耗和网络拥挤问题。组播是一种允许一个或多个发送者(组播源)发送单一的数据包 到多个接收者(一次的,同时的)的网络技术。组播可以大大的节省网络带宽,因为无 论有多少个目标地址,在整个网络的任何一条链路上只传送单一的数据包。所以说组播 技术的核心就是针对如何节约网络资源的前提下保证服务质量。
第 一 章 概述 1-1 简述计算机程序设计语言的发展历程。 解: 迄今为止计算机程序设计语言的发展经历了机器语言、汇编语言、高级语言等阶段,C++语言是一种面向对象的编程语言,也属于高级语言。 1-2 面向对象的编程语言有哪些特点? 解: 面向对象的编程语言与以往各种编程语言有根本的不同,它设计的出发点就是为了能更直接的描述客观世界存在的事物以及它们之间的关系。面向对象的编程语言将客观事物看作具有属性和行为的对象,通过抽象找出同一类对象的共同属性(静态特征)和行为(动态特征),形成类。通过类的继承与多态可以很方便地实现代码重用,大大缩短了软件开发周期,并使得软件风格统一。因此,面向对象的编程语言使程序能够比较直接地反问题域的本来面目,软件开发人员能够利用人类认识事物所采用的一般思维方法来进行软件开发。C++语言是目前应用最广的面向对象的编程语言。 1-3 什么是结构化程序设计方法?这种方法有哪些优点和缺点? 解: 结构化程序设计的思路是:自顶向下、逐步求精;其程序结构是按功能划分为若干个基本模块;各模块之间的关系尽可能简单,在功能上相对独立;每一模块内部均是由顺序、选择和循环三种基本结构组成;其模块化实现的具体方法是使用子程序。结构化程序设计由于采用了模块分解与功能抽象,自顶向下、分而治之的方法,从而有效地将一个较复杂的程序系统设计任务分解成许多易于控制和处理的子任务,便于开发和维护。 虽然结构化程序设计方法具有很多的优点,但它仍是一种面向过程的程序设计方法,它把数据和处理数据的过程分离为相互独立的实体。当数据结构改变时,所有相关的处理过程都要进行相应的修改,每一种相对于老问题的新方法都要带来额外的开销,程序的可重用性差。 由于图形用户界面的应用,程序运行由顺序运行演变为事件驱动,使得软件使用起来越来越方便,但开发起来却越来越困难,对这种软件的功能很难用过程来描述和实现,使用面向过程的方法来开发和维护都将非常困难。 1-4 什么是对象?什么是面向对象方法?这种方法有哪些特点? 解: 从一般意义上讲,对象是现实世界一个实际存在的事物,它可以是有形的,也可以是无形的。对象是构成世界的一个独立单位,它具有自己的静态特征和动态特征。面向对象方法的对象,是系统用来描述客观事物的一个实体,它是用来构成系统的一个基本单位,由一组属性和一组行为构成。 面向对象的方法将数据及对数据的操作方法放在一起,作为一个相互依存、不可分离的整体--对象。对同类型对象抽象出其共性,形成类。类的大多数数据,只能用本类的方法进行处理。类通过一个简单的外部接口,与外界发生关系,对象与对象之间通过消息进行通讯。这样,程序模块间的关系更为简单,程序模块的独立性、数据的安全性就有了良好的保障。通过实现继承与多态性,还可以大大提高程序的可重用性,使得软件的开发和维护都更为方便。 面向对象方法所强调的基本原则,就是直接面对客观存在的事物来进行软件开发,将人们在日常生活习惯的思维方式和表达方式应用在软件开发,使软件开发从过分专业化的方法、规则和技巧回到客观世界,回到人们通常的思维。 1-5 什么叫做封装? 解: 封装是面向对象方法的一个重要原则,就是把对象的属性和服务结合成一个独立的系统单位,并尽可能隐蔽对象的内部细节。 1-6 面向对象的软件工程包括哪些主要内容? 解: 面向对象的软件工程是面向对象方法在软件工程领域的全面应用,它包括面向对象的分析(OOA)、面向对象的设计(OOD)、面向对象的编程(OOP)、面向对象的测试(OOT)和面向对象的软件维护(OOSM)等主要内容。 1-7 简述计算机内部的信息可分为几类? 解: 计算机内部的信息可以分成控制信息和数据信息二大类;控制信息可分为指令和控制字两类;数据信息可分为数信息和非数信息两类。 1-8 什么叫二进制?使用二进制有何优点和缺点? 解: 二进制是基数为2,每位的权是以2 为底的幂的进制,遵循逢二进一原则,基本符号为0和1。采用二进制码表示信息,有如下几个优点:1.易于物理实现;2.二进制数运算简单;3.机器可靠性高;4.通用性强。其缺点是它表示数的容量较小,表示同一个数,二进制较其他进制需要更多的位数。 1-9 请将以下十进制数转换为二进制和十六进制补码: (1)2 (2)9 (3)93 (4)-32 (5)65535 (6)-1 解: (1) (2)10 = (10)2 = (2)16 (2) (9)10 = (1001)2 = (9)16 (3) (93)10 = (1011101)2 = (5D)16 (4) (-32)10 = (11100000)2 = (E0)16 (5) (65535)10 = (11111111 11111111)2 = (FFFF)16 (6) (-1)10 = (11111111 11111111)2 = (FFFF)16 1-10 请将以下数转换为十进制: (1)(1010)2 (2)(10001111)2 (3)(01011111 11000011)2 (4)(7F)16 (5)(2D3E)16 (6)(F10E)16 解: (1)(1010)2 = (10)10 (2)(10001111)2 = (143)10 (3)(01011111 11000011)2 = (24515)10 (4)(7F)16 = (127)10 (5)(2D3E)16 = (11582)10 (6)(F10E)16 = (61710)10 1-11 简要比较原码、反码、补码等几种编码方法。 解: 原码:将符号位数字化为 0 或 1,数的绝对与符号一起编码,即所谓"符号──绝对表示"的编码。 正数的反码和补码与原码表示相同。 负数的反码与原码有如下关系: 符号位相同(仍用1表示),其余各位取反(0变1,1变0)。 补码由该数反码的最末位加1求得。 第 二 章 C++简单程序设计 2-1 C++语言有那些主要特点和优点? 解: C++语言的主要特点表现在两个方面,一是全面兼容C,二是支持面向对象的方法。C++是一个更好的C,它保持了C的简洁、高效、接近汇编语言、具有良好的可读性和可移植性等特点,对C的类型系统进行了改革和扩充,因此C++比C更安全,C++的编译系统能检查出更多的类型错误。 C++语言最重要的特点是支持面向对象。 2-2 下列标识符哪些是合法的? Program, -page, _lock, test2, 3in1, @mail, A_B_C_D 解: Program, _lock, test2, A_B_C_D是合法的标识符,其它的不是。 2-3 例2.1每条语句的作用是什么? #include void main(void) { cout<<"Hello!\n"; cout<<"Welcome to c++!\n"; } 解: #include //指示编译器将文件iostream.h的代码 //嵌入到该程序该指令所在的地方 void main() //主函数名,void 表示函数没有返回 { //函数体标志 cout<<"Hello!\n"; //输出字符串Hello!到标准输出设备(显示器)上。 cout<<"Welcome to c++!\n"; //输出字符串Welcome to c++! } 在屏幕输出如下: Hello! Welcome to c++! 2-4 使用关键字const而不是#define语句的好处有哪些? 解: const定义的常量是有类型的,所以在使用它们时编译器可以查错;而且,这些变量在调试时仍然是可见的。 2-5 请写出C++语句声明一个常量PI,为3.1416;再声明一个浮点型变量a,把PI的赋给a。 解: const float PI = 3.1416; float a = PI; 2-6 在下面的枚举类型,Blue的是多少? enum COLOR { WHITE, BLACK = 100, RED, BLUE, GREEN = 300 }; 解: Blue = 102 2-7 注释有什么作用?C++有哪几种注释的方法?他们之间有什么区别? 解: 注释在程序的作用是对程序进行注解和说明,以便于阅读。编译系统在对源程序进行编译时不理会注释部分,因此注释对于程序的功能实现不起任何作用。而且由于编译时忽略注释部分,所以注释内容不会增加最终产生的可执行程序的大小。适当地使用注释,能够提高程序的可读性。在C++,有两种给出注释的方法:一种是延用C语言方法,使用"/*"和"*/"括起注释文字。另一种方法是使用"//",从"//"开始,直到它所在行的行尾,所有字符都被作为注释处理。 2-8 什么叫做表达式?x = 5 + 7是一个表达式吗?它的是多少? 解: 任何一个用于计算的公式都可称为表达式。x = 5 + 7是一个表达式,它的为12。 2-9 下列表达式的是多少? 1. 201 / 4 2. 201 % 4 3. 201 / 4.0 解: 1. 50 2. 1 3. 50.25 2-10 执行完下列语句后,a、b、c三个变量为多少? a = 30; b = a++; c = ++a; 解: a:32 ; b:30 ; c:32; 2-11 在一个for循环,可以初始化多个变量吗?如何实现? 解: 在for循环设置条件的第一个";"前,用,分隔不同的赋表达式。 例如: for (x = 0, y = 10; x < 100; x++, y++) 2-12 执行完下列语句后,n的为多少? int n; for (n = 0; n < 100; n++) 解: n的为100 2-13 写一条for语句,计数条件为n从100到200,步长为2;然后用while和do…while语句完成同样的循环。 解: for循环: for (int n = 100; n <= 200; n += 2); while循环: int x = 100; while (n <= 200) n += 2; do…while循环: int n = 100; do { n += 2; } while(n y) x = y; else // y > x || y == x y = x; 2-17 修改下面这个程序的错误,改正后它的运行结果是什么? #include void main() int i int j; i = 10; /* 给i赋 j = 20; /* 给j赋 */ cout << "i + j = << i + j; /* 输出结果 */ return 0; } 解: 改正: #include int main() { int i; int j; i = 10; // 给i赋 j = 20; /* 给j赋 */ cout << "i + j = " << i + j; /* 输出结果 */ return 0; } 程序运行输出: i + j = 30 2-18 编写一个程序,运行时提示输入一个数字,再把这个数字显示出来。 解: 源程序: #include int main() { int i; cout <> i; cout << "您输入一个数字是" << i << endl; return 0; } 程序运行输出: 请输入一个数字:5 您输入一个数字是5 2-19 C++有哪几种数据类型?简述其域。编程显示你使用的计算机的各种数据类型的字节数。 解: 源程序: #include int main() { cout << "The size of an int is:\t\t" << sizeof(int) << " bytes.\n"; cout << "The size of a short int is:\t" << sizeof(short) << " bytes.\n"; cout << "The size of a long int is:\t" << sizeof(long) << " bytes.\n"; cout << "The size of a char is:\t\t" << sizeof(char) << " bytes.\n"; cout << "The size of a float is:\t\t" << sizeof(float) << " bytes.\n"; cout << "The size of a double is:\t" << sizeof(double) << " bytes.\n"; return 0; } 程序运行输出: The size of an int is: 4 bytes. The size of a short int is: 2 bytes. The size of a long int is: 4 bytes. The size of a char is: 1 bytes. The size of a float is: 4 bytes. The size of a double is: 8 bytes. 2-20 打印ASCII码为32~127的字符。 解: #include int main() { for (int i = 32; i<128; i++) cout << (char) i; return 0; } 程序运行输出: !"#$%G'()*+,./0123456789:;?@ABCDEFGHIJKLMNOP_QRSTUVWXYZ[\]^'abcdefghijklmnopqrstuvwxyz~s 2-21 运行下面的程序,观察其输出,与你的设想是否相同? #include int main() { unsigned int x; unsigned int y = 100; unsigned int z = 50; x= y - z; cout << "Difference is: " << x; x = z - y; cout << "\nNow difference is: " << x <<endl; return 0; } 解: 程序运行输出: Difference is: 50 Now difference is: 4294967246 注意,第二行的输出并非 -50,注意x、y、z的数据类型。 2-22 运行下面的程序,观察其输出,体会i++与++i的差别。 #include int main() { int myAge = 39; // initialize two integers int yourAge = 39; cout << "I am: " << myAge << " years old.\n"; cout << "You are: " << yourAge << " years old\n"; myAge++; // postfix increment ++yourAge; // prefix increment cout << "One year passes...\n"; cout << "I am: " << myAge << " years old.\n"; cout << "You are: " << yourAge << " years old\n"; cout << "Another year passes\n"; cout << "I am: " << myAge++ << " years old.\n"; cout << "You are: " << ++yourAge << " years old\n"; cout << "Let's print it again.\n"; cout << "I am: " << myAge << " years old.\n"; cout << "You are: " << yourAge << " years old\n"; return 0; } 解: 程序运行输出: I am 39 years old You are 39 years old One year passes I am 40 years old You are 40 years old Another year passes I am 40 years old You are 41 years old Let's print it again I am 41 years old You are 41 years old 2-23 什么叫常量?什么叫变量? 解: 所谓常量是指在程序运行的整个过程始终不可改变的量,除了用文字表示常量外,也可以为常量命名,这就是符号常量;在程序的执行过程可以变化的量称为变量变量是需要用名字来标识的。 2-24 变量有哪几种存储类型? 解: 变量有以下几种存储类型: auto存储类型:采用堆栈方式分配内存空间,属于一时性存储,其存储空间可以被若干变量多次覆盖使用; register存储类型:存放在通用寄存器; extern存储类型:在所有函数和程序段都可引用; static存储类型:在内存是以固定地址存放的,在整个程序运行期间都有效。 2-25 写出下列表达式的: 1. 2 < 3 && 6 < 9 2. ! ( 4 5) || (6 > 2 解: 1. 1 2. -1 3. 0 4. 0 2-28 编写一个完整的程序,实现功能:向用户提问"现在正在下雨吗?",提示用户输入Y或N。若输入为Y,显示"现在正在下雨。"; 若输入为N,显示"现在没有下雨。";否则继续提问"现在正在下雨吗?" 解: 源程序: #include #include void main() { char flag; while(1) { cout <> flag; if ( toupper(flag) == 'Y') { cout << "现在正在下雨。"; break; } if ( toupper(flag) == 'N') { cout << "现在没有下雨。"; break; } } } 程序运行输出: 现在正在下雨吗?(Yes or No):x 现在正在下雨吗?(Yes or No):l 现在正在下雨吗?(Yes or No):q 现在正在下雨吗?(Yes or No):n 现在没有下雨。 或: 现在正在下雨吗?(Yes or No):y 现在正在下雨。 2-29 编写一个完整的程序,运行时向用户提问"你考试考了多少分?(0~100)",接收输入后判断其等级,显示出来。规则如下: 解: #include void main() { int i,score; cout <> score; if (score>100 || score<0) cout << "分数必须在0到100之间!"; else { i = score/10; switch (i) { case 10: case 9: cout << "你的成绩为优!"; break; case 8: cout << "你的成绩为良!"; break; case 7: case 6: cout << "你的成绩为!"; break; default: cout << "你的成绩为差!"; } } } 程序运行输出: 你考试考了多少分?(0~100):85 你的成绩为良! 2-30 (1)实现一个简单的菜单程序,运行时显示"Menu: A(dd) D(elete) S(ort) Q(uit), Select one:"提示用户输入,A表示增加,D表示删除,S表示排序,Q表示退出,输入为A、D、S时分别提示"数据已经增加、删除、排序。"输入为Q时程序结束。要求使用if … else语句进行判断,用break、continue控制程序流程。 解: #include #include void main() { char choice,c; while(1) { cout <> c; choice = toupper(c); if (choice == 'A') { cout << "数据已经增加. " << endl; continue; } else if (choice == 'D') { cout << "数据已经删除. " << endl; continue; } else if (choice == 'S') { cout << "数据已经排序. " << endl; continue; } else if (choice == 'Q') break; } } 程序运行输出: Menu: A(dd) D(elete) S(ort) Q(uit), Select one:a 数据已经增加. Menu: A(dd) D(elete) S(ort) Q(uit), Select one:d 数据已经删除. Menu: A(dd) D(elete) S(ort) Q(uit), Select one:s 数据已经排序. Menu: A(dd) D(elete) S(ort) Q(uit), Select one:q (2)实现一个简单的菜单程序,运行时显示"Menu: A(dd) D(elete) S(ort) Q(uit), Select one:"提示用户输入,A表示增加,D表示删除,S表示排序,Q表示退出,输入为A、D、S时分别提示"数据已经增加、删除、排序。"输入为Q时程序结束。要求使用Switch语句。 解: 源程序: #include #include void main() { char choice; while(1) { cout <> choice; switch(toupper(choice)) { case 'A': cout << "数据已经增加. " << endl; break; case 'D': cout << "数据已经删除. " << endl; break; case 'S': cout << "数据已经排序. " << endl; break; case 'Q': exit(0); break; default: ; } } } 程序运行输出: Menu: A(dd) D(elete) S(ort) Q(uit), Select one:a 数据已经增加. Menu: A(dd) D(elete) S(ort) Q(uit), Select one:d 数据已经删除. Menu: A(dd) D(elete) S(ort) Q(uit), Select one:s 数据已经排序. Menu: A(dd) D(elete) S(ort) Q(uit), Select one:q 2-31 用穷举法找出1~100间的质数,显示出来。分别使用while,do-while,for循环语句实现。 解: 源程序: 使用while循环语句: #include #include void main() { int i,j,k,flag; i = 2; while(i <= 100) { flag = 1; k = sqrt(i); j = 2; while (j <= k) { if(i%j == 0) { flag = 0; break; } j++; } if (flag) cout << i << "是质数." << endl; i++; } } 使用do…while循环语句: #include #include void main() { int i,j,k,flag; i = 2; do{ flag = 1; k = sqrt(i); j = 2; do{ if(i%j == 0) { flag = 0; break; } j++; }while (j <= k); if (flag) cout << i << "是质数." << endl; i++; }while(i <= 100); } 使用for循环语句: #include #include void main() { int i,j,k,flag; for(i = 2; i <= 100; i++) { flag = 1; k = sqrt(i); for (j = 2; j <= k; j++) { if(i%j == 0) { flag = 0; break; } } if (flag) cout << i << "是质数." << endl; } } 程序运行输出: 2是质数. 3是质数. 5是质数. 7是质数. 11是质数. 13是质数. 17是质数. 19是质数. 23是质数. 29是质数. 31是质数. 37是质数. 41是质数. 43是质数. 47是质数. 53是质数. 59是质数. 61是质数. 67是质数. 71是质数. 73是质数. 79是质数. 83是质数. 89是质数. 97是质数. 2-32 比较Break语句与Continue语句的不同用法。 解: Break使程序从循环体和switch语句内跳出,继续执行逻辑上的下一条语句,不能用在别处; continue 语句结束本次循环,接着开始判断决定是否继续执行下一次循环; 2-33 定义一个表示时间的结构体,可以精确表示年、月、日、小时、分、秒;提示用户输入年、月、日、小时、分、秒的,然后完整地显示出来。 解: 源程序见"实验指导"部分实验二 2-34 在程序定义一个整型变量,赋以1~100的,要求用户猜这个数,比较两个数的大小,把结果提示给用户,直到猜对为止。分别使用while、do…while语句实现循环。 解: //使用while语句 #include void main() { int n = 18; int m = 0; while(m != n) { cout <> m; if (n > m) cout << "你猜的太小了!" << endl; else if (n < m) cout << "你猜的太大了!" << endl; else cout << "你猜对了!" << endl; } } //使用do…while语句 #include void main() { int n = 18; int m = 0; do{ cout <> m; if (n > m) cout << "你猜的太小了!" << endl; else if (n < m) cout << "你猜的太大了!" << endl; else cout << "你猜对了!" << endl; }while(n != m); } 程序运行输出: 请猜这个数的为多少?(0~~100):50 你猜的太大了! 请猜这个数的为多少?(0~~100):25 你猜的太大了! 请猜这个数的为多少?(0~~100):10 你猜的太小了! 请猜这个数的为多少?(0~~100):15 你猜的太小了! 请猜这个数的为多少?(0~~100):18 你猜对了! 2-35 定义枚举类型weekday,包括Sunday到Saturday七个元素在程序定义weekday类型的变量,对其赋,定义整型变量,看看能否对其赋weekday类型的。 解: #include enum weekday { Sunday, Monday, Tuesday, Wednesday, Thursday, Friday, Saturday }; void main() { int i; weekday d = Thursday; cout << "d = " << d << endl; i = d; cout << "i = " << i << endl; d = (weekday)6; cout << "d = " << d << endl; d = weekday( 4 ); cout << "d = " << d << endl; } 程序运行输出: d = 4 i = 4 d = 6 d = 4 2-36口袋有红、黄、蓝、白、黑五种颜色的球若干个,每次从口袋取出三个不同颜色的球,问有多少种取法。 解: #include using namespace std; int main() { enum color{red,yellow,blue,white,black}; enum color pri; int n,loop,i,j,k; char c; n=0; for(i=red;i<=black;i++) for(j=red;j<=black;j++) if(i!=j) { for(k=red;k<black;k++) if((k!=i)&&(k!=j)) { n=n+1; cout.width(4); cout<<n; for(loop=1;loop<=3;loop++) { switch(loop) { case 1: pri=(enum color)i; break; case 2: pri=(enum color)j; break; case 3: pri=(enum color)k; break; default: break; } switch(pri) { case red:cout<<"red";break; case yellow:cout<<"yellow";break; case blue:cout<<"blue";break; case white:cout<<"white";break; case black:cout<<"black";break; default: break; } } cout<<endl; } } cout<<"total:"<<n<<endl; } 2-37输出九九算表 #include #include using namespace std; int main() { int i,j; for(i=1;i<5;i++) cout<<setw(4)<<i; cout<<endl; cout<<endl; for(i=1;i<5;i++) { for(j=1;j<5;j++) cout<<setw(4)<<(i*j); cout<<endl; } } 第三章 函数 3-1 C++的函数是什么?什么叫主调函数,什么叫被调函数,二者之间有什么关系?如何调用一个函数? 解: 一个较为复杂的系统往往需要划分为若干子系统,高级语言的子程序就是用来实现这种模块划分的。C和C++语言的子程序就体现为函数。调用其它函数的函数被称为主调函数,被其它函数调用的函数称为被调函数。一个函数很可能既调用别的函数又被另外的函数调用,这样它可能在某一个调用与被调用关系充当主调函数,而在另一个调用与被调用关系充当被调函数。 调用函数之前先要声明函数原型。按如下形式声明: 类型标识符 被调函数名 (含类型说明的形参表); 声明了函数原型之后,便可以按如下形式调用子函数: 函数名(实参列表) 3-2 观察下面程序的运行输出,与你设想的有何不同?仔细体会引用的用法。 源程序: #include int main() { int intOne; int &rSomeRef; = intOne; intOne = 5; cout << "intOne:\t\t" << intOne << endl; cout << "rSomeRef:\t" << rSomeRef << endl; int intTwo = 8; rSomeRef = intTwo; // not what you think! cout << "\nintOne:\t\t" << intOne << endl; cout << "intTwo:\t\t" << intTwo << endl; cout << "rSomeRef:\t" << rSomeRef << endl; return 0; } 程序运行输出: intOne: 5 rSomeRef: 5 intOne: 8 intTwo: 8 rSomeRef: 8 3-3 比较调用和引用调用的相同点与不同点。 解: 调用是指当发生函数调用时,给形参分配内存空间,并用实参来初始化形参(直接将实参的传递给形参)。这一过程是参数的单向传递过程,一旦形参获得了便与实参脱离关系,此后无论形参发生了怎样的改变,都不会影响到实参。 引用调用将引用作为形参,在执行主调函数的调用语句时,系统自动用实参来初始化形参。这样形参就成为实参的一个别名,对形参的任何操作也就直接作用于实参。 3-4 什么叫内联函数?它有哪些特点? 解: 定义时使用关键字 inline的函数叫做内联函数; 编译器在编译时在调用处用函数体进行替换,节省了参数传递、控制转移等开销; 内联函数体内不能有循环语句和switch语句; 内联函数的定义必须出现在内联函数第一次被调用之前; 对内联函数不能进行异常接口声明; 3-5 函数原型的参数名与函数定义的参数名以及函数调用的参数名必须一致吗? 解: 不必一致,所有的参数是根据位置和类型而不是名字来区分的。 3-6 重载函数时通过什么来区分? 解: 重载的函数的函数名是相同的,但它们的参数的个数和数据类型不同,编译器根据实参和形参的类型及个数的最佳匹配,自动确定调用哪一个函数。 3-7 编写函数,参数为两个unsigned short int型数,返回为第一个参数除以第二个参数的结果,数据类型为short int;如果第二个参数为0,则返回为-1。在主程序实现输入输出。 解: 源程序: #include short int Divider(unsigned short int a, unsigned short int b) { if (b == 0) return -1; else return a/b; } typedef unsigned short int USHORT; typedef unsigned long int ULONG; int main() { USHORT one, two; short int answer; cout <> one; cout <> two; answer = Divider(one, two); if (answer > -1) cout << "Answer: " << answer; else cout << "Error, can't divide by zero!"; return 0; } 程序运行输出: Enter two numbers. Number one:8 Number two:2 Answer: 4 3-8 编写函数把华氏温度转换为摄氏温度,公式为:C = (F - 32) * 5/9; 在主程序提示用户输入一个华氏温度,转化后输出相应的摄氏温度。 解: 源程序见"实验指导"部分实验三 3-9 编写函数判断一个数是否是质数,在主程序实现输入、输出。 解: #include #include int prime(int i); //判一个数是否是质数的函数 void main() { int i; cout <> i; if (prime(i)) cout << i << "是质数." << endl; else cout << i << "不是质数." << endl; } int prime(int i) { int j,k,flag; flag = 1; k = sqrt(i); for (j = 2; j <= k; j++) { if(i%j == 0) { flag = 0; break; } } if (flag) return 1; else return 0; } 程序运行输出: 请输入一个整数:1151 1151是质数. 3-10 编写函数求两个整数的最大公约数和最小公倍数。 解: 源程序: #include #include int fn1(int i,int j); //求最大公约数的函数 void main() { int i,j,x,y; cout <> i ; cout <> j ; x = fn1(i,j); y = i * j / x; cout << i << "和" << j << "的最大公约数是:" << x << endl; cout << i << "和" << j << "的最小公倍数是:" << y << endl; } int fn1(int i, int j) { int temp; if (i < j) { temp = i; i = j; j = i; } while(j != 0) { temp = i % j; i = j; j = temp; } return i; } 程序运行输出: 请输入一个正整数:120 请输入另一个正整数:72 120和72的最大公约数是:24 120和72的最小公倍数是:360 3-11 什么叫作嵌套调用?什么叫作递归调用? 解: 函数允许嵌套调用,如果函数1调用了函数2,函数2再调用函数3,便形成了函数的嵌套调用。 函数可以直接或间接地调用自身,称为递归调用。 3-12 在主程序提示输入整数n,编写函数用递归的方法求1 + 2 + … + n的。 解: #include #include int fn1(int i); void main() { int i; cout <> i ; cout << "从1累加到" <<i << "的和为:" << fn1(i) << endl; } int fn1(int i) { if (i == 1) return 1; else return i + fn1(i -1); } 程序运行输出: 请输入一个正整数:100 从1累加到100的和为:5050 3-13 编写递归函数GetPower(int x, int y)计算x的y次幂, 在主程序实现输入输出。 解: 源程序: #include long GetPower(int x, int y); int main() { int number, power; long answer; cout <> number; cout <> power; answer = GetPower(number,power); cout << number << " to the " << power << "th power is " <<answer <2; fib(1) = fib(2) = 1;观察递归调用的过程。 解: 源程序见"实验指导"部分实验三 3-15 用递归的方法编写函数求n阶勒让德多项式的,在主程序实现输入、输出; 解: #include float p(int n, int x); void main() { int n,x; cout <> n; cout <> x; cout << "n = " << n << endl; cout << "x = " << x << endl; cout << "P" << n << "(" << x << ") = " << p(n,x) << endl; } float p(int n, int x) { if (n == 0) return 1; else if (n == 1) return x; else return ((2*n-1)*x*p(n-1,x) - (n-1)*p(n-2,x)) /n ; } 程序运行输出: 请输入正整数n:1 请输入正整数x:2 n = 1 x = 2 P1(2) = 2 请输入正整数n:3 请输入正整数x:4 n = 3 x = 4 P3(4) = 154 第 四 章 类 4-1 解释public和private的作用,公有类型成员与私有类型成员有些什么区别? 解: 公有类型成员用public关键字声明,公有类型定义了类的外部接口;私有类型的成员用private关键字声明,只允许本类的函数成员来访问,而类外部的任何访问都是非法的,这样,私有的成员就整个隐蔽在类,在类的外部根本就无法看到,实现了访问权限的有效控制。 4-2 protected关键字有何作用? 解: protected用来声明保护类型的成员,保护类型的性质和私有类型的性质相似,其差别在于继承和派生时派生类的成员函数可以访问基类的保护成员。 4-3 构造函数和析构函数有什么作用? 解: 构造函数的作用就是在对象被创建时利用特定的构造对象,将对象初始化为一个特定的状态,使此对象具有区别于彼对象的特征,完成的就是是一个从一般到具体的过程,构造函数在对象创建的时候由系统自动调用。 析构函数与构造函数的作用几乎正好相反,它是用来完成对象被删除前的一些清理工作,也就是专门作扫尾工作的。一般情况下,析构函数是在对象的生存期即将结束的时刻由系统自动调用的,它的调用完成之后,对象也就消失了,相应的内存空间也被释放。 4-4 数据成员可以为公有的吗?成员函数可以为私有的吗? 解: 可以,二者都是合法的。数据成员和成员函数都可以为公有或私有的。但数据成员最好定义为私有的。 4-5 已知class A有数据成员int a,如果定义了A的两个对象A1、A2,它们各自的数据成员a的可以不同吗? 解: 可以,类的每一个对象都有自己的数据成员。 4-6 什么叫做拷贝构造函数?拷贝构造函数何时被调用? 解: 拷贝构造函数是一种特殊的构造函数,具有一般构造函数的所有特性,其形参是本类的对象的引用,其作用是使用一个已经存在的对象,去初始化一个新的同类的对象。在以下三种情况下会被调用:在当用类的一个对象去初始化该类的另一个对象时;如果函数的形参是类对象,调用函数进行形参和实参结合时;如果函数的返回是类对象,函数调用完成返回时; 4-7 拷贝构造函数与赋运算符(=)有何不同? 解: 赋运算符(=)作用于一个已存在的对象;而拷贝构造函数会创建一个新的对象。 4-8 定义一个Dog 类,包含的age、weight等属性,以及对这些属性操作的方法。实现并测试这个类。 解: 源程序: #include class Dog { public: Dog (int initialAge = 0, int initialWeight = 5); ~Dog(); int GetAge() { return itsAge;} // inline! void SetAge (int age) { itsAge = age;} // inline! int GetWeight() { return itsWeight;} // inline! void SetWeight (int weight) { itsAge = weight;} // inline! private: int itsAge, itsWeight; }; Dog::Dog(int initialAge, int initialWeight) { itsAge = initialAge; itsWeight = initialWeight; } Dog::~Dog() //destructor, takes no action { } int main() { Dog Jack(2,10); cout << "Jack is a Dog who is " ; cout << Jack.GetAge() << " years old and"; cout << Jack.GetWeight() << " pounds weight.\n"; Jack.SetAge(7); Jack.SetWeight(20); cout << "Now Jack is " ; cout << Jack.GetAge() << " years old and"; cout << Jack.GetWeight() << " pounds weight."; return 0; } 程序运行输出: Jack is a Dog who is 2 years old and 10 pounds weight. Now Jack is 7 years old 20 pounds weight. 4-9 设计并测试一个名为Rectangle的矩形类,其属性为矩形的左下角与右上角两个点的坐标,能计算矩形的面积。 解: 源程序: #include class Rectangle { public: Rectangle (int top, int left, int bottom, int right); ~Rectangle () {} int GetTop() const { return itsTop; } int GetLeft() const { return itsLeft; } int GetBottom() const { return itsBottom; } int GetRight() const { return itsRight; } void SetTop(int top) { itsTop = top; } void SetLeft (int left) { itsLeft = left; } void SetBottom (int bottom) { itsBottom = bottom; } void SetRight (int right) { itsRight = right; } int GetArea() const; private: int itsTop; int itsLeft; int itsBottom; int itsRight; }; Rectangle::Rectangle(int top, int left, int bottom, int right) { itsTop = top; itsLeft = left; itsBottom = bottom; itsRight = right; } int Rectangle::GetArea() const { int Width = itsRight-itsLeft; int Height = itsTop - itsBottom; return (Width * Height); } int main() { Rectangle MyRectangle (100, 20, 50, 80 ); int Area = MyRectangle.GetArea(); cout << "Area: " << Area << "\n"; return 0; } 程序运行输出: Area: 3000 Upper Left X Coordinate: 20 4-10 设计一个用于人事管理的People(人员)类。考虑到通用性,这里只抽象出所有类型人员都具有的属性:number(编号)、sex(性别)、birthday(出生日期)、id(身份证号)等等。其"出生日期"定义为一个"日期"类内嵌子对象。用成员函数实现对人员信息的录入和显示。要求包括:构造函数和析构函数、拷贝构造函数、内联成员函数、带缺省形参的成员函数、聚集。 解: 本题用作实验四的选做题,因此不给出答案。 4-11 定义一个矩形类,有长、宽两个属性,有成员函数计算矩形的面积 解: #include class Rectangle { public: Rectangle(float len, float width) { Length = len; Width = width; } ~Rectangle(){}; float GetArea() { return Length * Width; } float GetLength() { return Length; } float GetWidth() { return Width; } private: float Length; float Width; }; void main() { float length, width; cout <> length; cout <> width; Rectangle r(length, width); cout << "长为" << length << "宽为" << width << "的矩形的面积为:" << r.GetArea () << endl; } 程序运行输出: 请输入矩形的长度:5 请输入矩形的宽度:4 长为5宽为4的矩形的面积为:20 4-12 定义一个"数据类型" datatype类,能处理包含字符型、整型、浮点型三种类型的数据,给出其构造函数。 解: #include class datatype{ enum{ character, integer, floating_point } vartype; union { char c; int i; float f; }; public: datatype(char ch) { vartype = character; c = ch; } datatype(int ii) { vartype = integer; i = ii; } datatype(float ff) { vartype = floating_point; f = ff; } void print(); }; void datatype::print() { switch (vartype) { case character: cout << "字符型: " << c << endl; break; case integer: cout << "整型: " << i << endl; break; case floating_point: cout << "浮点型: " << f << endl; break; } } void main() { datatype A('c'), B(12), C(1.44F); A.print(); B.print(); C.print(); } 程序运行输出: 字符型: c 整型: 12 浮点型: 1.44 4-13 定义一个Circle类,有数据成员半径Radius,成员函数GetArea(),计算圆的面积,构造一个Circle的对象进行测试。 解: #include class Circle { public: Circle(float radius){ Radius = radius;} ~Circle(){} float GetArea() { return 3.14 * Radius * Radius; } private: float Radius; }; void main() { float radius; cout <> radius; Circle p(radius); cout << "半径为" << radius << "的圆的面积为:" << p.GetArea () << endl; } 程序运行输出: 请输入圆的半径:5 半径为5的圆的面积为:78.5 4-14 定义一个tree类,有成员ages,成员函数grow(int years)对ages加上years,age()显示tree对象的ages的。 解: #include class Tree { int ages; public: Tree(int n=0); ~Tree(); void grow(int years); void age(); }; Tree::Tree(int n) { ages = n; } Tree::~Tree() { age(); } void Tree::grow(int years) { ages += years; } void Tree::age() { cout << "这棵树的年龄为" << ages << endl; } void main() { Tree t(12); t.age(); t.grow(4); } 程序运行输出: 这棵树的年龄为12 这棵树的年龄为16 第 五 章 C++程序的基本结构 5-1 什么叫做作用域?有哪几种类型的作用域? 解: 作用域讨论的是标识符的有效范围,作用域是一个标识符在程序正文有效的区域。C++的作用域分为函数原形作用域、块作用域(局部作用域)、类作用域和文件作用域. 5-2 什么叫做可见性?可见性的一般规则是什么? 解: 可见性是标识符是否可以引用的问题; 可见性的一般规则是:标识符要声明在前,引用在后,在同一作用域,不能声明同名的标识符。对于在不同的作用域声明的标识符,遵循的原则是:若有两个或多个具有包含关系的作用域,外层声明的标识符如果在内层没有声明同名标识符时仍可见,如果内层声明了同名标识符则外层标识符不可见。 5-3 下面的程序的运行结果是什么,实际运行一下,看看与你的设想有何不同。 #include void myFunction(); int x = 5, y = 7; int main() { cout << "x from main: " << x << "\n"; cout << "y from main: " << y << "\n\n"; myFunction(); cout << "Back from myFunction!\n\n"; cout << "x from main: " << x << "\n"; cout << "y from main: " << y << "\n"; return 0; } void myFunction() { int y = 10; cout << "x from myFunction: " << x << "\n"; cout << "y from myFunction: " << y << "\n\n"; } 解: 程序运行输出: x from main: 5 y from main: 7 x from myFunction: 5 y from myFunction: 10 Back from myFunction! x from main: 5 y from main: 7 5-4 假设有两个无关系的类Engine和Fuel,使用时,怎样允许Fuel成员访问Engine的私有和保护的成员? 解: 源程序: class fuel; class engine { friend class fuel; private; int powerlevel; public; engine(){ powerLevel = 0;} void engine_fn(fuel &f); }; class fuel { friend class engine; private; int fuelLevel; public: fuel(){ fuelLevel = 0;} void fuel_fn( engine &e); }; 5-5 什么叫做静态数据成员?它有何特点? 解: 类的静态数据成员是类的数据成员的一种特例,采用static关键字来声明。对于类的普通数据成员,每一个类的对象都拥有一个拷贝,就是说每个对象的同名数据成员可以分别存储不同的数,这也是保证对象拥有自身区别于其它对象的特征的需要,但是静态数据成员,每个类只要一个拷贝,由所有该类的对象共同维护和使用,这个共同维护、使用也就实现了同一类的不同对象之间的数据共享。 5-6 什么叫做静态函数成员?它有何特点? 解: 使用static关键字声明的函数成员是静态的,静态函数成员属于整个类,同一个类的所有对象共同维护,为这些对象所共享。静态函数成员具有以下两个方面的好处,一是由于静态成员函数只能直接访问同一个类的静态数据成员,可以保证不会对该类的其余数据成员造成负面影响;二是同一个类只维护一个静态函数成员的拷贝,节约了系统的开销,提高程序的运行效率。 5-7 定义一个Cat类,拥有静态数据成员HowManyCats,记录Cat的个体数目;静态成员函数GetHowMany(),存取HowManyCats。设计程序测试这个类,体会静态数据成员和静态成员函数的用法。 解: 源程序: #include class Cat { public: Cat(int age):itsAge(age){HowManyCats++; } virtual ~Cat() { HowManyCats--; } virtual int GetAge() { return itsAge; } virtual void SetAge(int age) { itsAge = age; } static int GetHowMany() { return HowManyCats; } private: int itsAge; static int HowManyCats; }; int Cat::HowManyCats = 0; void TelepathicFunction(); int main() { const int MaxCats = 5; Cat *CatHouse[MaxCats]; int i; for (i = 0; i<MaxCats; i++) { CatHouse[i] = new Cat(i); TelepathicFunction(); } for ( i = 0; i<MaxCats; i++) { delete CatHouse[i]; TelepathicFunction(); } return 0; } void TelepathicFunction() { cout << "There are " << Cat::GetHowMany() << " cats alive!\n"; } 程序运行输出: There are 1 cats alive! There are 2 cats alive! There are 3 cats alive! There are 4 cats alive! There are 5 cats alive! There are 4 cats alive! There are 3 cats alive! There are 2 cats alive! There are 1 cats alive! There are 0 cats alive! 5-8 什么叫做友元函数?什么叫做友元类? 解: 友元函数是使用friend关键字声明的函数,它可以访问相应类的保护成员和私有成员。友元类是使用friend关键字声明的类,它的所有成员函数都是相应类的友元函数。 5-9 如果类A是类B的友元,类B是类C的友元,类D是类A的派生类,那么类B是类A的友元吗?类C是类A的友元吗?类D是类B的友元吗? 解: 类B不是类A的友元,友元关系不具有交换性; 类C不是类A的友元,友元关系不具有传递性; 类D不是类B的友元,友元关系不能被继承。 5-10 静态成员变量可以为私有的吗?声明一个私有的静态整型成员变量。 解: 可以,例如: private: static int a; 5-11 在一个文件定义一个全局变量n,主函数main(),在另一个文件定义函数fn1(),在main()对n赋,再调用fn1(),在fn1()也对n赋,显示n最后的。 解: #include #include "fn1.h" int n; void main() { n = 20; fn1(); cout << "n的为" <<n; } // fn1.h文件 extern int n; void fn1() { n=30; } 程序运行输出: n的为30 5-12 在函数fn1()定义一个静态变量n,fn1()对n的加1,在主函数,调用fn1()十次,显示n的。 解: #include void fn1() { static int n = 0; n++; cout << "n的为" << n <<endl; } void main() { for(int i = 0; i i =+10; } void Y::g(X* x) { x->i ++; } class Z { public: void f(X* x) { x->i += 5; } }; #endif // MY_X_Y_Z_H 程序运行输出:无 5-14 定义Boat与Car两个类,二者都有weight属性,定义二者的一个友元函数totalWeight(),计算二者的重量和。 解: 源程序: #include class Boat; class Car { private: int weight; public: Car(int j){weight = j;} friend int totalWeight(Car &aCar;, Boat &aBoat;); }; class Boat { private: int weight; public: Boat(int j){weight = j;} friend int totalWeight(Car &aCar;, Boat &aBoat;); }; int totalWeight(Car &aCar;, Boat &aBoat;) { return aCar.weight + aBoat.weight; } void main() { Car c1(4); Boat b1(5); cout << totalWeight(c1, b1) << endl; } 程序运行输出: 9 第 六 章 数组、指针与字符串 6-1 数组A[10][5][15]一共有多少个元素? 解: 10×5×15 = 750 个元素 6-2 在数组A[20]第一个元素和最后一个元素是哪一个? 解: 第一个元素是A[0],最后一个元素是A[19]。 6-3 用一条语句定义一个有五个元素的整型数组,并依次赋予1~5的初。 解: 源程序: int IntegerArray[5] = { 1, 2, 3, 4, 5 }; 或:int IntegerArray[] = { 1, 2, 3, 4, 5 }; 6-4 已知有一个数组名叫oneArray,用一条语句求出其元素的个数。 解: 源程序: nArrayLength = sizeof(oneArray) / sizeof(oneArray[0]); 6-5 用一条语句定义一个有5×3个元素的二维整型数组,并依次赋予1~15的初。 解: 源程序: int theArray[5][3] = { 1,2,3,4,5,6,7,8,9,10,11,12,13,14,15 }; 或:int theArray[5][3] = { {1,2,3}, {4,5,6}, {7,8,9}, {10,11,12},{13,14,15} }; 6-6 运算符*和&的作用是什么? 解: *称为指针运算符,是一个一元操作符,表示指针所指向的对象的;&称为取地址运算符,也是一个一元操作符,是用来得到一个对象的地址。 6-7 什么叫做指针?指针储存的地址和这个地址有何区别? 解: 指针是一种数据类型,具有指针类型的变量称为指针变量。指针变量存放的是另外一个对象的地址,这个地址就是另一个对象的内容。 6-8 定义一个整型指针,用new语句为其分配包含10个整型元素的地址空间。 解: 源程序: int *pInteger = new int[10]; 6-9 在字符串”Hello,world!”结束符是什么? 解: 是NULL字符。 6-10 定义一个有五个元素的整型数组,在程序提示用户输入元素,最后再在屏幕上显示出来。 解: 源程序: #include int main() { int myArray[5]; int i; for ( i=0; i<5; i++) { cout << "Value for myArray[" << i <> myArray[i]; } for (i = 0; i<5; i++) cout << i << ": " << myArray[i] << "\n"; return 0; } 程序运行输出: Value for myArray[0]: 2 Value for myArray[1]: 5 Value for myArray[2]: 7 Value for myArray[3]: 8 Value for myArray[4]: 3 0: 2 1: 5 2: 7 3: 8 4: 3 6-11 引用和指针有何区别?何时只能使用指针而不能使用引用? 解: 引用是一个别名,不能为NULL,不能被重新分配;指针是一个存放地址的变量。当需要对变量重新赋以另外的地址或赋为NULL时只能使用指针。 6-12 声明下列指针:float类型变量的指针pFloat,char类型的指针pString和struct customer型的指针prec。 解: float *pfloat; char *pString; struct customer *prec; 6-13 给定float类型的指针fp,写出显示fp所指向的的输出流语句。 解: cout << "Value == " << *fp; 6-14 程序定义一个double类型变量的指针。分别显示指针占了多少字节和指针所指的变量占了多少字节。 解: double *counter; cout << "\nSize of pointer == "sizeof(counter); cout << '\nSize of addressed value == "<<sizeof(*counter); 6-15 const int * p1 和 int * const p2的区别是什么? 解: const int * p1 声明了一个指向整型常量的指针p1,因此不能通过指针p1来改变它所指向的整型;int * const p2声明了一个指针型常量,用于存放整型变量的地址,这个指针一旦初始化后,就不能被重新赋了。 6-16 定义一个整型变量a,一个整型指针p,一个引用r,通过p把a的改为10,通过r把a的改为5 解: void main() { int a; int *p = &a; int &r = a; *p = 10; r = 5; } 6-17 下列程序有何问题,请仔细体会使用指针时应避免出现这个的问题。 #include int main() { int *p; *pInt = 9; cout << "The value at p: " << *p; return 0; } 解: 指针p没有初始化,也就是没有指向某个确定的内存单元,它指向内存的一个随机地址,给这个随机地址赋是非常危险的。 6-18 下列程序有何问题,请改正;仔细体会使用指针时应避免出现的这个问题。 #include int Fn1(); int main() { int a = Fn1(); cout << "the value of a is: " << a; return 0; } int Fn1() { int * p = new int (5); return *p; } 解: 此程序给*p分配的内存没有被释放掉。 改正: #include int* Fn1(); int main() { int *a = Fn1(); cout << "the value of a is: " << *a; delete a; return 0; } int* Fn1() { int * p = new int (5); return p; } 6-19 声明一个参数为整型,返回为长整型的函数指针;声明类A的一个成员函数指针,其参数为整型,返回长整型。 解: long (* p_fn1)(int); long ( A::*p_fn2)(int); 6-20 实现一个名为SimpleCircle的简单圆类,其数据成员int *itsRadius为一个指向其半径的指针,设计对数据成员的各种操作,给出这个类的完整实现并测试这个类。 解: 源程序: #include class SimpleCircle { public: SimpleCircle(); SimpleCircle(int); SimpleCircle(const SimpleCircle &); ~SimpleCircle() {} void SetRadius(int); int GetRadius()const; private: int *itsRadius; }; SimpleCircle::SimpleCircle() { itsRadius = new int(5); } SimpleCircle::SimpleCircle(int radius) { itsRadius = new int(radius); } SimpleCircle::SimpleCircle(const SimpleCircle & rhs) { int val = rhs.GetRadius(); itsRadius = new int(val); } int SimpleCircle::GetRadius() const { return *itsRadius; } int main() { SimpleCircle CircleOne, CircleTwo(9); cout << "CircleOne: " << CircleOne.GetRadius() << endl; cout << "CircleTwo: " << CircleTwo.GetRadius() << endl; return 0; }程序运行输出: CircleOne: 5 CircleTwo: 9 6-21 编写一个函数,统计一个英文句子字母的个数,在主程序实现输入、输出。 解: 源程序: #include #include int count(char *str) { int i,num=0; for (i=0; str[i]; i++) { if ( (str[i]>='a' && str[i]='A' && str[i]<='Z') ) num++; } return num; } void main() { char text[100]; cout << "输入一个英语句子:" << endl; gets(text); cout << "这个句子里有" << count(text) << "个字母。" << endl; } 程序运行输出: 输入一个英语句子: It is very interesting! 这个句子里有19个字母。 6-22 编写函数int index(char *s, char *t),返回字符串t 在字符串s出现的最左边的位置,如果在s没有与t匹配的子串,就返回-1。 解: 源程序: #include int index( char *s, char *t) { int i,j,k; for(i = 0; s[i] != '\0'; i++) { for(j = i, k = 0; t[k] != '\0' && s[j] == t[k]; j++, k++) ; if (t[k] =='\0') return i; } return -1; } void main() { int n; char str1[20],str2[20]; cout <> str1; cout <> str2; n = index(str1,str2); if (n > 0) cout << str2 << "在" << str1 << "左起第" << n+1 << "个位置。"<<endl; else cout << str2 << "不在" << str1 << "。" << endl; } 程序运行输出: 输入一个英语单词:abcdefgh 输入另一个英语单词:de de在abcdefghijk左起第4个位置。 6-23 编写函数reverse(char *s)的倒序递归程序,使字符串s倒序。 解: 源程序: #include #include void reverse(char *s, char *t) { char c; if (s < t) { c = *s; *s = *t; *t = c; reverse(++s, --t); } } void reverse( char *s) { reverse(s, s + strlen(s) - 1); } void main() { char str1[20]; cout <> str1; cout << "原字符串为:" << str1 << endl; reverse(str1); cout << "倒序反转后为:" << str1 << endl; } 程序运行输出: 输入一个字符串:abcdefghijk 原字符串为:abcdefghijk 倒序反转后为:kjihgfedcba 6-24 设学生人数N=8,提示用户输入N个人的考试成绩,然后计算出平均成绩,显示出来。 解: 源程序: #include #include #define N 8 float grades[N]; //存放成绩的数组 void main() { int i; float total,average; //提示输入成绩 for(i = 0; i < N; i++ ) { cout << "Enter grade #" <<(i +1) <> grades[i]; } total = 0; for (i = 0; i < N; i++) total += grades[i]; average = total / N; cout << "\nAverage grade: " << average << endl; } 程序运行输出: Enter grade #1: 86 Enter grade #2: 98 Enter grade #3: 67 Enter grade #4: 80 Enter grade #5: 78 Enter grade #6: 95 Enter grade #7: 78 Enter grade #8: 56 Average grade: 79.75 6-25 设计一个字符串类MyString,具有构造函数、析构函数、拷贝构造函数,重载运算符+、=、+=、[],尽可能地完善它,使之能满足各种需要。(运算符重载功能为选做,参见第8章) 解: #include #include class MyString { public: MyString(); MyString(const char *const); MyString(const MyString &); ~MyString(); char & operator[](unsigned short offset); char operator[](unsigned short offset) const; MyString operator+(const MyString&); void operator+=(const MyString&); MyString & operator= (const MyString &); unsigned short GetLen()const { return itsLen; } const char * GetMyString() const { return itsMyString; } private: MyString (unsigned short); // private constructor char * itsMyString; unsigned short itsLen; }; MyString::MyString() { itsMyString = new char[1]; itsMyString[0] = '\0'; itsLen=0; } MyString::MyString(unsigned short len) { itsMyString = new char[len+1]; for (unsigned short i = 0; i<=len; i++) itsMyString[i] = '\0'; itsLen=len; } MyString::MyString(const char * const cMyString) { itsLen = strlen(cMyString); itsMyString = new char[itsLen+1]; for (unsigned short i = 0; i<itsLen; i++) itsMyString[i] = cMyString[i]; itsMyString[itsLen]='\0'; } MyString::MyString (const MyString & rhs) { itsLen=rhs.GetLen(); itsMyString = new char[itsLen+1]; for (unsigned short i = 0; i<itsLen;i++) itsMyString[i] = rhs[i]; itsMyString[itsLen] = '\0'; } MyString::~MyString () { delete [] itsMyString; itsLen = 0; } MyString& MyString::operator=(const MyString & rhs) { if (this == &rhs;) return *this; delete [] itsMyString; itsLen=rhs.GetLen(); itsMyString = new char[itsLen+1]; for (unsigned short i = 0; i itsLen) return itsMyString[itsLen-1]; else return itsMyString[offset]; } char MyString::operator[](unsigned short offset) const { if (offset > itsLen) return itsMyString[itsLen-1]; else return itsMyString[offset]; } MyString MyString::operator+(const MyString& rhs) { unsigned short totalLen = itsLen + rhs.GetLen(); MyString temp(totalLen); for (unsigned short i = 0; i<itsLen; i++) temp[i] = itsMyString[i]; for (unsigned short j = 0; j<rhs.GetLen(); j++, i++) temp[i] = rhs[j]; temp[totalLen]='\0'; return temp; } void MyString::operator+=(const MyString& rhs) { unsigned short rhsLen = rhs.GetLen(); unsigned short totalLen = itsLen + rhsLen; MyString temp(totalLen); for (unsigned short i = 0; i<itsLen; i++) temp[i] = itsMyString[i]; for (unsigned short j = 0; j<rhs.GetLen(); j++, i++) temp[i] = rhs[i-itsLen]; temp[totalLen]='\0'; *this = temp; } int main() { MyString s1("initial test"); cout << "S1:\t" << s1.GetMyString() << endl; char * temp = "Hello World"; s1 = temp; cout << "S1:\t" << s1.GetMyString() << endl; char tempTwo[20]; strcpy(tempTwo,"; nice to be here!"); s1 += tempTwo; cout << "tempTwo:\t" << tempTwo << endl; cout << "S1:\t" << s1.GetMyString() << endl; cout << "S1[4]:\t" << s1[4] << endl; s1[4]='x'; cout << "S1:\t" << s1.GetMyString() << endl; cout << "S1[999]:\t" << s1[999] << endl; MyString s2(" Another myString"); MyString s3; s3 = s1+s2; cout << "S3:\t" << s3.GetMyString() << endl; MyString s4; s4 = "Why does this work?"; cout << "S4:\t" << s4.GetMyString() << endl; return 0; } 程序运行输出: S1: initial test S1: Hello World tempTwo: ; nice to be here! S1: Hello World; nice to be here! S1[4]: o S1: Hellx World; nice to be here! S1[999]: ! S3: Hellx World; nice to be here! Another myString S4: Why does this work? 6-26 编写一个3×3矩阵转置的函数,在main()函数输入数据。 解: #include void move (int matrix[3][3]) { int i, j, k; for(i=0; i<3; i++) for (j=0; j<i; j++) { k = matrix[i][j]; matrix[i][j] = matrix[j][i]; matrix[j][i] = k; } } void main() { int i, j; int data[3][3]; cout << "输入矩阵的元素" << endl; for(i=0; i<3; i++) for (j=0; j<3; j++) { cout << "第" << i+1 << "行第" << j+1 <> data[i][j]; } cout << "输入的矩阵的为:" << endl; for(i=0; i<3; i++) { for (j=0; j<3; j++) cout << data[i][j] << " "; cout << endl; } move(data); cout << "转置后的矩阵的为:" << endl; for(i=0; i<3; i++) { for (j=0; j<3; j++) cout << data[i][j] << " "; cout << endl; } } 程序运行输出: 输入矩阵的元素 第1行第1个元素为:1 第1行第2个元素为:2 第1行第3个元素为:3 第2行第1个元素为:4 第2行第2个元素为:5 第2行第3个元素为:6 第3行第1个元素为:7 第3行第2个元素为:8 第3行第3个元素为:9 输入的矩阵的为: 1 2 3 4 5 6 7 8 9 转置后的矩阵的为: 1 4 7 2 5 8 3 6 9 6-27 编写一个矩阵转置的函数,矩阵的维数在程序由用户输入。 解: #include void move (int *matrix ,int n) { int i, j, k; for(i=0; i<n; i++) for (j=0; j<i; j++) { k = *(matrix + i*n + j); *(matrix + i*n + j) = *(matrix + j*n + i); *(matrix + j*n + i) = k; } } void main() { int n, i, j; int *p; cout <> n; p = new int[n*n]; cout << "输入矩阵的元素" << endl; for(i=0; i<n; i++) for (j=0; j<n; j++) { cout << "第" << i+1 << "行第" << j+1 <> p[i*n + j]; } cout << "输入的矩阵的为:" << endl; for(i=0; i<n; i++) { for (j=0; j<n; j++) cout << p[i*n + j] << " "; cout << endl; } move(p, n); cout << "转置后的矩阵的为:" << endl; for(i=0; i<n; i++) { for (j=0; j<n; j++) cout << p[i*n + j] << " "; cout << endl; } } 程序运行输出: 请输入矩阵的维数:3 输入矩阵的元素 第1行第1个元素为:1 第1行第2个元素为:2 第1行第3个元素为:3 第2行第1个元素为:4 第2行第2个元素为:5 第2行第3个元素为:6 第3行第1个元素为:7 第3行第2个元素为:8 第3行第3个元素为:9 输入的矩阵的为: 1 2 3 4 5 6 7 8 9 转置后的矩阵的为: 1 4 7 2 5 8 3 6 9 6-28 定义一个Employee类,其包括表示姓名、街道地址、城市和邮编等属性,包括chage_name()和display()等函数;display()使用cout语句显示姓名、街道地址、城市和邮编等属性,函数change_name()改变对象的姓名属性,实现并测试这个类。 解: 源程序: #include #include class Employee { private: char name[30]; char street[30]; char city[18]; char zip[6]; public: Employee(char *n, char *str, char *ct, char *z); void change_name(char *n); void display(); }; Employee::Employee (char *n,char *str,char *ct, char *z) { strcpy(name, n); strcpy(street, str); strcpy(city, ct); strcpy(zip, z); } void Employee::change_name (char *n) { strcpy(name, n); } void Employee::display () { cout << name << " " << street << " "; cout << city << " "<< zip; } void main(void) { Employee e1("张三","平安大街3号", "北京", "100000"); e1.display(); cout << endl; e1.change_name("李四"); e1.display(); cout << endl; } 程序运行输出: 张三 平安大街3号 北京 100000 李四 平安大街3号 北京 100000 第 七 章 继承与派生 7-1 比较类的三种继承方式public公有继承、protected保护继承、private私有继承之间的差别。 解: 不同的继承方式,导致不同访问属性的基类成员在派生类的访问属性也有所不同: 公有继承,使得基类public(公有)和protected(保护)成员的访问属性在派生类不变,而基类private(私有)成员不可访问。 私有继承,使得基类public(公有)和protected(保护)成员都以private(私有)成员身份出现在派生类,而基类private(私有)成员不可访问。 保护继承,基类public(公有)和protected(保护)成员都以protected(保护)成员身份出现在派生类,而基类private(私有)成员不可访问。 7-2 派生类构造函数执行的次序是怎样的? 解: 派生类构造函数执行的一般次序为:调用基类构造函数;调用成员对象的构造函数;派生类的构造函数体的内容。 7-3 如果在派生类B已经重载了基类A的一个成员函数fn1(),没有重载成员函数fn2(),如何调用基类的成员函数fn1()、fn2()? 解: 调用方法为: A::fn1(); fn2(); 7-4 什么叫做虚基类?有何作用? 解: 当某类的部分或全部直接基类是从另一个基类派生而来,这些直接基类,从上一级基类继承来的成员就拥有相同的名称,派生类的对象的这些同名成员在内存同时拥有多个拷贝,我们可以使用作用域分辨符来唯一标识并分别访问它们。我们也可以将直接基类的共同基类设置为虚基类,这时从不同的路径继承过来的该类成员在内存只拥有一个拷贝,这样就解决了同名成员的唯一标识问题。 虚基类的声明是在派生类的定义过程,其语法格式为: class 派生类名:virtual 继承方式 基类名 上述语句声明基类为派生类的虚基类,在多继承情况下,虚基类关键字的作用范围和继承方式关键字相同,只对紧跟其后的基类起作用。声明了虚基类之后,虚基类的成员在进一步派生过程,和派生类一起维护一个内存数据拷贝。 7-5 定义一个Shape基类,在此基础上派生出Rectangle和Circle,二者都有GetArea()函数计算对象的面积。使用Rectangle类创建一个派生类Square。 解: 源程序: #include class Shape { public: Shape(){} ~Shape(){} virtual float GetArea() { return -1; } }; class Circle : public Shape { public: Circle(float radius):itsRadius(radius){} ~Circle(){} float GetArea() { return 3.14 * itsRadius * itsRadius; } private: float itsRadius; }; class Rectangle : public Shape { public: Rectangle(float len, float width): itsLength(len), itsWidth(width){}; ~Rectangle(){}; virtual float GetArea() { return itsLength * itsWidth; } virtual float GetLength() { return itsLength; } virtual float GetWidth() { return itsWidth; } private: float itsWidth; float itsLength; }; class Square : public Rectangle { public: Square(float len); ~Square(){} }; Square::Square(float len): Rectangle(len,len) { } void main() { Shape * sp; sp = new Circle(5); cout << "The area of the Circle is " <GetArea () << endl; delete sp; sp = new Rectangle(4,6); cout << "The area of the Recta

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值