linux g 编译?Linux开发板

大家好,如果您还对linux g 编译不太了解,没有关系,今天就由本站为大家分享linux g 编译的知识,包括Linux开发板的问题都会给大家分析到,还望可以解决大家的问题,下面我们就开始吧!

在linux里,c程序的编辑、编译、调试的详细步骤是什么

在linux下,一般的c程序开发流程为:

c程序的编辑:

一般情况下,简单的c程序编辑是通过vi文本编辑器进行的,语法格式和在windows下的编辑是一样的;

c程序的编译:

而编译则是对编辑好的c语言程序进行一个,预处理,编辑,汇编和链接的过程,一般格式为:编译器名-参数目标文件名源文件名;

一般工作在命令行模式;

c程序的调试:

调试阶段则是建立在c程序编译通过的情况下进行的,一般通过GDB调试进行,逐步排错和细节查看。

gcc-g什么意思

gcc-g的含义解释

答案:gcc-g是指GNU的C语言编译器。它是一个开源的编程工具,用于将C语言源代码编译成可执行程序。在大多数Linux系统上,它通常是默认安装的编译器。由于其功能强大和开源的特性,gcc-g广泛应用于各种软件开发和编程环境中。

详细解释:

1. gcc-g的定义:gcc-g是GNU编译器集合的一部分,特别针对C语言编程。GCC是一个开源项目,旨在提供一种可在多种操作系统上使用的编译器。它支持多种编程语言,包括C、C++等。由于gcc-g支持多种平台和操作系统,因此在跨平台软件开发中非常受欢迎。

2. gcc-g的功能:gcc-g的主要功能是将C语言的源代码转换为机器语言,生成可执行文件。它还包括一些调试和优化选项,帮助开发者创建高效且稳定的程序。此外,它还支持多种标准库和运行时环境,使得开发者能够更方便地编写和测试代码。

3. gcc-g的重要性:gcc-g作为开源编译器,对软件开发具有重大意义。它推动了软件创新和开源文化的发展。许多大型软件和操作系统的开发都依赖于gcc-g编译器,它在开源软件的开发过程中发挥着关键作用。同时,由于其免费和开源的特性,开发者可以自由地获取和使用它,促进了软件行业的快速发展。此外,gcc-g还具有良好的兼容性,能够在不同的操作系统和硬件平台上运行,提高了软件的可用性和可靠性。

总的来说,gcc-g是GNU项目下的一个强大且广泛应用的C语言编译器,它为开发者提供了强大的工具来编写、编译和优化C语言程序。它的开源特性使其成为许多软件开发项目的重要组成部分。

linux 用g++编译c++代码的问题

*

运行 gcc/egcs

*

gcc/egcs的主要选项

*

gdb

*

gdb的常用命令

*

gdb使用范例

*

其他程序/库工具(ar, objdump, nm, size, strings, strip,...)

*创建和使用静态库

*创建和使用共享库

*使用高级共享库特性

1.7.1运行 gcc/egcs

Linux中最重要的软件开发工具是 GCC。GCC是 GNU的 C和 C++编译器。实际上,GCC能够编译三种语言:C、C++和 Object C(C语言的一种面向对象扩展)。利用 gcc命令可同时编译并连接 C和 C++源程序。

#DEMO#: hello.c

如果你有两个或少数几个 C源文件,也可以方便地利用 GCC编译、连接并生成可执行文件。例如,假设你有两个源文件 main.c和 factorial.c两个源文件,现在要编译生成一个计算阶乘的程序。

-----------------------

清单 factorial.c

-----------------------

#include<stdio.h>

#include<stdlib.h>

int factorial(int n)

{

if(n<= 1)

return 1;

else

return factorial(n- 1)* n;

}

-----------------------

-----------------------

清单 main.c

-----------------------

#include<stdio.h>

#include<stdlib.h>

int factorial(int n);

int main(int argc, char**argv)

{

int n;

if(argc< 2){

printf("Usage:%s n\n", argv [0]);

return-1;

}

else{

n= atoi(argv[1]);

printf("Factorial of%d is%d.\n", n, factorial(n));

}

return 0;

}

-----------------------

利用如下的命令可编译生成可执行文件,并执行程序:

$ gcc-o factorial main.c factorial.c

$./factorial 5

Factorial of 5 is 120.

GCC可同时用来编译 C程序和 C++程序。一般来说,C编译器通过源文件的后缀名来判断是 C程序还是 C++程序。在 Linux中,C源文件的后缀名为.c,而 C++源文件的后缀名为.C或.cpp。

但是,gcc命令只能编译 C++源文件,而不能自动和 C++程序使用的库连接。因此,通常使用 g++命令来完成 C++程序的编译和连接,该程序会自动调用 gcc实现编译。假设我们有一个如下的 C++源文件(hello.C):

#include<iostream.h>

void main(void)

{

cout<<"Hello, world!"<< endl;

}

则可以如下调用 g++命令编译、连接并生成可执行文件:

$ g++-o hello hello.C

$./hello

Hello, world!

1.7.2 gcc/egcs的主要选项

表 1-3 gcc命令的常用选项

选项解释

-ansi只支持 ANSI标准的 C语法。这一选项将禁止 GNU C的某些特色,

例如 asm或 typeof关键词。

-c只编译并生成目标文件。

-DMACRO以字符串“1”定义 MACRO宏。

-DMACRO=DEFN以字符串“DEFN”定义 MACRO宏。

-E只运行 C预编译器。

-g生成调试信息。GNU调试器可利用该信息。

-IDIRECTORY指定额外的头文件搜索路径DIRECTORY。

-LDIRECTORY指定额外的函数库搜索路径DIRECTORY。

-lLIBRARY连接时搜索指定的函数库LIBRARY。

-m486针对 486进行代码优化。

-o FILE生成指定的输出文件。用在生成可执行文件时。

-O0不进行优化处理。

-O或-O1优化生成代码。

-O2进一步优化。

-O3比-O2更进一步优化,包括 inline函数。

-shared生成共享目标文件。通常用在建立共享库时。

-static禁止使用共享连接。

-UMACRO取消对 MACRO宏的定义。

-w不生成任何警告信息。

-Wall生成所有警告信息。

#DEMO#

MiniGUI的编译选项

1.7.3 gdb

GNU的调试器称为 gdb,该程序是一个交互式工具,工作在字符模式。在 X Window系统中,

有一个 gdb的前端图形工具,称为 xxgdb。gdb是功能强大的调试程序,可完成如下的调试

任务:

*设置断点;

*监视程序变量的值;

*程序的单步执行;

*修改变量的值。

在可以使用 gdb调试程序之前,必须使用-g选项编译源文件。可在 makefile中如下定义

CFLAGS变量:

CFLAGS=-g

运行 gdb调试程序时通常使用如下的命令:

gdb progname

在 gdb提示符处键入help,将列出命令的分类,主要的分类有:

* aliases:命令别名

* breakpoints:断点定义;

* data:数据查看;

* files:指定并查看文件;

* internals:维护命令;

* running:程序执行;

* stack:调用栈查看;

* statu:状态查看;

* tracepoints:跟踪程序执行。

键入 help后跟命令的分类名,可获得该类命令的详细清单。

#DENO#

1.7.4 gdb的常用命令

表 1-4常用的 gdb命令

命令解释

break NUM在指定的行上设置断点。

bt显示所有的调用栈帧。该命令可用来显示函数的调用顺序。

clear删除设置在特定源文件、特定行上的断点。其用法为:clear FILENAME:NUM。

continue继续执行正在调试的程序。该命令用在程序由于处理信号或断点而

导致停止运行时。

display EXPR每次程序停止后显示表达式的值。表达式由程序定义的变量组成。

file FILE装载指定的可执行文件进行调试。

help NAME显示指定命令的帮助信息。

info break显示当前断点清单,包括到达断点处的次数等。

info files显示被调试文件的详细信息。

info func显示所有的函数名称。

info local显示当函数中的局部变量信息。

info prog显示被调试程序的执行状态。

info var显示所有的全局和静态变量名称。

kill终止正被调试的程序。

list显示源代码段。

make在不退出 gdb的情况下运行 make工具。

next在不单步执行进入其他函数的情况下,向前执行一行源代码。

print EXPR显示表达式 EXPR的值。

1.7.5 gdb使用范例

-----------------

清单一个有错误的 C源程序 bugging.c

-----------------

#include<stdio.h>

#include<stdlib.h>

static char buff [256];

static char* string;

int main()

{

printf("Please input a string:");

gets(string);

printf("\nYour string is:%s\n", string);

}

-----------------

上面这个程序非常简单,其目的是接受用户的输入,然后将用户的输入打印出来。该程序使用了

一个未经过初始化的字符串地址 string,因此,编译并运行之后,将出现 Segment Fault错误:

$ gcc-o test-g test.c

$./test

Please input a string: asfd

Segmentation fault(core dumped)

为了查找该程序中出现的问题,我们利用 gdb,并按如下的步骤进行:

1.运行 gdb bugging命令,装入 bugging可执行文件;

2.执行装入的 bugging命令;

3.使用 where命令查看程序出错的地方;

4.利用 list命令查看调用 gets函数附近的代码;

5.唯一能够导致 gets函数出错的因素就是变量 string。用 print命令查看 string的值;

6.在 gdb中,我们可以直接修改变量的值,只要将 string取一个合法的指针值就可以了,为

此,我们在第 11行处设置断点;

7.程序重新运行到第 11行处停止,这时,我们可以用 set variable命令修改 string的取值;

8.然后继续运行,将看到正确的程序运行结果。

#DEMO#

1.7.6其他程序/库工具

strip:

nm:

size:

string:

1.7.7创建和使用静态库

创建一个静态库是相当简单的。通常使用 ar程序把一些目标文件(.o)组合在一起,成为一个单独的库,然后运行 ranlib,以给库加入一些索引信息。

1.7.8创建和使用共享库

特殊的编译和连接选项

-D_REENTRANT使得预处理器符号 _REENTRANT被定义,这个符号激活一些宏特性。

-fPIC选项产生位置独立的代码。由于库是在运行的时候被调入,因此这个

选项是必需的,因为在编译的时候,装入内存的地址还不知道。如果

不使用这个选项,库文件可能不会正确运行。

-shared选项告诉编译器产生共享库代码。

-Wl,-soname-Wl告诉编译器将后面的参数传递到连接器。而-soname指定了

共享库的 soname。

#可以把库文件拷贝到/etc/ld.so.conf中列举出的任何目录中,并以

root身份运行 ldconfig;或者

#运行 export LD_LIBRARY_PATH='pwd',它把当前路径加到库搜索路径中去。

1.7.9使用高级共享库特性

1. ldd工具

ldd用来显示执行文件需要哪些共享库,共享库装载管理器在哪里找到了需要的共享库.

2. soname

共享库的一个非常重要的,也是非常难的概念是 soname——简写共享目标名(short for shared object name)。这是一个为共享库(.so)文件而内嵌在控制数据中的名字。如前面提到的,每一个程序都有一个需要使用的库的清单。这个清单的内容是一系列库的 soname,如同 ldd显示的那样,共享库装载器必须找到这个清单。

soname的关键功能是它提供了兼容性的标准。当要升级系统中的一个库时,并且新库的 soname和老的库的 soname一样,用旧库连接生成的程序,使用新的库依然能正常运行。这个特性使得在 Linux下,升级使用共享库的程序和定位错误变得十分容易。

在 Linux中,应用程序通过使用 soname,来指定所希望库的版本。库作者也可以通过保留或者改变 soname来声明,哪些版本是相互兼容的,这使得程序员摆脱了共享库版本冲突问题的困扰。

查看/usr/local/lib目录,分析 MiniGUI的共享库文件之间的关系

3.共享库装载器

当程序被调用的时候,Linux共享库装载器(也被称为动态连接器)也自动被调用。它的作用是保证程序所需要的所有适当版本的库都被调入内存。共享库装载器名字是 ld.so或者是 ld-linux.so,这取决于 Linux libc的版本,它必须使用一点外部交互,才能完成自己的工作。然而它接受在环境变量和配置文件中的配置信息。

文件/etc/ld.so.conf定义了标准系统库的路径。共享库装载器把它作为搜索路径。为了改变这个设置,必须以 root身份运行 ldconfig工具。这将更新/etc/ls.so.cache文件,这个文件其实是装载器内部使用的文件之一。

可以使用许多环境变量控制共享库装载器的操作(表1-4+)。

表 1-4+共享库装载器环境变量

变量含义

LD_AOUT_LIBRARY_PATH除了不使用 a.out二进制格式外,与 LD_LIBRARY_PATH相同。

LD_AOUT_PRELOAD除了不使用 a.out二进制格式外,与 LD_PRELOAD相同。

LD_KEEPDIR只适用于 a.out库;忽略由它们指定的目录。

LD_LIBRARY_PATH将其他目录加入库搜索路径。它的内容应该是由冒号

分隔的目录列表,与可执行文件的 PATH变量具有相同的格式。

如果调用设置用户 ID或者进程 ID的程序,该变量被忽略。

LD_NOWARN只适用于 a.out库;当改变版本号是,发出警告信息。

LD_PRELOAD首先装入用户定义的库,使得它们有机会覆盖或者重新定义标准库。

使用空格分开多个入口。对于设置用户 ID或者进程 ID的程序,

只有被标记过的库才被首先装入。在/etc/ld.so.perload中指定

了全局版本号,该文件不遵守这个限制。

4.使用 dlopen

另外一个强大的库函数是 dlopen()。该函数将打开一个新库,并把它装入内存。该函数主要用来加载库中的符号,这些符号在编译的时候是不知道的。比如 Apache Web服务器利用这个函数在运行过程中加载模块,这为它提供了额外的能力。一个配置文件控制了加载模块的过程。这种机制使得在系统中添加或者删除一个模块时,都不需要重新编译了。

可以在自己的程序中使用 dlopen()。dlopen()在 dlfcn.h中定义,并在 dl库中实现。它需要两个参数:一个文件名和一个标志。文件名可以是我们学习过的库中的 soname。标志指明是否立刻计算库的依赖性。如果设置为 RTLD_NOW的话,则立刻计算;如果设置的是 RTLD_LAZY,则在需要的时候才计算。另外,可以指定 RTLD_GLOBAL,它使得那些在以后才加载的库可以获得其中的符号。

当库被装入后,可以把 dlopen()返回的句柄作为给 dlsym()的第一个参数,以获得符号在库中的地址。使用这个地址,就可以获得库中特定函数的指针,并且调用装载库中的相应函数。

阅读剩余
THE END