linux成长之路(gcc编译器、静态库、动态库)

Jeremy Lin

GCC简介

GCC(GNU Complier Collection)是GNU推出的功能强大、性能优越的多平台编译器套件,它包括了C、C++、Objective-C、Fortran、Java、Ada和Go语言的前端,也包括了这些语言的库,当前最新的版本是GCC 5.1。GCC可以在多种硬件平台上编译出可执行程序,其执行效率与一般的编译器相比平均效率要高20%-30%。GCC编译器能将C、C++语言源程序、汇程式程序和目标程序编译、连接成可执行文件,如果没有给出可执行文件的名字,GCC将生成一个名为
a.out 的文件。在Linux系统中,可执行文件没有统一的后缀,系统从文件的属性来区分可执行文件和不可执行文件。而GCC通过后缀来区分输入文件的类别,下面我们来介绍GCC所遵循的部分约定规则:

.c的文件                         C语言源代码文件;

.a的文件                        由目标文件构成的档案库文件;

.C,.cc或.cxx 的文件     C++源代码文件且必须要经过预处理;

.h的文件                         程序所包含的头文件;

.i 的文件                         C源代码文件且不应该对其执行预处理;

.ii的文件                         C++源代码文件且不应该对其执行预处理;

.m的文件                        Objective-C源代码文件;

.mm的文件                     Objective-C++源代码文件;

.o的文件                         编译后的目标文件;

.s的文件                         汇编语言源代码文件;

.S的文件                         经过预编译的汇编语言源代码文件。

GCC执行过程

虽然我们称gcc是C语言的编译器,但适用gcc由C语言源代码生成可执行文件的过程不仅仅是编译的过程,而是要经历4个相互关联的步骤:

(1)预处理(Preprocessing)

(2)编译(Compilation)

(3)汇编(Assembly)

(4)连接(Linking)

(1)预处理阶段:输入的是C语言的源文件,通常为*.c。它们通常带有.h之类的头文件。这个阶段主要是处理源文件中的#ifdef、#include和#define命令。该阶段会生成一个中间文件*.i,但实际工作中通常不会专门生成这种文件,因为基本上用不到;若非要生成这种文件,可以利用下面的命令:

gcc -E hello.c -o hello.i

hello.c的源代码如下:

#include <stdio.h>
int main(int argc, char**argv)
{
	printf("Hello Linux\n");
	return 0;
}

生成的hello.i的结果如下(很长,所以只截取一部分看看)

……

……

(2)编译阶段:输入的是中间文件*.i,编译后生成汇编语言文件*.s。这个阶段对应的GCC命令如下:

gcc -S hello.i -o hello.s

如下所示:

(3)汇编阶段:输入的是汇编文件*.s,输出的转换生成的机器语言*.o。这个阶段对应的命令如下:

gcc -c hello.s -o hello.o

(4)连接阶段:将输入的机器代码文件*.s,汇集成一个可执行的二进制文件。命令如下:

gcc hello.o -o helloX

GCC基本用法与选项

在使用GCC编译器的时候,我们必须给出一系列必要的调用参数和文件名称。GCC编译器的调用参数大约有100多个,其中多数参数我们可能根本就用不到,这里只介绍其中最集中、最常用的参数。

GCC的基本用法:

gcc [options] [filenames]

其中options就是编译器所需要的参数,filenames给出相关的文件名称。

-c,  只编译,不连接成为可执行文件。编译器只是由输入的*.c等源代码文件生成*.o目标文件,通常用于编译不包含主程序的子程序文件;

-o output_filename,确定输出文件的名称为output_filename,同时这个名称不能和源文件同名。如果不给出这个选项,gcc给出预设的可执行文件a.out;

-g,产生符号调试工具(GNU的gdb)所必要的符号资讯,要想对源代码进行调试,我们必须加入这个选项;

-O,对程序进行优化编译、连接,采用这个选项,整个源代码会在编译、连接过程中进行优化处理,这样产生的可执行文件的执行效率可以提高,但是,编译、连接的速度相应的要慢一些;

-O2,比-O更好的优化编译、连接,相应的速度会更慢一些;

-Wall,编译警告选项,在编译过程中如果gcc遇到一些它认为可能发生错误的地方就会提出一些相应的警告和提示消息;

-I dirname,将dirname所指出的目录加入到程序头文件搜索目录列表中,是在预编译过程中使用的参数;

-S,编译指定的源文件,但不进行汇编;

-E,预处理指定的源文件,不进行编译;

编译静态库

静态库是一些目标代码的集合。Linux环境下的静态库目标文件一般是以 .a 作为目标文件的文件扩展名。

Linux 环境下使用 ar 命令创建一个静态库。静态库的优点在于使用简单,编译快速。静态库在应用程序生成时,已经编译成为可重定位的目标文件,因此可以不必要再编译,节省编译时间,以最短的时间生成可执行程序。而且,静态库在编译的时候已经复制到可执行程序的代码中,不需要像使用动态库那样再程序加载时再次链接。在理论上,可执行程序使用静态库,要比使用动态库速度快1%~5%。

(1)创建静态库

假设有一源代码 static_lib.c:

int add( int a, int b )
{
	return a + b;
}
int sub( int a, int b )
{
	return a - b;
}
int mul( int a, int b )
{
	return a * b;
}
int div( int a, int b )
{
	return a / b;
}

现在要将其编译成为静态库文件——static_lib.a,则可使用如下的命令:

gcc -c static_lib.c
ar rcs static_lib.a static_lib.c

其中 ar 可以将目标文件加入到一个已经存在的静态库中(若不存在则创建),rcs 参数分别表示:把列表中的目标文件加入到静态库中(参数r),若指定静态库不存在,则创建该静态库文件(参数c),最后更新静态库文件的索引,使之包含新加入的目标文件的内容(参数s)。

(2)使用静态库

静态库创建成功后,需要链接到应用程序中使用。应用程序可以引用该库中的函数或者全局变量。为了使应用程序可以正确引用该库中的全局符号,需要制作一个包含该静态库中全局符号声明的文件。这个头文件可以包含在应用程序的头文件中,这样就可以引用静态库中的全局符号了。

头文件static_lib.h

extern int add( int a, int b);
extern int sub( int a, int b );
extern int mul( int a, int b );
extern int div( int a, int b );

应用程序main.c:

#include <stdio.h>
#include "static_lib.h"

int main(void)
{
	int a, b;

	printf("please input a and b \n");
	scanf( "%d%d", &a, &b );
	printf("the add:  %d", add(a,b));
	printf("the sub:  %d", sub(a,b));
	printf("the mul:  %d", mul(a,b));
	printf("the div:  %d", div(a,b));
	return 0;
}

我们使用 gcc 的 -l 选项来指定静态库,或者使用 -L 参数来指定库文件的搜索路径。-l 和 -L 之后都直接带参数而不跟空格。因而,对于本例,编译命令如下:

gcc main.c -lstatic_lib.a -o app

或者使用 “ . ” 作为静态库的搜索路径,表示在当前目录下搜索需要的静态库文件。

gcc -L. main.c -o app

注意:-l 是链接器选项,一定要放在被编译的源文件的文件名之后。

gcc 也支持使用-static 选项对静态库进行链接。

gcc main.c -static ./static_lib.a -o app

编译动态库

动态库又称为共享库或者动态链接库,现代程序中大量地使用动态链接库。例如,Windows 环境中的dll文件和Linux环境下的 so文件。动态库是在程序启动时被装载的。当一个程序装载了一个动态库后,其他应用程序仍然可以装载同一个动态库。

(1)创建动态库

假设有一份需要转化成动态库的代码share_lib.c

void insert_sort( int *array, int length)
{
	int i, j;
	for (i = 1; i < length; ++i)
	{
		int temp;
		j = i;
		temp = array[j];

		while(j > 0){
			if (temp < array[j - 1])
			{
				array[j] = array[j - 1];
				j--;
			}else
			break;
		}
		array[j] = temp;
	}
}

int binary_search( int *array, int item, int length)
{
	int high, low, mid;
	high = length - 1;
	low = 0;
	mid = (high + low) / 2;

	while(low <= high)
	{
		if (array[mid] > item)
		{
			high = mid;
			mid = (high + low) / 2;
		}else if ( array[mid] < item)
		{
			low = mid;
			mid = (high + low)/2;
		}
		else
			return mid;
	}
return -1;
}

Linux使用gcc创建一个动态库。由于动态库可以被多个进程共享加载,所以需要生成位置无关的目标文件。这时需要使用gcc编译器的-fPIC选项,该选项用于生成位置无关的代码。除了需要使用-fPIC选项外,还需要使用-shared 选项,该选项将位置无关的代码制作为动态库。

gcc -shared -fPIC share_lib.so share_lib.c

(2)使用动态库

动态库创建完后,应用程序可以引用该库中的函数或全局变量。为了使应用程序可以正确引用该库中的全局符号,需要制作一个包含该动态库中全局符号声明的头文件。

如share_lib.h

extern void insert_sort( int *array, int length);
extern int binary_search( int *array, int item, int length);

使用动态库的main.c

#include <stdio.h>
#include "share_lib.h"

 int main(int argc, char const *argv[])
{
	int array[5] = {5,4,2,3,1};
	int item;
	int pos;

	insert_sort( array, 5);

	printf("please input a number\n");
	scanf("%d", &item);
	pos = binary_search( array, item, 5);

	if (pos == -1)
	{
	        printf("can't find\n");
	}else
	{
                      printf("the position is %d\n", pos+1);
	}

	return 0;
}

使用如下命令创建程序:

gcc main.c ./share_lib.so -o app

本文地址:http://blog.csdn.net/linj_m/article/details/45039347

更多资源请关注 博客:LinJM-机器视觉  微博:林建民-机器视觉

时间: 2024-10-01 05:27:31

linux成长之路(gcc编译器、静态库、动态库)的相关文章

gcc编译工具生成动态库和静态库之一----介绍

 1.库的分类 根据链接时期的不同,库又有静态库和动态库之分. 静态库是在链接阶段被链接的(好像是废话,但事实就是这样),所以生成的可执行文件就不受库的影响了,即使库被删除了,程序依然可以成功运行. 有别于静态库,动态库的链接是在程序执行的时候被链接的.所以,即使程序编译完,库仍须保留在系统上,以供程序运行时调用.(TODO:链接动态库时链接阶段到底做了什么) 2 静态库和动态库的比较 链接静态库其实从某种意义上来说也是一种粘贴复制,只不过它操作的对象是目标代码而不是源码而已.因为静态库被链接后

Linux 静态库&amp;动态库调用

1.什么是库在windows平台和linux平台下都大量存在着库.本质上来说库是一种可执行代码的二进制形式,可以被操作系统载入内存执行.由于windows和linux的本质不同,因此二者库的二进制是不兼容的.本文仅限于介绍linux下的库.2.库的种类linux下的库有两种:静态库和共享库(动态库).二者的不同点在于代码被载入的时刻不同.静态库的代码在编译过程中已经被载入可执行程序,因此体积较大.共享库的代码是在可执行程序运行时才载入内存的,在编译过程中仅简单的引用,因此代码体积较小.3.库存在

C/C++ 跨平台交叉编译、静态库/动态库编译、MinGW、Cygwin、CodeBlocks使用原理及链接参数选项

目录 0. 引言 1. 交叉编译 2. Cygwin简介 3. 静态库编译及使用 4. 动态库编译及使用 5. MinGW简介 6. CodeBlocks简介 0. 引言 UNIX是一个注册商标,是要满足一大堆条件并且支付可观费用才能够被授权使用的一个操作系统.linux是unix的克隆版本,是由其创始人Linus和诸多世界知名的黑客手工打造的一个操作系统.为什么linux和unix之间有很多软件可以很轻松的移植?因为linux也满足POSIX规范,所以在运行机制上跟unix相近.同时,POSI

018_C语言静态库动态库

 linux下C语言编程环境 头文件目录:/usr/include/ 库目录:/usr/lib/ 动态库与静态库的区别:在计算机程序源代码需要通过翻译.汇编.链接过程而生成可执行文件,再由加载器加载到内存,两者区别就在于程序加载时链接到相应库的方式不同,程序编译时指定了静态库,编译后可直接运行,而编译时指定动态库时,编译后还要先指定库的路径. 静态库: ar -rs libmy_math.a *.o 生成静态库libmy_math.a ar -t 查看静态库 gcc test.c -L. -

转: gcc 指定运行时动态库路径

gcc 指定运行时动态库路径 Leave a reply 由于种种原因,Linux 下写 c 代码时要用到一些外部库(不属于标准C的库),可是由于没有权限,无法将这写库安装到系统目录,只好安装用户目录下如 /home/youname/lib,可是怎么编译才能让程序正常编译,并且正常运行呢.这样使用gcc:gcc -I/path/to/include/dir -L/path/to/lib/dir -llibname -Wl,-rpath,/path/to/lib/dir -o test test.

Linux下c函数dlopen实现加载动态库so文件代码举例

dlopen()是一个强大的库函数.该函数将打开一个新库,并把它装入内存.该函数主要用来加载库中的符号,这些符号在编译的时候是不知道的.这种机制使得在系统中添加或者删除一个模块时,都不需要重新编译了.可以在自己的程序中使用 dlopen().dlopen() 在 dlfcn.h 中定义,并在 dl 库中实现.它需要两个参数:一个文件名和一个标志.文件名就是一个动态库so文件,标志指明是否立刻计算库的依赖性.如果设置为 RTLD_NOW 的话,则立刻计算:如果设置的是 RTLD_LAZY,则在需要

C++开发新版本vs使用旧版本vs编译的静态库动态库

关于vs潜在的升级问题概述 (Visual C++)查看官网的介绍:潜在的升级问题概述 (Visual C++).主要问题: 1. 如果使用 /GL(全程序优化)进行编译,则生成的对象文件只能使用生成它时所用的同一工具集进行链接.即若在编译静态库时开启了全程序优化,那么编译出的静态库不可移植. 2. 在 Visual Studio 2013 及更低版本中,不能保证使用不同编译器工具集和运行时库版本生成的对象文件 (OBJ).静态库 (LIB).动态库 (DLL) 和可执行文件 (EXE) 之间的

静态、动态库制作使用

本质 库是一种可执行代码的二进制格式,可以被载入内存中执行 库分静态库和动态库两种. 一.概念 静态函数库 库的名字一般是libxxx.a 函数库的链接是在编译时期完成 所有相关对象与牵涉的函数库被链接合成一个可执行文件 运行时,与函数库再无瓜葛 动态函数库 库的名字一般是libxxx.so 函数库的链接是在运行时期完成 运行时,在运行环境中必须有相应的库 二.优缺点 静态函数库缺点 文件较大,浪费空间 库或者程序一方改变都使得重新编译 动态函数库优点 文件较小,节省空间. 库具有公用性,内存中

windows库的创建和使用:静态库+动态库

windows库的创建和使用:静态库+动态库 一.静态库的创建和使用 1. 静态库创建 (1)首先创建工程test,测试代码如下: 1) test.h void test_print(); 2) test.cpp #include "test.h" #include <stdio.h> void test_print() { printf("test_print in static lib."); } 3)右击工程test:属性-->配置属性--&