- 浏览: 148164 次
- 性别:
- 来自: 北京
文章分类
- 全部博客 (110)
- CoreJava (18)
- 待解决 (1)
- 数据结构 (3)
- 科普 (2)
- 网站 (1)
- DataBase (8)
- Access (1)
- Asp (1)
- JSP (1)
- 操作系统 (8)
- Tech (1)
- Linux (11)
- Career (5)
- MongoDB (1)
- Embedded (1)
- JavaScript (1)
- UltraIso (1)
- Linux命令 (1)
- DesignPattern (1)
- Ruby (13)
- Vim (1)
- 自考 (2)
- Github (5)
- zrProject (1)
- Emacs (4)
- Math (3)
- Ajax (1)
- 没看懂 (1)
- HTML (1)
- Philosophy (1)
- 软件 (1)
- 面试 (1)
- 考试报名 (3)
- Regex (1)
- 日语 (1)
- 生活 (1)
最新评论
Linux C 编程入门之一:gcc 编译动态库和静态库
cheungmine
2012
1 准备工作
Windows7+Cygwin+gcc
在同一个目录下准备好下面3个文件,其中3-2,3-3用来生成动态库或静态库:
主调用程序源代码3-1:main.c
[cpp] view plaincopyprint?
/**
* 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
[cpp] view plaincopyprint?
/**
* hello_fn.c
*/
#include <stdio.h>
void hello(const char *msg)
{
printf("Hello %s!\n", msg);
}
库头文件源代码3-3:hello_fn.h
[cpp] view plaincopyprint?
/**
* hello_fn.h
*/
void hello(const char *msg);
2 编译库
2.1 首先编译源文件生成对象(obj)文件(main.o, hello_fn.o):
[plain] view plaincopyprint?
$ 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)
[plain] view plaincopyprint?
$ 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 如果不链接库的情况下编译一个主程序是:
[plain] view plaincopyprint?
$ gcc main.o -o main
或者
$ gcc -c main.c -o main
但是由于我们在main.c的代码中写固定了调用库的代码(hello函数),所以,必须链接到这个库才行。
3.2 链接到动态库libhello.so
[plain] view plaincopyprint?
$ gcc main.o -o main ./libhello.so
这样在当前目录下就生成了:main.exe(我的cygwin环境,Linux环境下没有扩展名)
运行这个main.exe:
[plain] view plaincopyprint?
$ ./main.exe
删除libhello.so,再运行main.exe会报错误:error while loading shared libraries: libhello.so: cannot open shared object...
3.3 链接到静态库libhello.a
[plain] view plaincopyprint?
$ gcc main.o -o main2 ./libhello.a
删除libhello.a,运行main2.exe,一切正常。说明程序的确链接到静态库了。
4 查看程序依赖的库
[plain] view plaincopyprint?
$ file main.exe main2.exe
$ ldd main.exe main2.exe
如果我们的动态库libhello.so与主程序不在同一个目录下,怎么办?
复制libhello.so和libhello.a到另外一个目录,比如:/cygdrive/c/temp,那么编译主程序为:
[plain] view plaincopyprint?
$ gcc main.o -o main /cygdrive/c/temp/libhello.so
执行:
$ export PATH=/cygdrive/c/temp:$PATH
$ ./main.exe
5 运行时加载动态库
修改main.c文件为如下清单:
[cpp] view plaincopyprint?
/**
* 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如下:
[cpp] view plaincopyprint?
编译源文件
$ 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
发表评论
-
find 与 rm
2013-03-12 11:02 927查找并删除所有不以.zip结尾的所有文件 find ... -
gcc 编译动态库和静态库 2
2012-07-09 10:28 1220在windows下动态链接库是以.dll后缀的文件,而在 ... -
Linux 配置网关服务器详解
2012-03-19 14:27 1713http://carywu.blog.51cto. ... -
使用debian配置网关服务器及内网连入IPV6
2012-03-19 14:23 1022http://mail.ustc.edu.cn/~lixueb ... -
禁用触摸板
2012-03-04 12:23 574sudo rmmod psmouse 这个是禁用的 s ... -
环境变量
2012-03-04 10:56 7841、引言 ... -
Grub修改入门
2012-02-27 23:04 4561如何设定开机启动项ubuntu 10.10 grub ... -
千奇百怪的分区表错误 - 光盘安装只能看到整个硬盘而无法看到分区的同学请过来瞧瞧。
2012-01-28 22:56 1198最近分区表出问题的人好像很多啊,我把以前写的一些帖子翻出来总 ... -
Ubuntu虚拟机 NAT上网
2012-01-23 19:32 751一、首先设置上网的方式:NAT 二、编辑连 ... -
虚拟机安装Ubuntu的上网设置(有线网络和无线网络)
2012-01-23 19:31 16148虚拟机下ubuntu共享方式上网: 一. 有线网络 ...
相关推荐
GCC 编译动态和静态链接库 GCC 编译动态和静态链接库是 Linux 应用开发中的一个重要组件,对于嵌入式 Linux 应用开发来说尤为重要。本文将详细介绍如何在 Linux 中创建静态库和动态库,以及使用它们。 静态库 ...
本篇将详细介绍在Linux环境下如何编译动态库和静态库。 首先,我们来看动态库的生成过程。动态库的编译通常分为两个步骤: 1. 编译源文件为`.o`对象文件: 使用`cpp`或`gcc`命令,加上`-c`选项来编译源代码,并...
这将编译动态库(`.so`)和静态库(`.a`),并安装到指定的路径。 5. **测试与验证**:编译完成后,为了确保库的可用性,可以运行Boost的测试套件,用`bjam test`执行所有测试案例。如果所有测试都通过,那么说明库...
# 使用gcc编译为动态库 gcc -shared -o libhello.so hello.c ``` 至此,我们已经成功地创建了一个静态库`libhello.a`和一个动态库`libhello.so`。接下来可以根据实际需求将这些库用于其他项目中。 以上便是关于...
### Linux下Gcc生成和使用静态库和动态库详解 #### 一、基本概念 **1.1 什么是库** 库本质上是一种可执行代码的二进制形式,它可以被操作系统载入内存执行。无论是Windows还是Linux平台,都广泛地使用着库。然而...
默认编译是静态库,但考虑到 linux 上动态库使用较多,所以使用 -DBUILD_SHARED_LIBS=ON 参数编译为动态库。 在 centos 7 下使用 gcc 4.8.5 + cmake 3.16.9 编译,包含 bin, include, lib, lib64, share 五个目录,...
在项目属性页中,导航到“配置属性” > “C/C++” > “预处理器”,可能需要添加预处理器定义,例如`CJSON_EXPORTS`,这取决于你是编译动态库还是静态库。 5. **编译项目**:现在你可以编译项目。如果一切设置正确...
总之,理解如何在Linux下使用GCC编译动态库对于软件开发者而言至关重要,因为它可以帮助我们构建可扩展、高效且易于维护的软件。通过掌握创建和使用库的技巧,开发者可以更好地利用现有的开源工具和组件,提高开发...
本篇将详细解析如何在CentOS 7环境下编译libwebsockets动态库和静态库,并探讨其在实际应用中的价值。 首先,让我们了解一下libwebsockets库。libwebsockets是一个C语言编写的WebSocket库,它支持HTTP和WebSocket...
然后,进入OpenCV源码目录,并根据需要选择编译动态库或静态库。动态库的编译命令如下: ```bash cmake -DSOFTFP=ON -DCMAKE_TOOLCHAIN_FILE=./platforms/linux/arm-gnueabi.toolchain.cmake -B build_arm ``` 静态...
Linux 使用 GCC 生成静态库和动态库 在 Linux 系统中,使用 GCC 编译器可以生成静态库和动态库。静态库和动态库是两种不同的库类型,它们的主要区别在于代码被载入的时间不同。 静态库是在编译时被载入到可执行...
如果要同时编译动态库和静态库,只需运行不带`link`和`runtime-link`选项的`b2`命令即可。 安装完成后,别忘了更新你的编译器链接路径,这样其他程序就可以找到新编译的库: ```bash echo 'export LD_LIBRARY_PATH...
在嵌入式开发领域,了解如何使用gcc工具链来构建和管理静态库(`.a`文件)与动态库(`.so`文件)是非常重要的。本文将详细介绍如何通过gcc创建这两种类型的库,并讨论它们各自的用途、优缺点以及在实际项目中的应用...
4. **配置编译**:运行`./config`或`./Configure`命令,根据需要选择目标平台和选项,例如`no-shared`表示编译静态库,`shared`则表示编译动态库。 5. **编译和安装**:执行`make`命令进行编译,然后使用`make ...
在windows环境下,我们通常在IDE如VS的工程中开发C++项目,对于生成和使用静态库(*.lib)与动态库(*.dll)可能都已经比较熟悉,但是,在linux环境下,则是另一套模式,对应的静态库(*.a)与动态库(*.so)的生成...
"gcc静态库和动态库生成" 本文主要介绍如何使用gcc生成静态库和动态库,并使用它们。在Linux中,我们通常将一些公用函数制作成函数库,供其他程序使用。函数库分为静态库和动态库两种。 一、静态库 静态库在程序...
在Linux系统中,动态库(Dynamic Library)和静态库(Static Library)是程序开发中不可或缺的部分,它们提供了代码复用和模块化的功能。本篇文章将深入探讨这两种库的创建、使用以及它们之间的区别。 首先,我们来...