Linux C 编程入门之一:gcc 编译动态库和静态库
cheungmine
2012
1 准备工作
Windows7+Cygwin+gcc
在同一个目录下准备好下面3个文件,其中3-2,3-3用来生成动态库或静态库:
主调用程序源代码3-1:main.c
/**
* main.c
*/
#include <stdio.h>
#include <math.h>
#include "hello_fn.h"
int
main ()
{
hello("cheungmine");
printf("sqrt(2.0) = %f\n", sqrt(2.0));
return 0;
}
库源代码3-2:hello_fn.c/**
* hello_fn.c
*/
#include <stdio.h>
void hello(const char *msg)
{
printf("Hello %s!\n", msg);
}
库头文件源代码3-3:hello_fn.h
/**
* hello_fn.h
*/
void hello(const char *msg);
2 编译库
2.1 首先编译源文件生成对象(obj)文件(main.o, hello_fn.o):
$ gcc -W -Wall -ansi -pedantic -I. -c main.c
$ gcc -W -Wall -ansi -pedantic -I. -c hello_fn.c
2.2 然后从对象文件编译动态库文件(libhello.so)和静态库(libhello.a)
$ gcc -shared hello_fn.o -o libhello.so
或者直接从源代码编译:
$ gcc -shared -I. hello_fn.c -o libhello.so
编译静态库相对简单,就是相当于目标文件归档:
$ ar r libhello.a hello_fn.o
3 编译使用库的主程序
3.1 如果不链接库的情况下编译一个主程序是:
$ gcc main.o -o main
或者
$ gcc -c main.c -o main
但是由于我们在main.c的代码中写固定了调用库的代码(hello函数),所以,必须链接到这个库才行。
3.2 链接到动态库libhello.so
$ gcc main.o -o main ./libhello.so
这样在当前目录下就生成了:main.exe(我的cygwin环境,Linux环境下没有扩展名)
运行这个main.exe:
$ ./main.exe
删除libhello.so,再运行main.exe会报错误:error while loading shared libraries: libhello.so: cannot open shared object...
3.3 链接到静态库libhello.a
$ gcc main.o -o main2 ./libhello.a
删除libhello.a,运行main2.exe,一切正常。说明程序的确链接到静态库了。
4 查看程序依赖的库
$ file main.exe main2.exe
$ ldd main.exe main2.exe
如果我们的动态库libhello.so与主程序不在同一个目录下,怎么办?
复制libhello.so和libhello.a到另外一个目录,比如:/cygdrive/c/temp,那么编译主程序为:
$ gcc main.o -o main /cygdrive/c/temp/libhello.so
执行:
$ export PATH=/cygdrive/c/temp:$PATH
$ ./main.exe
5 运行时加载动态库
修改main.c文件为如下清单:
/**
* main.c
*/
#include <stdio.h>
#include <math.h>
#include <dlfcn.h>
#include "hello_fn.h"
void dl_hello()
{
void *dp;
char *error;
void (*fn_hello)(const char*);
dp = dlopen("libhello.so", RTLD_LAZY );
if(!dp) {
printf("%s\n", dlerror());
exit(1);
}
fn_hello = dlsym(dp, "hello");
if(!fn_hello) {
printf("%s\n", dlerror());
exit(1);
}
fn_hello("cheungmine: load library when running");
dlclose(dp);
}
int
main ()
{
// hello("cheungmine");
dl_hello();
printf("sqrt(2.0) = %f\n", sqrt(2.0));
return 0;
}
然后重新编译main.exe和libhello.so如下:
编译源文件
$ gcc -Wall -I. -c main.c
$ gcc -Wall -I. -c hello_fn.c
编译动态库
$ gcc -shared hello_fn.o -o libhello.so
链接主程序,但不链接到动态库。
$ gcc main.o -o main.exe
执行下面的代码可以看到libhello.so并不在main.exe的依存之列:
$ ldd main.exe
移动库到其他目录,通过修改环境变量,程序main.exe执行正确:
$ mv libhello.so /cygdrive/c/temp
$ export PATH=.:/cygdrive/c/temp:$PATH
$ ./main.exe
6 总结
通过上面的练习,基本清楚了如何用gcc编译程序,包括静态链接库和动态链接库。通过下面的表格可以看到
Linux和Windows的区别:
Windows Unix/Linux
----------------------------------------------------------------------
静态链接库 hello.lib libhello.a
动态链接库 hello.dll libhello.so
延迟加载 LoadLibrary dlopen
GetProcAddress dlsym
FreeLibrary dlclose
本文全部内容在cygwin上运行的,和真实的Linux环境还是存在差异的。gcc版本3.4.4。
分享到:
相关推荐
GCC编译动态和静态链接库.pdf 学习资料 复习资料 教学资源
Linux系统下使用gcc 5.3编译器编译的boost库1.68版本的动态库和静态库,多线程参数编译,经测试可用
默认编译是静态库,但考虑到 linux 上动态库使用较多,所以使用 -DBUILD_SHARED_LIBS=ON 参数编译为动态库。 在 centos 7 下使用 gcc 4.8.5 + cmake 3.16.9 编译,包含 bin, include, lib, lib64, share 五个目录,...
转载的gcc编译方法,开发时老是忘记,作为资源吧,省得找了。
函数库分为静态库和动态库两种。 静态库在程序编译时会被连接到目标代码中,程序运行时将不再需要该静态库。动态库在程序编译时并不会被连接到目标代码中,而是在程序运行是才被载入,因此在程序运行时还需要动态库...
Linux 如何使用gcc生成静态库和动态库,使用GCC编译生成静态库和动态库的方法
本代码是在VS2005下编译产生的CJson为静态库文件,编译测试通过,方便大家使用(看了网上很多的例程都不能使用,自动动手编写的),本编译的文件是CJSON的库文件cjson.c,可以去官方下载
1、二者均需要由.o文件生成,二者前缀均为lib,静态库后缀以 .a,动态库的后缀以.so 2、生成目标可执行文件后,静态库可以直接删除,因为静态库已经被加载到可...3、当静态库和动态库同名时, gcc命令将优先使用动态库
根据链接时期的不同,库又有静态库和动态库之分。 静态库是在链接阶段被链接的(好像是废话,但事实就是这样),所以生成的可执行文件就不受库的影响了,即使库被删除了,程序依然可以成功运行。 有别于静态库,动态...
静态库的制作:(inc/lib/src/main.c) gcc -c src/*.c(编译src目录下的.c文件生成.o文件) ar rs libmath.a *.o(生成静态库:libmath.a(命名规则:必须以lib开头,以.a结尾,中间是库名:math)) 静态库的使用: gcc main.c -...
在windows环境下,我们通常在IDE如VS的工程中开发C++项目,对于生成和使用静态库(*.lib)与动态库(*.dll)可能都已经比较熟悉,但是,在linux环境下,则是另一套模式,对应的静态库(*.a)与动态库(*.so)的生成...
ubuntu12.04 32位下gcc4.6.3编译glibc-2.15版本静态库文件,包内一个源码,一个编译后
gcc编译动态和静态链接库.pdf,教你如何编译和使用库,提高程序的理解力
linux下编译/交叉编译openssl-1.1.1g生成动态库和静态库方法: 一、编译linux版本 1. cp openssl-1.1.1g.tar.gz /opt/ 2. tar -zxvf openssl-1.1.1g.tar.gz 3. mkdir build 4. ./config --prefix=/opt/openssl-1.1.1...
1.编译生成静态库 gcc -c libdemo.c -o libdemo.o ar rcs libdemo.a libdemo.o gcc testdemo.c -o testdemo -static -L. -ldemo 运行: ./testdemo
默认编译是静态库,但考虑到 linux 上动态库使用较多,所以使用 -DBUILD_SHARED_LIBS=ON 参数编译为动态库。 在 ubuntu 下使用 gcc 4.8.5 + cmake 3.28 编译,包含 bin, include, lib, lib64, share 五个目录,可以...
源码经过编译生成对象文件,对象文件进一步处理可以生成可执行程序或者库文件。...对于可执行程序和动态库,此“进一步处理”指的是“链接”,使用gcc命令。 对于静态库,此“进一步处理”指的是“打包”,使用ar命令。
zlib源代码以及编译好的win32动态库静态库,支持win32工程。直接使用。
动态库的制作: 方法一: gcc -c -fPIC add.c sub.c div.c mul.c //-c表示生成.o目标文件,-f后加一些编译选项,PIC表示与位置无关 gcc -shared -o libmymath.so add.o sub.o mul.o div....