C语言程序设计:现代方法(第2版)第三章全部习题答案
前言
本人在通过《C语言程序设计:现代方法(第2版)》自学C语言时,发现国内并没有该书完整的课后习题答案,所以就想把自己在学习过程中所做出的答案分享出来,以供大家参考。这些答案是本人自己解答,并参考GitHub上相关的分享和Chegg.com相关资料。因为并没有权威的答案来源,所以可能会存在错误的地方,如有错误还希望大家能够帮助指出。
第三章练习题和编程题答案
练习题
3.1节
1.下面的printf函数调用产生的输出分别是什么?
(a) printf("6d,%4d", 86, 1040);
(b) printf("%12.5e", 30.253);
(c) printf("%.4f", 83.162);
(d) printf("%-6.2g", .0000009979);
答:
(a) ┗━┛┗━┛┗━┛┗━┛86, 1040 “%6d”表示右对齐、最小6个字段宽,数字86只占了2个字段宽,所以前面用4个空格(┗━┛)补齐。同理“%4d”也是右对齐、最小4个字段宽,数字1040正好为4个字段宽,所有没有空格。
(b) 3.02530e+001 “%12.5e”表示右对齐、最小12个字段宽,同时小数点后显示5位数字,因为.0253不足5位,所以小数部分末尾会添加零占位。
(c) 83.1620 “%.4f”表示小数点后显示4位,因为.162不足4位,所以小数部分末尾会添加零占位。
(d) 1e-006 “%-6.2g”表示左对齐、最小6个字段宽,.2表示可以显示最大2个(去掉小数点后)有效数字。之所以显示1e是因为数字9979只能显示99这两个有效数字,所有后面的7会四舍五入向前进位,这样实际要显示的 数字就从.0000009979变成了.0000010000,所以结果显示为1e,如果将.2变为.3的话就会显示为9.98e-007。在%f和%e当中,如果需要舍弃小数点后部分数字,同样会使用这种四舍五入的显示方式。
2.编写printf函数调用,以下列格式显示float型变量x:
(a) 指数表示形式,字段宽度8,左对齐,小数点后保留1位数字。
(b) 指数表示形式,字段宽度10,右对齐,小数点后保留6位数字。
(c) 定点十进制表示形式,字段宽度8,左对齐,小数点后保留3位数字。
(d) 定点十进制表示形式,字段宽度6,右对齐,小数点后无数字。
答:
(a) printf("%-8.1e", x);
(b) printf("%10.6e", x);
(c) printf("%-8.3f", x);
(d) printf("6.0f", x);
3.2节
3.说明下列每对scanf格式串是否等价?如果不等价,请指出它们的差异。
(a) "%d"与" %d"
(b)"%d-%d-%d"与"%d -%d -%d"
(c)"f"与"%f "
(d)"%f,%f"与"%f, %f"
答:
(a) 等价,因为scanf在开始寻找数的时候会忽略掉空格,所以即使在前面添加任意个空格均不影响结果。
(b)不等价,虽然在开始是是scanf会忽略掉空格,但是开始读入之后并不会忽略。所以前者%d-%d-%d中不能在“-”之间加入任何空格,否则会异常退出,而后者%d -%d -%d则可以在“-”之间加入或不加入任意个空格
(c)不等价,第二个格式串中以空格作为结尾,虽然不影响录入%f中的数字,但是第二个格式串并不会在回车或空格后结束,它会一直等到读入第一个非空字符时才会结束。而第一个%f会在遇到空格或回车之类的符号后就结束运行。
(d)不等价,与(b)理由相同,第一个格式串中逗号直接不能加入任何空格,而第二个则可以加入0个或n个空格。
4.假设scanf函数调用的格式如下:
scanf("%d%f%d", &i, &x, &j);
如果用户录入
10.3 5 6
调用执行后,变量i、x和j的值分别是多少?(假设变量i和变量j都是int型,而变量x是float型。)
答:
最后的结果是i=10,x=0.3,j=5,
首先因为i是int型所以scanf会读入“10.”,但因为后面出现了小数点(“.”),而i中不能有小数点,所以只将10存入一种并将小数点“放回”去。
之后x是float型,但读入.3后因为遇到了非数字空格,所以scanf会将.3存入x中,而将空格“放回”。
最后的变量j在读取时会像跳过空白字符,直到遇到非空白字符5,而5之后又是非数字的空格,所以只会将5存入变量j中。
而因为此时这个scanf函数结束了,所以最后的数字6会被保留到下一次scanf函数调用时再读取。
5.假设scanf函数调用的是格式如下:
scanf("%f%d%f", &x, &i, &y);
如果用户输入
12.3 45.6 789
调用执行后,变量x、i和y的值分别是多少?(假设变量x和变量y都是float型,变量i是int型。)
答:
最后的结果是x=12.3,i=45,y=0.6
与上一道题类似,首先因为x是float型,所以scanf会读入“12.3”,之后遇到空格,此时结束读入并将12.3存入x中。
之后因为i是int型不能有小数点,所以scanf在读入“45.6”时只会将45存入i中,并将“.6”放回。
y是float型可以有小数点,所以scanf会将上次没有读入的“.6”存入y中,并在遇到空格时结束这次读入。
因为此时scanf函数已经执行完毕,所以最后的789会在下次scanf执行时才被读入。
6.指出如何修改3.2节中的addfrac.c程序,使用户可以输入在字符/的前后都有空格的分数
答:
1 /*只需要在原程序中的/前后各加入一个空格即可 */ 2 /* Adds two fractions */ 3 4 #include <stdio.h> 5 6 int main(void) 7 { 8 int num1, denom1, num2, denom2, result_num, result_denom; 9 10 printf("Enter first fraction: "); 11 scanf("%d / %d", &num1, &denom1); /*至于要在原程序中的/前后各加入一个空格即可 */ 12 13 printf("Enter second fraction: "); 14 scanf("%d / %d", &num2, &denom2); 15 16 result_num = num1 * denom2 + num2 * denom1; 17 result_denom = denom1 * denom2; 18 printf("The sum is %d/%d\n", result_num, result_denom); 19 20 return 0; 21 }
编程题
1.编写一个程序,以月/日/年(即mm/dd/yy)的格式接受用户录入的日期信息,并以年月日(即yy/mm/dd)的格式将其显示出来:
Enter a date (mm/dd/yyyy): 2/17/2011
You entered the date 20110217
答:
1 /*月日年转成年月日*/ 2 #include <stdio.h> 3 4 int main(void) 5 { 6 int day = 0, month = 0, year = 0; 7 8 printf("Enter a date (mm/dd/yyyy): "); 9 scanf("%d/%d/%d", &month, &day, &year); 10 11 printf("You entered the date %d%.2d%.2d", year, month, day); /*月和日用%.2d显示,保证月份和日期会以两位数形式显示 */ 12 13 return 0; 14 }
2.编程一个程序,对于用户录入的产品信息进行格式化,程序会话应类似下面这样:
Enter item number: 583
Enter unit price: 13.5
Enter purchase date (mm/dd/yyyy): 10/24/2010
Item Unit Purchase
price Date
583 $ 13.50 10/24/2010
其中,产品编号和日期项采用左对齐方式,单位价格采用右对齐方式,允许最大取值为9999.99的美元。提示:各个列使用制表符控制。
答:
1 /*输入信息并显示,产品编号和日期左对齐,价格右对齐*/ 2 3 #include <stdio.h> 4 5 int main(void) 6 { 7 /* 声明item number变量item_num,unit price变量price,日期变量day、month和year */ 8 int item_num = 0, day = 0, month = 0, year = 0; 9 /* unit price包含小数点,所以其变量应为float型 */ 10 float price = 1.0f; 11 12 /* 用户输入数据,程序录入对应变量中 */ 13 printf("Enter item number: "); 14 scanf("%d", &item_num); 15 printf("Enter unit price: "); 16 scanf("%f", &price); 17 printf("Enter purchase date (mm/dd/yyyy): "); 18 scanf("%d/%d/%d", &month, &day, &year); 19 20 printf("Item\t\tUnit\t\tPurchase\n"); /* 第一行显示,中间通过两个制表符\t分隔 */ 21 printf("\t\tprice\t\tDate\n"); /* 第二行显示,中间通过两个制表符\t分隔 */ 22 /* 编号和日期是左对齐,所以要在前面加上负号。价格采用%7.2f显示,是因为最大价格是9999.99共七位,在小于七位时用空格占位,.2则是保证小数点后只显示2位数字。 */ 23 /* 日期中的.2是为了让最少以两位数形式显示,年份为四位数,所以采用.4保证最少显示四位数 */ 24 printf("%-d\t\t$%7.2f\t%-.2d/%-.2d/%-.4d\n", item_num, price, month, day, year); 25 26 return 0; 27 }
3.图书用国际标准书号(ISBN)进行标识。2007年1月1日之后分配的ISBN包含13位数字(旧的ISBN使用10位数字),分为5组,如978-0-393-97950-3.第一组(GS1)目前为978或979.第二组(组标识)指明语言或者原出版社国(如0和1用于英语的国家)。第三组(出版商编号)便是出版商(393是W.W.Norton出版社的编号)。第四组(产品编号)是由出版商分配的用于识别具体哪一本书的(97950)。ISBN的末尾是一个校验数字,用于验证前面数字的准确性。编写一个程序来分解用户录入的ISBN信息:
Enter ISBN: 978-0-393-97950-3
GS1 prefix: 978
Group identifier: 0
Publish code: 393
Item number: 97950
Check digit: 3
注意:每组中数字的个数是可变的,不能认为每组的长度都与示例一样。用实际的ISBN值(通常放在书的封底和版权页上)测试你编写的程序。
答:
1 /*ISBN分解*/ 2 3 #include <stdio.h> 4 5 int main(void) 6 { 7 /* 第一组数字的变量定义为gs1,第二组数字的变量定义为group_id,第三组数字的变量定义为pub_code,第四组数字的变量定义为item_num,校验位定义为check_d */ 8 int gs1 = 0, group_id = 0, pub_code = 0, item_num = 0, check_d = 0; 9 10 printf("Enter ISBN: "); 11 /* 每组数字看成一个整体,通过-来识别每组数字,并将其存入对应的变量中,这样每组数字个数即使发生变化也能正确的显示出来 */ 12 scanf("%d-%d-%d-%d-%d", &gs1, &group_id, &pub_code, &item_num, &check_d); 13 14 /* 直接将每组变量分别显示出来即可 */ 15 printf("GS1 prefix: %d\nGroup identifier: %d\nPublish code: %d\nItem number: %d\nCheck digit: %d\n", gs1, group_id, pub_code, item_num, check_d); 16 17 return 0; 18 }
4.编写一个程序,提示用户以[(xxx)xxx-xxx]的格式输入电话号码,并以xxx.xxx.xxxx的格式显示该号码:
Enter phone number [(xxx) xxx-xxxx]: (404) 817-6900
You entered: 404.817.6900
答:
1 /*输入电话号码(xxx) xxx-xxxx并转换格式显示xxx.xxx.xxxx */ 2 3 #include <stdio.h> 4 5 int main(void) 6 { 7 /* 将电话号码拆分成3组,分别定义成num1、num2和num3 */ 8 int num1 = 0, num2 = 0, num3 = 0; 9 10 printf("Enter phone number [(xxx) xxx-xxxx]: "); 11 /* 根据提示的格式分别将3组数字存入变量中 */ 12 scanf("(%d) %d-%d", &num1, &num2, &num3); 13 14 printf("You entered: %d.%d.%d\n", num1, num2, num3); 15 16 return 0; 17 }
5.编写一个程序,要求用户(按任意次序)输入从1到16的所有整数,然后用4×4矩阵的形式将它们显示出来,再计算每行、每列和每条对角线上的和:
Enter the numbers from 1 to 16 in any order: 16 3 2 13 5 10 11 8 9 6 7 12 4 15 14 1
16 3 2 13
5 10 11 8
9 6 7 12
4 15 14 1
Row sums: 34 34 34 34
Column sums: 34 34 34 34
Diagonal sums: 34 34
如果行、列和对角线上的和都一样(如本例所示),则称这些数组成一个幻方(magic square)。这里给出的幻方出现于艺术家和数学家Albrecht Dürer在1514年的一幅画中。(注意,矩阵的最后一行中间的两个数给出了该画的创作年代。)
答:
1 /*输入1到16所有整数,以4*4显示,并计算每行、每列、和对角线的和 */ 2 3 #include <stdio.h> 4 5 int main(void) 6 { 7 /* 定义十六个数字的变量,从num1到num16 */ 8 int num1 = 0, num2 = 0, num3 = 0, num4 = 0, num5 = 0, num6 = 0, num7 = 0, num8 = 0; 9 /* 拆成两行定义是因为写在一行里实在是太长了,不方便阅读 */ 10 int num9 = 0, num10 = 0, num11 = 0, num12 = 0, num13 = 0, num14 = 0, num15 = 0, num16 = 0; 11 /* 定义4×4矩阵的每行和的变量,从row1到row4 */ 12 int row1 = 0, row2 = 0, row3 = 0, row4 = 0; 13 /* 定义4×4矩阵的每列和的变量,从col1到col4 */ 14 int col1 = 0, col2 = 0, col3 = 0, col4 = 0; 15 /* 定义两个对角线和的变量dia1和dia2 */ 16 int dia1 = 0, dia2 = 0; 17 18 printf("Enter the numbers from 1 to 16 in any order: "); 19 /* 将十六个数字分别存入到16个变量里 */ 20 scanf("%d%d%d%d%d%d%d%d", &num1, &num2, &num3, &num4, &num5, &num6, &num7, &num8); 21 /* 同样因为太长了,所以拆成两行写 */ 22 scanf("%d%d%d%d%d%d%d%d", &num9, &num10, &num11, &num12, &num13, &num14, &num15, &num16); 23 24 /* 显示出矩阵,根据题目示例,每个数字应该保证两个字符宽,右对齐。通过制表符来使每行和每列对齐 */ 25 printf("%2d\t%2d\t%2d\t%2d\t\n%2d\t%2d\t%2d\t%2d\t\n", num1, num2, num3, num4, num5, num6, num7, num8); 26 printf("%2d\t%2d\t%2d\t%2d\t\n%2d\t%2d\t%2d\t%2d\t\n\n\n", num9, num10, num11, num12, num13, num14, num15, num16); 27 28 /* 计算每行的和 */ 29 row1 = num1 + num2 + num3 + num4; 30 row2 = num5 + num6 + num7 + num8; 31 row3 = num9 + num10 + num11 + num12; 32 row4 = num13 + num14 + num15 + num16; 33 34 /* 计算每列的和 */ 35 col1 = num1 + num5 + num9 + num13; 36 col2 = num2 + num6 + num10 + num14; 37 col3 = num3 + num7 + num11 + num15; 38 col4 = num4 + num8 + num12 + num16; 39 40 /* 计算两个对角线的和 */ 41 dia1 = num1 + num6 + num11 + num16; 42 dia2 = num4 + num7 + num10 + num13; 43 44 /* 显示结果 */ 45 printf("Row sums: %d %d %d %d \nColumn sums: %d %d %d %d \n", row1, row2, row3, row4, col1, col2, col3, col4); 46 printf("Diagonal sums: %d %d \n", dia1, dia2); 47 48 return 0; 49 }
6.修改3.2节的addfrac.c程序,使用户可以同时输入两个分数,中间用加好隔开:
Enter two fractions separated by a plus sign: 5/6+3/4
The sum is: 38/24
答:
1 /*同时输入两个分数并计算*/ 2 3 #include <stdio.h> 4 5 int main(void) 6 { 7 /* 定义两个分数的变量,分子分别是x1和x2,分母是y1和y2 */ 8 int x1 = 0, y1 = 0, x2 = 0, y2 = 0; 9 /* 计算出的分子用result_num变量表示,分母用result_denom变量表示 */ 10 int result_num = 0, result_denom = 0; 11 12 printf("Enter two fractions separated by a plus sign: "); 13 /* 读取两个分数 */ 14 scanf("%d/%d+%d/%d", &x1, &y1, &x2, &y2); 15 16 /* 计算两个分数的和 */ 17 result_num = x1 * y2 + x2 * y1; 18 result_denom = y1 * y2; 19 printf("The sum is: %d/%d\n", result_num, result_denom); 20 21 return 0; 22 }