练习14:编写并使用函数

原文:Exercise 14: Writing And Using Functions

译者:飞龙

到现在为止,你只使用了作为stdio.h头文件一部分的函数。在这个练习中你将要编写并使用自己的函数。

  1. #include <stdio.h>
  2. #include <ctype.h>
  3. // forward declarations
  4. int can_print_it(char ch);
  5. void print_letters(char arg[]);
  6. void print_arguments(int argc, char *argv[])
  7. {
  8. int i = 0;
  9. for(i = 0; i < argc; i++) {
  10. print_letters(argv[i]);
  11. }
  12. }
  13. void print_letters(char arg[])
  14. {
  15. int i = 0;
  16. for(i = 0; arg[i] != '\0'; i++) {
  17. char ch = arg[i];
  18. if(can_print_it(ch)) {
  19. printf("'%c' == %d ", ch, ch);
  20. }
  21. }
  22. printf("\n");
  23. }
  24. int can_print_it(char ch)
  25. {
  26. return isalpha(ch) || isblank(ch);
  27. }
  28. int main(int argc, char *argv[])
  29. {
  30. print_arguments(argc, argv);
  31. return 0;
  32. }

在这个例子中你创建了函数来打印任何属于“字母”和“空白”的字符。下面是一个分解:

ex14.c:2

包含了新的头文件,所以你可以访问isalphaisblank

ex14.c:5-6

告诉C语言你稍后会在你的程序中使用一些函数,它们实际上并没有被定义。这叫做“前向声明”,它解决了要想使用函数先要定义的鸡和蛋的问题。

ex14.c:8-15

定义print_arguments,它知道如何打印通常由main函数获得的相同字符串数组。

ex14.c:17-30

定义了can_print_it,它只是简单地将isalpha(ch) || isblank(ch)的真值(0或1)返回给它的调用者print_letters

ex14.c:38-42

最后main函数简单地调用print_arguments,来启动整个函数链。

我不应该描述每个函数里都有什么,因为这些都是你之前遇到过的东西。你应该看到的是,我只是像你定义main函数一样来定义其它函数。唯一的不同就是如果你打算使用当前文件中没有碰到过的函数,你应该事先告诉C。这就是代码顶部的“前向声明”的作用。

你会看到什么

向这个程序传入不同的命令行参数来玩转它,这样会遍历你函数中的所有路径。这里演示了我和它的交互:

  1. $ make ex14
  2. cc -Wall -g ex14.c -o ex14
  3. $ ./ex14
  4. 'e' == 101 'x' == 120
  5. $ ./ex14 hi this is cool
  6. 'e' == 101 'x' == 120
  7. 'h' == 104 'i' == 105
  8. 't' == 116 'h' == 104 'i' == 105 's' == 115
  9. 'i' == 105 's' == 115
  10. 'c' == 99 'o' == 111 'o' == 111 'l' == 108
  11. $ ./ex14 "I go 3 spaces"
  12. 'e' == 101 'x' == 120
  13. 'I' == 73 ' ' == 32 'g' == 103 'o' == 111 ' ' == 32 ' ' == 32 's' == 115 'p' == 112 'a' == 97 'c' == 99 'e' == 101 's' == 115
  14. $

isalphaisblank做了检查提供的字符是否是字母或者空白字符的所有工作。当我最后一次运行时,它打印出除了'3'之外的任何东西,因为它是一个数字。

如何使它崩溃

下面是使它崩溃的两种不同的方法:

  • 通过移除前向声明来把编译器搞晕。它会报告can_print_itprint_letters的错误。
  • 当你在main中调用print_arguments时,试着使argc加1,于是它会越过argv数组的最后一个元素。

附加题

  • 重新编写这些函数,使它们的数量减少。比如,你真的需要can_print_it吗?
  • 使用strlen函数,让print_arguments知道每个字符串参数都有多长,之后将长度传入print_letters。然后重写print_letters,让它只处理固定的长度,不按照'\0'终止符。你需要#include <string.h>来实现它。
  • 使用man来查询isalphaisblank的信息。使用其它相似的函数来只打印出数字或者其它字符。
  • 上网浏览不同的人喜欢什么样的函数格式。永远不要使用“K&R”语法,因为它过时了,而且容易使人混乱,但是当你碰到一些人使用这种格式时,要理解代码做了什么。