目录

1. 编写C程序

  • 比如编写myfunc.c文件,里面包含两个函数,一个是say_hello,另一个是cal_sum
#include "myfunc.h"

void say_hello()
{
printf("hello world\n");
} int cal_sum(int x, int y)
{
return x + y;
}
  • myfunc.c编写接口文件
#ifndef __MY_FUNC_H
#define __MY_FUNC_H #include <stdio.h>
#include <stdlib.h> void say_hello();
int cal_sum(int x, int y); #endif

2. 编译动态链接库

  • 首先编译myfunc.c
gcc -c -fPIC -o myfunc.o myfunc.c

-c 表示只编译(compile),而不链接,输出目标文(obj文件)。

-o 表示输出文件的文件名。

-fPIC PIC指Position Independent Code, 生成适合在共享库中使用的与位置无关的代码。编译成共享库要求此选项。适用于动态链接并避免对全局偏移表大小的任何限制。

  • 生成共享库文件libmyfunc.so
gcc -shared myfunc.o -o libmyfunc.so

-share 生成一个共享对象,然后可以与其他对象链接以形成可执行文件。

两条命令合成一条就是:

gcc -fPIC -shared myfunc.c -o libmyfunc.so

3. 使用共享库

接下来我们使用test.c来调用共享库。test.c内容如下:

#include "myfunc.h"

int main(int argc, char const *argv[])
{
int result = 0; say_hello();
result = cal_sum(2, 3);
printf("%d\n", result); return 0;
}

编译上述包含.h头文件的程序,GCC编译器需要知道头文件的位置

  • 对于#include <...>,GCC编译器会在默认include搜索路径中寻找。
  • 对于#include "...",GCC编译器会在当前路径搜索.h文件。你也可以使用-I选项提供额外的搜索路径,比如-I /home/test/

除此之外,GCC编译器还需要知道我们用了哪个库文件,库文件在哪里

  • 使用-l选项说明库文件的名字。这里,我们使用的是libmyfunc.so库文件,所以选项是这样写的:-l myfunc
  • 使用-L选项说明库文件的路径。这里,我们的库文件是在当前路径,所以选项是这样写的:-L .(.表示当前路径)。

所以,最终我们链接库文件生成的可执行文件命令是这样的:

gcc -o test test.c -l myfunc -L .

附加:

可以使用下面的命令,来获知系统的include默认搜索路径:

$ gcc -print-prog-name=cc1 -v

获知库默认搜索路径:

$ gcc -print-search-dirs

4. 执行程序

$ ./test

执行程序后发现出现这样的情况:

./test: error while loading shared libraries: libmyfunc.so: cannot open shared object file: No such file or directory

这是因为执行程序的时候,操作系统不知道libmyfunc.so的位置,系统无法找到libmyfunc.so库文件。尽管我们用GCC编译的时候,用-L选项提供了libmyfunc.so文件的位置,但是这个信息并没有被写入到可执行程序里面。下面用命令ldd命令测试(ldd命令是用于显示可执行文件所依赖的库):

$ ldd test

    linux-vdso.so.1 =>  (0x00007ffccc9fe000)
libmyfunc.so => not found
libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007f0d31a44000)
/lib64/ld-linux-x86-64.so.2 (0x00007f0d31e0e000)

可以看出可执行文件test无法找到libmyfunc.so库文件。

解决办法有几个:

  1. libmyfunc.so放到gcc默认搜索目录,比如/usr/lib/x86_64-linux-gnu或者/lib/x86_64-linux-gnu都可以,这样做简单粗暴。但要是这样做的话,需要root权限来完成,除此之外,我感觉污染了整个系统。

  2. /etc/ld.so.conf.d目录下新建一个.conf文件,比如mylib.conf,在里面添加第三方库(libmyfunc.so)目录路径

  3. 设置LD_LIBRARY_PATH环境变量,比如export LD_LIBRARY_PATH=.。设置这个环境变量之后,操作系统将在先在LD_LIBRARY_PATH下搜索库文件,再到默认路径中搜索文件。这样,可执行文件就可以在LD_LIBRARY_PATH中找到第三方库(libmyfunc.so)的位置。

  4. 编译的时候添加-Wl,-rpath选项,比如gcc -o test test.c -l myfunc -L . -Wl,-rpath=.-Wl选项告诉编译器将后面的参数传递给链接器

最后运行可执行文件test的结果:

$ gcc -o test test.c -l myfunc -L . -Wl,-rpath=.
$ ./test
hello world
5

5. 参考资料

  1. C编译: 动态连接库 (.so文件)
  2. Linux动态库生成与使用指南
  3. Linux 动态库剖析
  4. Linux动态链接
  5. 运行时动态库:not found 及介绍-linux的-Wl,-rpath命令

Linux动态链接库的生成和使用的更多相关文章

  1. 【ARM-Linux开发】linux下Eclipse进行C编程时动态链接库的生成和使用

    linux下Eclipse进行C编程时动态链接库的生成和使用 引用 http://linux.chinaitlab.com/soft/864157.html 欢迎进入Linux社区论坛,与200万技术 ...

  2. Linux动态链接库的创建与使用

    Linux动态链接库的创建与使用1. 介绍     使用GNU的工具我们如何在Linux下创建自己的程序函数库?一个“程序函数库”简单的说就是一个文件包含了一些编译好的代码和数据,这些编译好的代码和数 ...

  3. Linux静态库生成指南

    Linux静态库生成指南   Linux上的静态库,其实是目标文件的归档文件.在Linux上创建静态库的步骤如下: 写源文件,通过 gcc -c xxx.c 生成目标文件. 用 ar 归档目标文件,生 ...

  4. Linux Makefile自动生成--config.h

    Linux Makefile自动生成--config.h http://blog.csdn.net/spch2008/article/details/12510805

  5. 在 Linux 中自动生成 Cordova/Phonegap for Android 的 APK 安装程序

    在 Linux 中自动生成 Cordova/Phonegap for Android 的 APK 安装程序 本贴首发于: http://xuekaiyuan.com/forum.php?mod=vie ...

  6. Linux下随机生成密码的命令总结

    有时候经常为如何设置一个安全.符合密码复杂度的密码而绞尽脑汁,说实话,这实在是一个体力活而且浪费时间,更重要的是设置密码的时候经常纠结.终于有一天实在忍不住了,于是学习.整理了一下如何使用Linux下 ...

  7. 使用linux perf工具生成java程序火焰图

    pre.cjk { font-family: "Nimbus Mono L", monospace } p { margin-bottom: 0.1in; line-height: ...

  8. VS2010动态链接库的生成及调用(C++)

    一.动态链接库的生成 首先利用VS2010新建一个空的工程或者win32工程 2.在工程中添加头文件和源文件 3.工程属性配置 3.1 可以在解决方案目录下新建以下几个文件夹 bin (用于存放Rel ...

  9. Linux动态库生成与使用指南

    相关阅读: Linux静态库生成指南 Linux下动态库文件的文件名形如 libxxx.so,其中so是 Shared Object 的缩写,即可以共享的目标文件. 在链接动态库生成可执行文件时,并不 ...

随机推荐

  1. 求出100以内的素数(java实现)

    j package test1; //2018/11/30 //求100以内的所有素数 public class Main10 { public static void main(String[] a ...

  2. Carthage入门篇-安装和使用

    在iOS开发过程中,用到最多的三方库管理工具也许是Cocoapods.Cocoapods会自动为你创建一个workspace,然后自动将你要用到的三方库集成到Project中,而整个过程,你只需要对P ...

  3. Docker平台的基本使用方法

    1.运行一个 container并加载镜像centos,运行起来这个实例后,在实例中执行 /bin/bash命令 docker常用参数: run  运行 -i  以交互模式运行容器,通常与 -t 同时 ...

  4. Python----简单线性回归

    简单线性回归 1.研究一个自变量(X)和一个因变量(y)的关系   简单线性回归模型定义:y=β0+β1x+ε 简单线性回归方程:E(y)=β0+β1x 其中: β0为回归线的截距 β1为回归线的斜率 ...

  5. Mysql 中的日期时间字符串查询

    一.将数据库中的Date格式的数据,或者指定日期的字符串格式化为想要的样式 DATE_FORMAT (date, format)能根据格式串format 格式化日期或日期和时间值date,返回结果字符 ...

  6. Servlet 易错点和注意点

    目录 @WebServlet("/")与@WebServlet("/*")的区别 @WebServlet("/")与@WebServlet( ...

  7. Python scrapy爬虫数据保存到MySQL数据库

    除将爬取到的信息写入文件中之外,程序也可通过修改 Pipeline 文件将数据保存到数据库中.为了使用数据库来保存爬取到的信息,在 MySQL 的 python 数据库中执行如下 SQL 语句来创建 ...

  8. JS JSON对象相关

    1.多对象合并 将2个或2个以上对象(object{....})中的属性进行合并,即最后合并为一个object{.....} 解决办法:Object.assign 方法 var form = {nam ...

  9. 用redis实现分布式锁,秒杀案例(转)

    分布式锁的简单实现代码: 需要的jar包: jedis-2.9.0.jar. commons-pool2-2.4.2.jar import java.util.List; import java.ut ...

  10. React Native & ES6 & emoji

    React Native & ES6 & emoji && 逻辑运算符 https://developer.mozilla.org/zh-CN/docs/Web/Jav ...