https://wiki.ubuntu.org.cn/api.php?action=feedcontributions&user=%E4%B9%9D%E5%A4%B4%E9%B8%9F%E9%BE%99&feedformat=atom
Ubuntu中文 - 用户贡献 [zh-hans]
2024-03-28T08:54:58Z
用户贡献
MediaWiki 1.26.3
https://wiki.ubuntu.org.cn/index.php?title=Insight%E7%9A%84%E4%BD%BF%E7%94%A8%E6%96%B9%E6%B3%95%EF%BC%88gdb%E7%9A%84%E6%9C%80%E4%BC%98%E5%89%8D%E7%AB%AF%EF%BC%89&diff=140460
Insight的使用方法(gdb的最优前端)
2010-07-07T13:30:25Z
<p>九头鸟龙:/* 其他 */</p>
<hr />
<div>= '''insight概述''' =<br />
<br />
Insight是一个Tcl/Tk编写的gdb的图形用户接口,它是由在Red hat以及Cygnus的工程师写的。Red hat大方的将其捐赠给大众使用,甚至不时的提供人力继续完善它的功能。Insight项目始于1994年,至今已经有颇为成熟,它能很好的发挥大部分gdb的强大功能。<br />
<br />
<br>编写这个词条的目的,不是为了真的怎样教大家使用这个软件,因为这个软件已经相当的完善,不需要介绍就能轻松上手,主要是为了那些在internet上盲目搜索在linux下c,c++开发用什么调试器好的人提供一个hint!免得他们走歪路,或者甚至看到gdb那种单板机式的调试界面后退缩,甚至从此离开linux的人们.我可以很负责任的说,相对于xxgdb,ddd等基于gdb的的gui调试器来说,insight绝对是最棒,最易用的。<br />
<br />
<br>当然在linux下开发c,c++应用程序并非必须用gdb来调试,随着当今java的兴起,很多java IDE都提供c,c++开发功能,着名的有Eclipse(有iBM赞助的java IDE),NetBeans(Sun的java官方IDE),当然在安装eclipse和netbeans的时候你应该选择c/c++开发模块。<br />
<br />
<br>但是对于那些喜欢用emacs进行一条龙开发,而又对gdb颇有微词的人,Insight绝对是你最好的选择,而笔者自然就是这样的一种人:)<br />
<br />
= '''截图''' =<br />
<br />
如果大家平时使用的是gdb调试器,那么看到下面的的画面可能会激动不已,先奉上几张画面吧。<br />
<br />
[[Image:Source-and-asm-separate.png]]<br />
<br />
这是源代码与汇编代码的分离界面<br />
<br />
<br><br />
<br />
[[Image:Stack-browsing-with-regs-and-locals.png]]<br />
<br />
比较完善的功能图<br />
<br />
<br><br />
<br />
<br><br />
<br />
= '''安装''' =<br />
<br />
看完界面图是不是已经激动不已了呢,没错,vc调试器中有的功能他都有,vc调试器中没有的它也有,并且同样以gui表现出来。<br />
<br />
好吧,让我们来谈谈如何安装它把。<br />
<br />
当前稳定版本的insight是6.6版,我们可以通过下面两种方式来获得:<br />
<br />
<br>ftp: 从发行目录下载<br />
<br />
<br><br />
<br />
ftp://sourceware.org/pub/insight/releases .<br />
<br />
<br>CVS:在gdb-6.6的分支下检查一个只读拷贝<br />
<br />
<br>cvs -z9 -d&nbsp;:pserver:anoncvs@sourceware.org:/cvs/src login<br>[密码 "anoncvs" ]<br>cvs -z9 -d&nbsp;:pserver:anoncvs@sourceware.org:/cvs/src co -r gdb_6_6-branch insight<br />
<br />
当然,最简单的,你可以通过使用新立得软件包管理器来获得,直接搜索insight就可以了,或者你可以直接在命令行中打入:<br />
<pre>sudo apt-get install insight</pre> <br />
<br><br />
<br />
= '''开始''' =<br />
<br />
好了,安装结束,想使用insight,很简单,直接在命令行中打入insight就可以了,进入insight以後,你可以在他的file菜单下选择要调试的程序名,或者<br />
<pre>insight ./foo</pre> <br />
<br>通过後跟程序名的方法,就能直接对对给定的程序进行调试了。<br />
<br />
<br><br />
<br />
= '''其他''' =<br />
<br />
因为insight为很好的GUI接口,任何能熟练使用vc调试器的人都能很好的掌握它,所以具体使用方法作者不再赘述,但是对于那些不是很熟悉gdb的人,通过以下链接您就能对gdb有一个初步并且印象深刻的认识了:[[用GDB调试程序]]<br />
<br />
<br />
----<br />
作者:yacper<br />
<br />
邮箱:[mailto:yacper@gmail.com yacper@gmail.com]<br />
<br />
<br> 希望能够结交linux,以及热爱在linux下开发的朋友,一起在知识的海洋里遨游...<br />
<br />
[[Category:程序开发]]</div>
九头鸟龙
https://wiki.ubuntu.org.cn/index.php?title=Gcchowto&diff=140420
Gcchowto
2010-07-06T10:44:44Z
<p>九头鸟龙:/* 编译多个源文件 */</p>
<hr />
<div>{{TOCright}}<br />
<br />
本文翻译自 [http://www.network-theory.co.uk/gcc/intro/ ''An Introduction to GCC''] 的部分章节(有改动)。<br />
<br />
<br />
== 准备工作 ==<br />
<br />
'''注意:'''本文可能会让你失望,如果你有下列疑问的话:为什么要在终端输命令啊? GCC 是什么东西,怎么在菜单中找不到? GCC 不能有像 VC 那样的窗口吗?…… 那么你真正想要了解的可能是 anjuta,kdevelop,geany,code blocks,eclipse,netbeans 等 [[C Cpp IDE|IDE 集成开发环境]]。即使在这种情况下,由于 GCC 是以上 IDE 的後台的编译器,本文仍值得你稍作了解。<br />
<br />
如果你还没装编译环境或自己不确定装没装,不妨先执行<br />
<br />
sudo apt-get install build-essential<br />
<br />
如果你需要编译 Fortran 程序,那么还需要安装 [[Gcc Fortran Intro|gfortran(或 g77)]]<br />
<br />
sudo apt-get install gfortran<br />
<br />
== 编译简单的 C 程序 ==<br />
<br />
C 语言经典的入门例子是 '''Hello World''',下面是一示例代码:<br />
<br />
<source lang="c"><br />
#include <stdio.h><br />
int main(void)<br />
{<br />
printf("Hello, world!\n");<br />
return 0;<br />
}<br />
</source><br />
<br />
我们假定该代码存为文件‘hello.c’。要用 '''gcc''' 编译该文件,使用下面的命令:<br />
<br />
$ gcc -g -Wall hello.c -o hello<br />
<br />
该命令将文件‘hello.c’中的代码编译为机器码并存储在可执行文件 ‘hello’中。机器码的文件名是通过 '''-o''' 选项指定的。该选项通常作为命令行中的最后一个参数。如果被省略,输出文件默认为 ‘a.out’。<br />
<br />
'''注意到'''如果当前目录中与可执行文件重名的文件已经存在,它将被覆盖。<br />
<br />
选项 '''-Wall''' 开启编译器几乎所有常用的警告──'''强烈建议你始终使用该选项'''。编译器有很多其他的警告选项,但 '''-Wall''' 是最常用的。默认情况下GCC 不会产生任何警告信息。当编写 C 或 C++ 程序时编译器警告非常有助于检测程序存在的问题。<br />
<br />
本例中,编译器使用了 ''-Wall'' 选项而没产生任何警告,因为示例程序是完全合法的。<br />
<br />
选项 ""-g"" 表示在生成的目标文件中带调试信息,调试信息可以在程序异常中止产生core后,帮助分析错误产生的源头,包括产生错误的文件名和行号等非常多有用的信息。<br />
<br />
要运行该程序,输入可执行文件的路径如下:<br />
<br />
$ ./hello<br />
Hello, world!<br />
<br />
这将可执行文件载入内存,并使 CPU 开始执行其包含的指令。 路径 '''./''' 指代当前目录,因此 '''./hello''' 载入并执行当前目录下的可执行文件 ‘hello’。<br />
<br />
点击[http://ubuntu:ubuntuftp@ftp.ubuntu.org.cn/home/dbzhang800/wiki/gcchowto1.avi 此处]下载本节的操作视频<br />
<br />
== 捕捉错误 ==<br />
<br />
如上所述,当用 C 或 C++ 编程时,编译器警告是非常重要的助手。为了说明这一点,下面的例子包含一个微妙的错误:为一个整数值错误地指定了一浮点数控制符‘%f’。<br />
<br />
<source lang="c"><br />
#include <stdio.h><br />
<br />
int main (void)<br />
{<br />
printf ("Two plus two is %f\n", 4);<br />
return 0;<br />
}<br />
</source><br />
<br />
一眼看去该错误并不明显,但是它可被编译器捕捉到,只要启用了警告选项 '''-Wall'''。<br />
<br />
编译上面的程序‘bad.c’,将得到如下的消息:<br />
<br />
$ gcc -Wall -o bad bad.c<br />
main.c: 在函数‘main’中:<br />
main.c:5: 警告: 格式‘%f’需要类型‘double’,但实参 2 的类型为‘int’<br />
<br />
<br />
这表明文件 ‘bad.c’第 6 行中的格式字符串用法不正确。GCC 的消息总是具有下面的格式 ''文件名:行号:消息''。编译器对错误与警告区别对待,前者将阻止编译,后者表明可能存在的问题但并不阻止程序编译。<br />
<br />
本例中,对整数值来说,正确的格式控制符应该是 '''%d'''。<br />
<br />
如果不启用 '''-Wall''',程序表面看起来编译正常,但是会产生不正确的结果:<br />
<!--现在也会警告--><br />
$ gcc bad.c -o bad<br />
$ ./bad<br />
Two plus two is 0.000000<br />
<br />
显而易见,开发程序时不检查警告是非常危险的。如果有函数使用不当,将可能导致程序崩溃或产生错误的结果。开启编译器警告选项 '''-Wall''' 可捕捉 C 编程时的多数常见错误。<br />
<br />
== 编译多个源文件 ==<br />
<br />
一个源程序可以分成几个文件。这样便于编辑与理解,尤其是程序非常大的时候。这也使各部分独立编译成为可能。<br />
<br />
下面的例子中我们将程序 ''Hello World'' 分割成 3 个文件:‘hello.c’,‘hello_fn.c’和头文件‘hello.h’。这是主程序‘hello.c’:<br />
<br />
<source lang="c"><br />
#include "hello.h"<br />
int main(void)<br />
{<br />
hello ("world");<br />
return 0;<br />
}<br />
</source><br />
<br />
在先前例子的‘hello.c’中,我们调用的是库函数 '''printf''',本例中我们用一个定义在文件‘hello_fn.c’中的函数 '''hello''' 取代它。<br />
<br />
主程序中包含有头文件‘hello.h’,该头文件包含函数 '''hello''' 的声明。我们不需要在‘hello.c’文件中包含系统头文件‘stdio.h’来声明函数 '''printf''',因为‘hello.c’没有直接调用 '''printf'''。<br />
<br />
文件‘hello.h’中的声明只用了一行就指定了函数 '''hello''' 的原型。<br />
<br />
<source lang="c"><br />
void hello (const char * name);<br />
</source><br />
<br />
函数 '''hello''' 的定义在文件‘hello_fn.c’中:<br />
<br />
<source lang="c"><br />
#include <stdio.h><br />
#include "hello.h"<br />
<br />
void hello (const char * name)<br />
{<br />
printf ("Hello, %s!\n", name);<br />
}<br />
</source><br />
<br />
语句 '''#include "FILE.h"''' 与 '''#include &lt;FILE.h&gt;''' 有所不同:前者在搜索系统头文件目录之前将先在当前目录中搜索文件‘FILE.h’,后者只搜索系统头文件而不查看当前目录。<br />
<br />
要用''gcc''编译以上源文件,使用下面的命令:<br />
<br />
$ gcc -Wall hello.c hello_fn.c -o newhello<br />
<br />
本例中,我们使用选项 '''-o''' 为可执行文件指定了一个不同的名字 '''newhello'''。注意到头文件‘hello.h’并未在命令行中指定。源文件中的的 '''#include "hello.h"''' 指示符使得编译器自动将其包含到合适的位置。<br />
<br />
要运行本程序,输入可执行文件的路径名:<br />
<br />
$ ./newhello<br />
Hello, world!<br />
<br />
源程序各部分被编译为单一的可执行文件,它与我们先前的例子产生的结果相同。<br />
<br />
点击[http://ubuntu:ubuntuftp@ftp.ubuntu.org.cn/home/dbzhang800/wiki/gcchowto2.avi 此处]下载本节的操作视频<br />
<br />
== 简单的 Makefile 文件 ==<br />
<br />
为便于不熟悉 '''make''' 的读者理解,本节提供一个简单的用法示例。Make 凭借本身的优势,可在所有的 Unix 系统中被找到。要了解关于Gnu make 的更多信息,请参考 Richard M. Stallman 和 Roland McGrath 编写的 ''GNU Make'' 手册。<br />
<br />
Make 从 ''makefile''(默认是当前目录下的名为‘Makefile’的文件)中读取项目的描述。makefile指定了一系列''目标''(比如可执行文件)和''依赖''(比如对象文件和源文件)的编译规则,其格式如下:<br />
<br />
目标: 依赖<br />
命令<br />
<br />
对每一个目标,make 检查其对应的依赖文件修改时间来确定该目标是否需要利用对应的命令重新建立。注意到,makefile 中''命令''行必须以单个的 '''<font color="blue">TAB</font>''' 字符进行缩进,不能是空格。<br />
<br />
GNU Make 包含许多默认的规则(参考''隐含''规则)来简化 makefile 的构建。比如说,它们指定‘'''.o'''’文件可以通过编译‘'''.c'''’文件得到,可执行文件可以通过将‘'''.o'''’链接到一起获得。隐含规则通过被叫做''make变量''的东西所指定,比如 '''CC'''(C 语言编译器)和 '''CFLAGS'''(C程序的编译选项);在makefile文件中它们通过独占一行的 '''变量=值''' 的形式被设置。对 '''C++''' ,其等价的变量是'''CXX'''和'''CXXFLAGS''',而变量'''CPPFLAGS'''则是编译预处理选项。<br />
<br />
现在我们为上一节的项目写一个简单的 makefile 文件:<br />
<br />
CC=gcc<br />
CFLAGS=-Wall<br />
hello: hello.o hello_fn.o<br />
clean:<br />
rm -f hello hello.o hello_fn.o<br />
<br />
该文件可以这样来读:使用 C 语言编译器 '''gcc''',和编译选项‘-Wall’,从对象文件‘hello.o’和‘hello_fn.o’生成目标可执行文件 '''hello'''(文件‘hello.o’和‘hello_fn.o’通过隐含规则分别由‘hello.c’和‘hello_fn.c’生成)。目标'''clean'''没有依赖文件,它只是简单地移除所有编译生成的文件。'''rm'''命令的选项 ‘-f’(force) 抑制文件不存在时产生的错误消息。<br />
<br />
另外,需要注意的是,如果包含main函数的cpp文件为A.cpp, makefile中最好把可执行文件名也写成<br />
A。<br />
<br />
要使用该 makefile 文件,输入 '''make'''。不加参数调用make时,makefile文件中的第一个目标被建立,从而生成可执行文件‘hello’:<br />
<br />
$ make<br />
gcc -Wall -c -o hello.o hello.c<br />
gcc -Wall -c -o hello_fn.o hello_fn.c<br />
gcc hello.o hello_fn.o -o hello<br />
$ ./hello<br />
Hello, world!<br />
<br />
一个源文件被修改要重新生成可执行文件,简单地再次输入 '''make''' 即可。通过检查目标文件和依赖文件的时间戳,程序 make 可识别哪些文件已经修改并依据对应的规则更新其对应的目标文件:<br />
<br />
$ vim hello.c (打开编辑器修改一下文件)<br />
$ make<br />
gcc -Wall -c -o hello.o hello.c<br />
gcc hello.o hello_fn.o -o hello<br />
$ ./hello<br />
Hello, world!<br />
<br />
最后,我们移除 make 生成的文件,输入 make clean:<br />
<br />
$ make clean<br />
rm -f hello hello.o hello_fn.o<br />
<br />
一个专业的 makefile文件通常包含用于安装(make install)和测试(make check)等额外的目标。<br />
<br />
本文中涉及到的例子都足够简单以至于可以完全不需要makefile,但是对任何大些的程序都使用 make 是很有必要的。<br />
<br />
== 链接外部库 ==<br />
<br />
库是预编译的目标文件(object files)的集合,它们可被链接进程序。静态库以后缀为‘.a’的特殊的''存档文件(archive file)''存储。<br />
<br />
标准系统库可在目录 '''/usr/lib''' 与 '''/lib''' 中找到。比如,在类 Unix 系统中 C 语言的数学库一般存储为文件 '''/usr/lib/libm.a'''。该库中函数的原型声明在头文件 '''/usr/include/math.h''' 中。C 标准库本身存储为 '''/usr/lib/libc.a''',它包含 ANSI/ISO C 标准指定的函数,比如‘printf’。对每一个 C 程序来说,libc.a 都默认被链接。<br />
<br />
下面的是一个调用数学库 '''libm.a''' 中 '''sin''' 函数的的例子,创建文件'''calc.c''':<br />
<br />
<source lang="c"><br />
#include <math.h><br />
#include <stdio.h><br />
<br />
int main (void)<br />
{<br />
double x = 2.0;<br />
double y = sin (x);<br />
printf ("The value of sin(2.0) is %f\n", y);<br />
return 0;<br />
}<br />
</source><br />
<br />
尝试单独从该文件生成一个可执行文件将导致一个链接阶段的错误:<br />
<br />
$ gcc -Wall calc.c -o calc<br />
/tmp/ccbR6Ojm.o: In function 'main':<br />
/tmp/ccbR6Ojm.o(.text+0x19): undefined reference to ‘sin’<br />
<br />
函数 '''sin''',未在本程序中定义也不在默认库‘libc.a’中;除非被指定,编译器也不会链接‘libm.a’。<br />
<br />
为使编译器能将 '''sin''' 链接进主程序‘calc.c’,我们需要提供数学库‘libm.a’。一个容易想到但比较麻烦的做法是在命令行中显式地指定它:<br />
<br />
$ gcc -Wall calc.c /usr/lib/libm.a -o calc<br />
<br />
函数库‘libm.a’包含所有数学函数的目标文件,比如'''sin''','''cos''','''exp''','''log'''及'''sqrt'''。链接器将搜索所有文件来找到包含 '''sin''' 的目标文件。<br />
<br />
一旦包含 '''sin''' 的目标文件被找到,主程序就能被链接,一个完整的可执行文件就可生成了:<br />
<br />
$ ./calc<br />
The value of sin(2.0) is 0.909297<br />
<br />
可执行文件包含主程序的机器码以及函数库‘libm.a’中 '''sin''' 对应的机器码。<br />
<br />
为避免在命令行中指定长长的路径,编译器为链接函数库提供了快捷的选项‘-l’。例如,下面的命令<br />
<br />
$ gcc -Wall calc.c -lm -o calc<br />
<br />
与我们上面指定库全路径‘/usr/lib/libm.a’的命令等价。<br />
<br />
一般来说,选项 '''-l'''''NAME''使链接器尝试链接系统库目录中的函数库文件 '''lib'''''NAME''.'''a'''。一个大型的程序通常要使用很多 '''-l''' 选项来指定要链接的数学库,图形库,网络库等。<br />
<br />
==编译C++与Fortran==<br />
GCC 是 GNU 编译器集合(GNU Compiler Collection)的首字母缩写词。GNU 编译器集合包含 C,C++,Objective-C,Fortran,Java 和 Ada 的前端以及这些语言对应的库(libstdc++,libgcj,……)。<br />
<br />
前面我们只涉及到 C 语言,那么如何用 gcc 编译其他语言呢?本节将简单介绍 C++ 和 Fortran 编译的例子。<br />
<br />
首先我们尝试编译简单的 C++ 的经典程序 '''Hello world''':<br />
<br />
<source lang="c"><br />
#include <iostream><br />
int main(int argc,char *argv[])<br />
{<br />
std::cout << "hello, world" << std::endl;<br />
return 0;<br />
}<br />
</source><br />
<br />
将文件保存为‘hello.cpp’,用 gcc 编译,结果如下:<br />
$ gcc -Wall hello.cpp -o hello<br />
/tmp/cch6oUy9.o: In function `__static_initialization_and_destruction_0(int, int)':<br />
hello.cpp:(.text+0x23): undefined reference to `std::ios_base::Init::Init()'<br />
/tmp/cch6oUy9.o: In function `__tcf_0':<br />
hello.cpp:(.text+0x6c): undefined reference to `std::ios_base::Init::~Init()'<br />
/tmp/cch6oUy9.o: In function `main':<br />
hello.cpp:(.text+0x8e): undefined reference to `std::cout'<br />
hello.cpp:(.text+0x93): undefined reference to `std::basic_ostream<char, std::char_traits<char> >& std::operator<< <std::char_traits<char> >(std::basic_ostream<char, std::char_traits<char> >&, char const*)'<br />
/tmp/cch6oUy9.o:(.eh_frame+0x11): undefined reference to `__gxx_personality_v0'<br />
collect2: ld returned 1 exit status<br />
<br />
出错了!!而且错误还很多,很难看懂,这可怎么办呢?在解释之前,我们先试试下面的命令:<br />
$ gcc -Wall hello.cpp -o hello -lstdc++<br />
<br />
噫,加上-lstdc++选项后,编译竟然通过了,而且没有任何警告。运行程序,结果如下:<br />
$ ./hello <br />
hello, world<br />
<br />
通过上节,我们可以知道,-lstdc++ 选项用来通知链接器链接静态库 libstdc++.a。而从字面上可以看出,libstdc++.a 是C++ 的标准库,这样一来,上面的问题我们就不难理解了──编译 C++ 程序,需要链接 C++ 的函数库 libstdc++.a。<br />
<br />
编译 C 的时候我们不需要指定 C 的函数库,为什么 C++ 要指定呢?这是由于早期 gcc 是指 GNU 的 C 语言编译器(GNU C Compiler),随着 C++,Fortran 等语言的加入,gcc的含义才变化成了 GNU 编译器集合(GNU Compiler Collection)。C作为 gcc 的原生语言,故编译时不需额外的选项。<br />
<br />
不过幸运的是,GCC 包含专门为 C++ 、Fortran 等语言的编译器前端。于是,上面的例子,我们可以直接用如下命令编译:<br />
$ g++ -Wall hello.cpp -o hello<br />
<br />
GCC 的 C++ 前端是 g++,而 Fortran 的情况则有点复杂:在 gcc-4.0 版本之前,Fortran 前端是 g77,而gcc-4.0之后的版本对应的 Fortran 前端则改为 gfortran。下面我们先写一个简单的 Fortran 示例程序:<br />
<source lang="fortran"><br />
C Fortran 示例程序<br />
PROGRAM HELLOWORLD<br />
WRITE(*,10)<br />
10 FORMAT('hello, world')<br />
END PROGRAM HELLOWORLD<br />
</source><br />
<br />
将文件保存‘hello.f’,用 GCC 的 Fortran 前端编译运行该文件<br />
$ gfortran -Wall hello.f -o hello<br />
$ ./hello<br />
hello, world<br />
<br />
我们已经知道,直接用 gcc 来编译 C++ 时,需要链接 C++ 标准库,那么用 gcc 编译 Fortran时,命令该怎么写呢?<br />
$ gcc -Wall hello.f -o helloworld -lgfortran -lgfortranbegin<br />
<br />
'''注意''':上面这条命令与 gfortran 前端是等价的(g77 与此稍有不同)。其中库文件 libgfortranbegin.a (通过命令行选项 -lgfortranbegin 被调用) 包含运行和终止一个 Fortran 程序所必须的开始和退出代码。库文件 libgfortran.a 包含 Fortran 底层的输入输出等所需要的运行函数。<br />
<br />
对于 g77 来说,下面两条命令是等价的(注意到 g77 对应的 gcc 是 4.0 之前的版本):<br />
$ g77 -Wall hello.f -o hello<br />
$ gcc-3.4 -Wall hello.f -o hello -lfrtbegin -lg2c<br />
<br />
命令行中的两个库文件分别包含 Fortran 的开始和退出代码以及 Fortran 底层的运行函数。<br />
<br />
==其他参考==<br />
{{Gccmorerefer}}</div>
九头鸟龙