美文网首页
动态库编译详解

动态库编译详解

作者: 放纵的卡尔 | 来源:发表于2022-02-21 01:27 被阅读0次

动态库编译详解:

当前类介绍:upper.c ( upper) 依赖于 bottom.c(play)

1.生成一个动态库

yueqingchan@yueqingchan:~/Desktop/TestGcc-Files/TestDynamic1$ gcc -c upper.c bottom.c -I ./include/
yueqingchan@yueqingchan:~/Desktop/TestGcc-Files/TestDynamic1$ gcc main.c -o main.out -L ./ -lplay -Wl,-rpath ./,ypa
gcc: error: ./,ypa: 没有那个文件或目录
yueqingchan@yueqingchan:~/Desktop/TestGcc-Files/TestDynamic1$ gcc main.c -o main.out -L ./ -lplay -Wl,-rpath ./mypath

2.运行程序并且链接动态库

说明:当执行可执行程序的时候,需要去/lib. /user/lib下和LD_LIBRARY_PATH下寻找so.并不会在当前目录下寻找.

所以执行./main.out会报错.如下:

yueqingchan@yueqingchan:~/Desktop/TestGcc-Files/TestDynamic1$ ./main.out 
./main.out: error while loading shared libraries: libplay.so: cannot open shared object file: No such file or directory

解决方案:指定.so运行搜寻路径

1.-Wl,-rpath ./mypath 加入参数,并且将libplay.so copy到./mypath目录下.

yueqingchan@yueqingchan:~/Desktop/TestGcc-Files/TestDynamic1$ cp libplay.so ./mypath/
yueqingchan@yueqingchan:~/Desktop/TestGcc-Files/TestDynamic1$ ./main.out 
Hello, World!
Bottom--->底层的播放功能
Upper---->上层的封装能力

2.设置LD_LIBRARY_PATH,指定目录.

yueqingchan@yueqingchan:~/Desktop/TestGcc-Files/TestDynamic1$ cd mypath/ && rm libplay.so 
yueqingchan@yueqingchan:~/Desktop/TestGcc-Files/TestDynamic1$ export LD_LIBRARY_PATH=./
yueqingchan@yueqingchan:~/Desktop/TestGcc-Files/TestDynamic1$ ./main.out 
Hello, World!
Bottom--->底层的播放功能
Upper---->上层的封装能力

说明:指定了-Wl,-rpath, 设置LD_LIBRARY_PATH也是可以生效的.并不是说只会去-Wl,-rpath下寻找.

3.动态库链接静态库

yueqingchan@yueqingchan:~/Desktop/TestGcc-Files/TestDynamic1$ gcc -c upper.c bottom.c -I ./include/  
yueqingchan@yueqingchan:~/Desktop/TestGcc-Files/TestDynamic1$ ar rcs -o libbottom.a bottom.o

yueqingchan@yueqingchan:~/Desktop/TestGcc-Files/TestDynamic1$ gcc -fPIC -shared  upper.o   -o libplay.so -L ./ -lbottom 
yueqingchan@yueqingchan:~/Desktop/TestGcc-Files/TestDynamic1$ gcc main.c -o main.out -L ./ -lplay  -I ./include/  -Wl,-rpath ./
yueqingchan@yueqingchan:~/Desktop/TestGcc-Files/TestDynamic1$ ./main.out 
Hello, World!
Bottom--->底层的播放功能
Upper---->上层的封装能力

4.动态库依赖传递的情况

首先生成一个bottom.so,然后用upper.so去依赖bottom.so, 然后main.c 再去依赖upper.so.

yueqingchan@yueqingchan:~/Desktop/TestGcc-Files/TestDynamic1$ gcc -c *.c -I ./include/ &&rm main.o 
yueqingchan@yueqingchan:~/Desktop/TestGcc-Files/TestDynamic1$ gcc -fPIC -shared -o libbottom.so bottom.o 
yueqingchan@yueqingchan:~/Desktop/TestGcc-Files/TestDynamic1$ gcc -fPIC -shared -o libplay.so upper.o -L ./ -lbottom 
yueqingchan@yueqingchan:~/Desktop/TestGcc-Files/TestDynamic1$ gcc main.c -o main.out -L ./  libplay.so 
/usr/bin/ld: warning: libbottom.so, needed by libplay.so, not found (try using -rpath or -rpath-link)
/usr/bin/ld: libplay.so: undefined reference to `play'
collect2: error: ld returned 1 exit status
yueqingchan@yueqingchan:~/Desktop/TestGcc-Files/TestDynamic1$ export LD_LIBRARY_PATH=./
yueqingchan@yueqingchan:~/Desktop/TestGcc-Files/TestDynamic1$ gcc main.c -o main.out -L ./  libplay.so 
yueqingchan@yueqingchan:~/Desktop/TestGcc-Files/TestDynamic1$ ./main.out 
Hello, World!
Bottom--->底层的播放功能
Upper---->上层的封装能力

说明:这里编译的时候直接出错,是因为没有指定搜寻路径,所以无法通过编译.

解决编译问题方案.

1.我们依然采用LD_LIBRARY_PATH的方式可以解决编译和运行的问题.

yueqingchan@yueqingchan:~/Desktop/TestGcc-Files/TestDynamic1$ export LD_LIBRARY_PATH=./
yueqingchan@yueqingchan:~/Desktop/TestGcc-Files/TestDynamic1$ gcc main.c -o main.out -L ./  libplay.so 
yueqingchan@yueqingchan:~/Desktop/TestGcc-Files/TestDynamic1$ ./main.out 
Hello, World!
Bottom--->底层的播放功能
Upper---->上层的封装能力

2.生成libplay的时候,直接指定-Wl,-rpath 给libbottom.可以解决编译不通过的问题.

yueqingchan@yueqingchan:~/Desktop/TestGcc-Files/TestDynamic1$ gcc -fPIC -shared -o libplay.so upper.o -L ./  -Wl,-rpath ./  -lbottom  
yueqingchan@yueqingchan:~/Desktop/TestGcc-Files/TestDynamic1$ ldd libbottom.so 
    linux-vdso.so.1 (0x00007fff45ae3000)
    libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007fd501ef3000)
    /lib64/ld-linux-x86-64.so.2 (0x00007fd502102000)
指定了rpath,在去执行./main.out,此时还需要设置libplay.so 的运行路径,加入rpath=./
yueqingchan@yueqingchan:~/Desktop/TestGcc-Files/TestDynamic1$ gcc main.c -o main.out -L ./  libplay.so 
yueqingchan@yueqingchan:~/Desktop/TestGcc-Files/TestDynamic1$ ./main.out 
./main.out: error while loading shared libraries: libplay.so: cannot open shared object file: No such file or directory
yueqingchan@yueqingchan:~/Desktop/TestGcc-Files/TestDynamic1$ gcc main.c -o main.out -L ./  libplay.so  -Wl,-rpath ./
yueqingchan@yueqingchan:~/Desktop/TestGcc-Files/TestDynamic1$ ./main.out 
Hello, World!
Bottom--->底层的播放功能
Upper---->上层的封装能力

3.依赖所有库

yueqingchan@yueqingchan:~/Desktop/TestGcc-Files/TestDynamic1$ export LD_LIBRARY_PATH=
yueqingchan@yueqingchan:~/Desktop/TestGcc-Files/TestDynamic1$ gcc main.c -o main.out -L ./  libplay.so  libbottom.so
yueqingchan@yueqingChan:~/Desktop/TestGcc-Files/TestDynamic1$ gcc -c *.c -I ./include/
yueqingchan@yueqingChan:~/Desktop/TestGcc-Files/TestDynamic1$ gcc -fPIC -shared -o libbottom.so bottom.o 
yueqingchan@yueqingChan:~/Desktop/TestGcc-Files/TestDynamic1$ gcc -fPIC -shared -o libplay.so  upper.o   -L ./ -lbottom 
yueqingchan@yueqingChan:~/Desktop/TestGcc-Files/TestDynamic1$ gcc main.c -o main.out -L ./   libplay.so  libbottom.so 
yueqingchan@yueqingChan:~/Desktop/TestGcc-Files/TestDynamic1$ ./main.out 
./main.out: error while loading shared libraries: libplay.so: cannot open shared object file: No such file or directory
yueqingchan@yueqingChan:~/Desktop/TestGcc-Files/TestDynamic1$ gcc main.c -o main.out -L ./   libplay.so  libbottom.so  -Wl,-rpath=./
yueqingchan@yueqingChan:~/Desktop/TestGcc-Files/TestDynamic1$ ./main.out 
./main.out: error while loading shared libraries: libbottom.so: cannot open shared object file: No such file or directory
yueqingchan@yueqingChan:~/Desktop/TestGcc-Files/TestDynamic1$ export LD_LIBRARY_PATH=./
yueqingchan@yueqingChan:~/Desktop/TestGcc-Files/TestDynamic1$ ./main.out 
Hello, World!
Bottom--->底层的播放功能
Upper---->上层的封装能力

依赖所有库只能解决编译问题,无法处理运行的路径.

另一种思路:我们在执行main.out的时候 执行-Wl,-rpath.并不在生成libplay的时候指定,看下是否正常.

yueqingchan@yueqingchan:~/Desktop/TestGcc-Files/TestDynamic1$ gcc -fPIC -shared -o libplay.so upper.o -L ./  libbottom.so 
yueqingchan@yueqingchan:~/Desktop/TestGcc-Files/TestDynamic1$ gcc main.c -o main.out -L ./  libplay.so  
/usr/bin/ld: warning: libbottom.so, needed by libplay.so, not found (try using -rpath or -rpath-link)
/usr/bin/ld: libplay.so: undefined reference to `play'
collect2: error: ld returned 1 exit status
yueqingchan@yueqingchan:~/Desktop/TestGcc-Files/TestDynamic1$ gcc main.c -o main.out -L ./  libplay.so  -Wl,-rpath ./
yueqingchan@yueqingchan:~/Desktop/TestGcc-Files/TestDynamic1$ ./main.out 
./main.out: error while loading shared libraries: libbottom.so: cannot open shared object file: No such file or directory

由此可见,-Wl,-rpath 只能针对直接依赖的libplay.so指定了路径,但是libbottom还是无法查找到 .但是LD_LIBRARY是可以的.

rpath只能对直接依赖的so设置搜寻目录,并且可以设置所有依赖的编译路径.

总结: 解决编译问题,在生成libplay的时候指定-Wl,-rpath运行路径,或者设置LD_LIBRARAY_PATH,都可以解决这个问题.

      或者直接依赖所有的.so,但是这样是不合理的,我们不需要去依赖间接的依赖.

5.动态库依赖

当我们现在拥有的so包含一个直接依赖的so和很多间接依赖的so,但是没有设置rpath.所以是不能直接依赖主so进行编译和运行的.

为了通过编译:

1.在只链接主so的情况下可以去设置rpath或者LD_LIBRARY_PATH.

2.或者链接所有so.

为了通过运行:

为了正常运行可以设置LD_LIBRARY_PATH.

6.如果我们想只去依赖一个直接so,并且不让使用方主动去设置LD_LIBRARY_PATH就可以直接运行程序,还有一种方式:

--disable-new-dtags,--copy-dt-needed-entries

yueqingchan@yueqing-Chan:~/Desktop/TestGcc-Files/TestDynamic1$ export LD_LIBRARY_PATH=
yueqingchan@yueqing-Chan:~/Desktop/TestGcc-Files/TestDynamic1$ gcc main.c -o main.out -Wl,-rpath=./,--disable-new-dtags,--copy-dt-needed-entries -L ./ -lplay
yueqingchan@yueqing-Chan:~/Desktop/TestGcc-Files/TestDynamic1$ ./main.out 
Hello, World!
Bottom--->底层的播放功能
Upper---->上层的封装能力
yueqingchan@yueqing-Chan:~/Desktop/TestGcc-Files/TestDynamic1$ 

结论概述:

1.我们在生成间接依赖的库的时候,为了保证其他库可以直接依赖,需要加入-Wl,-rpath.保证编译通过.

2.LD_LIBRARY_PATH可以解决一切编译运行问题.

相关文章

  • iOS静态库及动态库

    2018-06-19 浅析静态库链接原理编译原理以及什么是静态库和动态库 转之: iOS 静态库和动态库(库详解)...

  • 动态库编译详解

    动态库编译详解: 当前类介绍:upper.c ( upper) 依赖于 bottom.c(play) 1.生成一个...

  • c++编译跨平台动态库

    window编译动态库 linux编译动态库 makefile文件

  • android JNI 报错:error: undefined

    编译报错 报错原因:jnigraphics库没有动态链接 解决 Cmake编译解决添加动态链接库 jnigraph...

  • 5月6日:gcc编译选项

    一、GCC 动态库处理 ldd XX ###编译通过,但运行时发现不了动态库 image.png编译时...

  • MachO动态库绑定过程详解

    动态库绑定过程详解 首先动态库绑定分为lazy bind 和no_lazy_bind,lazy bind 主要用于...

  • Android FFmpeg so库编译 支持Https

    概要 编译armv7架构的openssl、ffmpeg的a、so库,目前openssl只编译了静态库,编译动态库,...

  • 静态库

    静态库 静态库动态库简介 .a 静态库:被编译好的二进制文件 静态库:.a, .framework 动态库: .d...

  • Linux动态库

    深入探讨Linux静态库与动态库的详解(一看就懂)Linux下动态库的生成和使用Linux下动态库生成和使用转-u...

  • iOS开发-动态库加载(实时模块更新)

    动态库和静态库 静态库和动态库是相对编译期和运行期:静态库在程序编译时会被链接到目标代码中,程序运行时将不再需要改...

网友评论

      本文标题:动态库编译详解

      本文链接:https://www.haomeiwen.com/subject/yjsblrtx.html