📄 00000010.htm
字号:
char *string2; <br /> int size, i; <br /> size = strlen (string); <br /> string2 = (char *) malloc (size + 1); <br /> for (i = 0; i < size; i++) <br /> string2[size - i] = string[i]; <br /> string2[size+1] = `\0'; <br /> printf ("The string printed backward is %s\n", string2); <br />} <br /> 用下面的命令编译它: <br />gcc -o test test.c <br /> 这个程序执行时显示如下结果: <br />The string is hello there <br />The string printed backward is <br /> 输出的第一行是正确的, 但第二行打印出的东西并不是我们所期望的. 我们所设想 <br />的输出应该是: <br />The string printed backward is ereht olleh <br /> 由于某些原因, my_print2 函数没有正常工作. 让我们用 gdb 看看问题究竟出在 <br />哪儿, 先键入如下命令: <br />gdb greeting <br />---------------------------------------------------------------------------- <br />---- <br />注意: 记得在编译 greeting 程序时把调试选项打开. <br />---------------------------------------------------------------------------- <br />---- <br /> 如果你在输入命令时忘了把要调试的程序作为参数传给 gdb , 你可以在 gdb 提示 <br />符下用 file 命令来载入它: <br />(gdb) file greeting <br /> 这个命令将载入 greeting 可执行文件就象你在 gdb 命令行里装入它一样. <br /> 这时你能用 gdb 的 run 命令来运行 greeting 了. 当它在 gdb 里被运行后结果大 <br />约会象这样: <br />(gdb) run <br />Starting program: /root/greeting <br />The string is hello there <br />The string printed backward is <br />Program exited with code 041 <br /> 这个输出和在 gdb 外面运行的结果一样. 问题是, 为什么反序打印没有工作? 为了 <br />找出症结所在, 我们可以在 my_print2 函数的 for 语句后设一个断点, 具体的做法是 <br />在 gdb 提示符下键入 list 命令三次, 列出源代码: <br />(gdb) list <br />(gdb) list <br />(gdb) list <br />---------------------------------------------------------------------------- <br />---- <br />技巧: 在 gdb 提示符下按回车健将重复上一个命令. <br />---------------------------------------------------------------------------- <br />---- <br /> 第一次键入 list 命令的输出如下: <br />1 #include <stdio.h> <br />2 <br />3 main () <br />4 { <br />5 char my_string[] = "hello there"; <br />6 <br />7 my_print (my_string); <br />8 my_print2 (my_string); <br />9 } <br />10 <br /> 如果按下回车, gdb 将再执行一次 list 命令, 给出下列输出: <br />11 my_print (char *string) <br />12 { <br />13 printf ("The string is %s\n", string); <br />14 } <br />15 <br />16 my_print2 (char *string) <br />17 { <br />18 char *string2; <br />19 int size, i; <br />20 <br /> 再按一次回车将列出 greeting 程序的剩余部分: <br />21 size = strlen (string); <br />22 string2 = (char *) malloc (size + 1); <br />23 for (i = 0; i < size; i++) <br />24 string2[size - i] = string[i]; <br />25 string2[size+1] = `\0'; <br />26 printf ("The string printed backward is %s\n", string2); <br />27 } <br /> 根据列出的源程序, 你能看到要设断点的地方在第24行, 在 gdb 命令行提示符下键 <br />入如下命令设置断点: <br />(gdb) break 24 <br /> gdb 将作出如下的响应: <br />Breakpoint 1 at 0x139: file greeting.c, line 24 <br />(gdb) <br /> 现在再键入 run 命令, 将产生如下的输出: <br />Starting program: /root/greeting <br />The string is hello there <br />Breakpoint 1, my_print2 (string = 0xbfffdc4 "hello there") at greeting.c :24 <br /> <br />24 string2[size-i]=string[i] <br /> 你能通过设置一个观察 string2[size - i] 变量的值的观察点来看出错误是怎样产 <br />生的, 做法是键入: <br />(gdb) watch string2[size - i] <br /> gdb 将作出如下回应: <br />Watchpoint 2: string2[size - i] <br /> 现在可以用 next 命令来一步步的执行 for 循环了: <br />(gdb) next <br /> 经过第一次循环后, gdb 告诉我们 string2[size - i] 的值是 `h`. gdb 用如下 <br />的显示来告诉你这个信息: <br />Watchpoint 2, string2[size - i] <br />Old value = 0 `\000' <br />New value = 104 `h' <br />my_print2(string = 0xbfffdc4 "hello there") at greeting.c:23 <br />23 for (i=0; i<size; i++) <br /> 这个值正是期望的. 后来的数次循环的结果都是正确的. 当 i=10 时, 表达式 str <br />ing2[size - i] 的值等于 `e`, size - i 的值等于 1, 最后一个字符已经拷到新串里 <br />了. <br /> 如果你再把循环执行下去, 你会看到已经没有值分配给 string2[0] 了, 而它是新 <br />串的第一个字符, 因为 malloc 函数在分配内存时把它们初始化为空(null)字符. 所以 <br /> string2 的第一个字符是空字符. 这解释了为什么在打印 string2 时没有任何输出了 <br />. <br /> 现在找出了问题出在哪里, 修正这个错误是很容易的. 你得把代码里写入 string2 <br /> 的第一个字符的的偏移量改为 size - 1 而不是 size. 这是因为 string2 的大小为 <br />12, 但起始偏移量是 0, 串内的字符从偏移量 0 到 偏移量 10, 偏移量 11 为空字符保 <br />留. <br /> 为了使代码正常工作有很多种修改办法. 一种是另设一个比串的实际大小小 1 的变 <br />量. 这是这种解决办法的代码: <br />#include <stdio.h> <br />main () <br />{ <br /> char my_string[] = "hello there"; <br /> my_print (my_string); <br /> my_print2 (my_string); <br />} <br />my_print (char *string) <br />{ <br /> printf ("The string is %s\n", string); <br />} <br />my_print2 (char *string) <br />{ <br /> char *string2; <br /> int size, size2, i; <br /> size = strlen (string); <br /> size2 = size -1; <br /> string2 = (char *) malloc (size + 1); <br /> for (i = 0; i < size; i++) <br /> string2[size2 - i] = string[i]; <br /> string2[size] = `\0'; <br /> printf ("The string printed backward is %s\n", string2); <br />} <br />另外的 C 编程工具 <br /> Slackware Linux 的发行版中还包括一些我们尚未提到的 C 开发工具. 本节将介绍 <br />这些工具和它们的典型用法. <br />xxgdb <br /> xxgdb 是 gdb 的一个基于 X Window 系统的图形界面. xxgdb 包括了命令行版的 <br /> gdb 上的所有特性. xxgdb 使你能通过按按钮来执行常用的命令. 设置了断点的地方 <br />也用图形来显示. <br /> 你能在一个 Xterm 窗口里键入下面的命令来运行它: <br />xxgdb <br /> 你能用 gdb 里任何有效的命令行选项来初始化 xxgdb . 此外 xxgdb 也有一些特有 <br />的命令行选项, 表 27.2 列出了这些选项. <br /> 表 27.2. xxgdb 命令行选项. <br />选 项 描 述 <br />db_name 指定所用调试器的名字, 缺省是 gdb. <br />db_prompt 指定调试器提示符, 缺省为 gdb. <br />gdbinit 指定初始化 gdb 的命令文件的文件名, 缺省为 .gdbinit. <br />nx 告诉 xxgdb 不执行 .gdbinit 文件. <br />bigicon 使用大图标. <br />calls <br /> 你可以在 sunsite.unc.edu FTP 站点用下面的路径: <br />/pub/Linux/devel/lang/c/calls.tar.Z <br /> 来取得 calls , 一些旧版本的 Linux CD-ROM 发行版里也附带有. 因为它是一个有 <br />用的工具, 我们在这里也介绍一下. 如果你觉得有用的话, 从 BBS, FTP, 或另一张CD- <br />ROM 上弄一个拷贝. calls 调用 GCC 的预处理器来处理给出的源程序文件, 然后输出 <br />这些文件的里的函数调用树图. <br />---------------------------------------------------------------------------- <br />---- <br />注意: 在你的系统上安装 calls , 以超级用户身份登录后执行下面的步骤: 1. 解压和 <br /> untar 文件. 2. cd 进入 calls untar 后建立的子目录. 3. 把名叫 calls 的文件移 <br />动到 /usr/bin 目录. 4. 把名叫 calls.1 的文件移动到目录 /usr/man/man1 . 5. 删 <br />除 /tmp/calls 目录. 这些步骤将把 calls 程序和它的指南页安装载你的系统上. <br />---------------------------------------------------------------------------- <br />---- <br /> 当 calls 打印出调用跟踪结果时, 它在函数后面用中括号给出了函数所在文件的文 <br />件名: <br />
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -