转载请注明出处,谢谢 https://www.cnblogs.com/tianknight/p/10648021.html

前言

关于gcc这三个参数,参考了诸多文档后,仍然理解上有偏差,仿照下面博客中的方法,自己调试了一波,总算是理解了。还是建议大家动手实践一下。

参考资料如下:

源码准备

新建三个文件:test.c hello.c world.c ,其源码依赖关系为:test.c 依赖 hello.c;hello.c 依赖 world.c

源码内容

test.c

#include <stdio.h>
void world(void);
void hello(void)
{
printf("hello ");
world();
}

hello.c

#include<stdio.h>
void hello(void);
void main(void)
{
hello();
}

world.c

#include<stdio.h>
void world(void)
{
printf("world.\n");
}

尝试编译,保证源码没有问题

# -o 指定输出文件
[root@localhost testc]# ls
hello.c test.c world.c
[root@localhost testc]# gcc -o hehe *.c
[root@localhost testc]# ls
hehe hello.c test.c world.c
[root@localhost testc]# ./hehe
hello world.

编译

首先编译world.c

# -shared 编译链接库
# -fPIC 我的理解是编译链接库时给代码动态分配内存
[root@localhost testc]# gcc -shared -fPIC -o libworld.so world.c
[root@localhost testc]# ls
hello.c libworld.so test.c world.c
[root@localhost testc]# ldd libworld.so
linux-vdso.so.1 => (0x00007ffd7498f000)
libc.so.6 => /lib64/libc.so.6 (0x00007fcb49815000)
/lib64/ld-linux-x86-64.so.2 (0x00007fcb49dea000) #上述命令和下面的等价,建议选取单条命令即可:gcc -shared -fPIC -o libworld.so world.c
# -c 只激活预处理,编译,和汇编,也就是他只把程序做成obj文件
[root@localhost testc]# gcc -c -fPIC world.c
[root@localhost testc]# ls
hello.c test.c world.c world.o
[root@localhost testc]# gcc -shared -fPIC -o libworld.so world.o
[root@localhost testc]# ls
hello.c libworld.so test.c world.c world.o
[root@localhost testc]# ldd libworld.so
linux-vdso.so.1 => (0x00007ffd0dfa9000)
libc.so.6 => /lib64/libc.so.6 (0x00007f4357dcb000)
/lib64/ld-linux-x86-64.so.2 (0x00007f43583a0000)

编译并链接hello.c

# -lxxx 指定需要动态链接的库文件
# -L 指定动态连接库文件的位置(编译时)
# . 指当前路径
# 下面编译出的libhello.so文件已经显式依赖libworld.so文件,但是没有找到libworld.so的位置
[root@localhost testc]# gcc -shared -fPIC -o libhello.so hello.c -lworld -L.
[root@localhost testc]# ls
hello.c libhello.so libworld.so test.c world.c
[root@localhost testc]# ldd libhello.so
linux-vdso.so.1 => (0x00007ffe61b89000)
libworld.so => not found
libc.so.6 => /lib64/libc.so.6 (0x00007ff73cc1e000)
/lib64/ld-linux-x86-64.so.2 (0x00007ff73d1f4000)

调试编译test.c

[root@localhost testc]# ls
hello.c libhello.so libworld.so test.c world.c # 编译出错,提示找不到hello
[root@localhost testc]# gcc -o haha test.c
/tmp/ccQCWcSW.o: In function 'main':
test.c:(.text+0x5): undefined reference to 'hello'
collect2: error: ld returned 1 exit status # 添加libhello.so的链接索引,并指定库的搜索路径为'.'(当前路径)
# 依然编译失败,提示找不到libworld.so,该库被libhello.so依赖,并提示建议使用-rpath 或 -rpath-link解决
[root@localhost testc]# gcc -o haha test.c -lhello -L.
/usr/bin/ld: warning: libworld.so, needed by ./libhello.so, not found (try using -rpath or -rpath-link)
./libhello.so: undefined reference to 'world'
collect2: error: ld returned 1 exit status # 手动添加libworld.so的依赖,编译通过,查看haha的链接库,已经显式指出依赖,但是没有找到其位置
[root@localhost testc]# gcc -o haha test.c -lhello -L. -lworld
[root@localhost testc]# ldd haha
linux-vdso.so.1 => (0x00007fff556ea000)
libhello.so => not found
libworld.so => not found
libc.so.6 => /lib64/libc.so.6 (0x00007f1ff0c97000)
/lib64/ld-linux-x86-64.so.2 (0x00007f1ff106b000) # 执行编译出的haha,执行报错,提示找不到依赖库
[root@localhost testc]# ./haha
./haha: error while loading shared libraries: libhello.so: cannot open shared object file: No such file or directory #修改系统环境变量'LD_LIBRARY_PATH',增加索引库的位置,查看依赖OK,执行haha, 结果OK, 清空'LD_LIBRARY_PATH'
[root@localhost testc]# export LD_LIBRARY_PATH=/home/testc/ && echo $LD_LIBRARY_PATH
/home/testc/
[root@localhost testc]# ldd haha
linux-vdso.so.1 => (0x00007ffd647d2000)
libhello.so => /home/testc/libhello.so (0x00007fb7aa063000)
libworld.so => /home/testc/libworld.so (0x00007fb7a9e60000)
libc.so.6 => /lib64/libc.so.6 (0x00007fb7a9a8e000)
/lib64/ld-linux-x86-64.so.2 (0x00007fb7aa266000)
[root@localhost testc]# ./haha
hello world.
[root@localhost testc]# export LD_LIBRARY_PATH= && echo $LD_LIBRARY_PATH # 将-lworld 替换为 -Wl,-rpath-link=. ,编译OK,依然找不到索引库,添加LD_LIBRARY_PATH后,执行OK
[root@localhost testc]# gcc -o haha test.c -lhello -L. -Wl,-rpath-link=.
[root@localhost testc]# ldd haha
linux-vdso.so.1 => (0x00007ffdf67c0000)
libhello.so => not found
libc.so.6 => /lib64/libc.so.6 (0x00007fbdbb94b000)
/lib64/ld-linux-x86-64.so.2 (0x00007fbdbbd1f000)
[root@localhost testc]# ./haha
./haha: error while loading shared libraries: libhello.so: cannot open shared object file: No such file or directory
[root@localhost testc]# export LD_LIBRARY_PATH=/home/testc && echo $LD_LIBRARY_PATH
/home/testc
[root@localhost testc]# ldd haha
linux-vdso.so.1 => (0x00007fff89504000)
libhello.so => /home/testc/libhello.so (0x00007fc9e75c6000)
libc.so.6 => /lib64/libc.so.6 (0x00007fc9e71f3000)
libworld.so => /home/testc/libworld.so (0x00007fc9e6ff1000)
/lib64/ld-linux-x86-64.so.2 (0x00007fc9e77c9000)
[root@localhost testc]# ./haha
hello world.
[root@localhost testc]# export LD_LIBRARY_PATH= && echo $LD_LIBRARY_PATH # 将-Wl,-rpath-link=. 换成 -Wl,-rpath=. 编译OK, 查看链接库OK,执行OK
# 修改LD_LIBRARY_PATH后,链接库的位置没有变化
[root@localhost testc]# gcc -o haha test.c -lhello -L. -Wl,-rpath=.
[root@localhost testc]# ls
haha hello.c libhello.so libworld.so test.c world.c
[root@localhost testc]# ldd haha
linux-vdso.so.1 => (0x00007fff86195000)
libhello.so => ./libhello.so (0x00007f4c11254000)
libc.so.6 => /lib64/libc.so.6 (0x00007f4c10e81000)
libworld.so => ./libworld.so (0x00007f4c10c7f000)
/lib64/ld-linux-x86-64.so.2 (0x00007f4c11457000)
[root@localhost testc]# ./haha
hello world.
[root@localhost testc]# export LD_LIBRARY_PATH=/home/testc/ && echo $LD_LIBRARY_PATH
/home/testc/
[root@localhost testc]# ldd haha
linux-vdso.so.1 => (0x00007ffc9f36c000)
libhello.so => ./libhello.so (0x00007f35cf07c000)
libc.so.6 => /lib64/libc.so.6 (0x00007f35ceca9000)
libworld.so => ./libworld.so (0x00007f35ceaa7000)
/lib64/ld-linux-x86-64.so.2 (0x00007f35cf27f000)
[root@localhost testc]# export LD_LIBRARY_PATH= && echo $LD_LIBRARY_PATH

结论

  • 编译时链接库需要分为两类: 直接引用 间接引用
  • 直接引用 被源码中直接调用的库
  • 间接引用 被调用库的依赖库
  • -lxxx 指定具体的库名称,编译时需要显式指定直接引用的库名称
  • -L 指定链接库的位置,编译时需要显式指定直接引用的库位置
  • -Wl,-rpath-link ,用于编译时指定间接引用的库位置

    如果知道所有间接引用的库文件名称,并且不嫌麻烦,也可以用-lxxx显式指定每一个库(不推荐-lxxx)
  • -Wl,-rpath ,有两个作用:

    1. 用于编译时指定间接引用的库位置,作用同-Wl,-rpath-link

    2. 用于运行时指定所有引用库的位置,作用同修改环境变量(LD_LIBRARY_PATH),并且库路径引用优先级高于LD_LIBRARY_PATH
  • 使用建议

    1. 编译命令中使用-Wl,-rpath-link 指定间接引用库位置(编译时),使用-Wl,-rpath 指定引用库位置(运行时)

    2. -Wl,-rpath-link 在 -Wl,-rpath 前

-L -Wl,-rpath-link -Wl,-rpath区别精讲的更多相关文章

  1. GCC 中 -L、-rpath和-rpath-link的区别

    GCC 中 -L.-rpath和-rpath-link的区别 来源 http://blog.csdn.net/q1302182594/article/details/42102961 关于这3个参数的 ...

  2. -L、-rpath和-rpath-link的区别

    链接器ld的选项有 -L,-rpath 和 -rpath-link,看了下 man ld,大致是这个意思: -L::  “链接”的时候去找的目录,也就是所有的 -lFOO 选项里的库,都会先从 -L ...

  3. css link和@import区别

    1.link语法结构 <link href="CSSurl路径" rel="stylesheet" type="text/css" / ...

  4. css link和@import区别用法

    这里link与@import介绍的是html引入css的语法单词.两者均是引入css到html的单词. 1.link语法结构<link rel="stylesheet" ty ...

  5. css学习--inline-block详解及dispaly:inline inline-block block 三者区别精要概括

    *知识储备: 内联元素:是不可以控制宽和高.margin等:并且在同一行显示,不换行. 块级元素:是可以控制宽和高.margin等,并且会换行. 1.inline-block 详解 (1)一句话就是在 ...

  6. 第三百四十四节,Python分布式爬虫打造搜索引擎Scrapy精讲—craw母版l创建自动爬虫文件—以及 scrapy item loader机制

    第三百四十四节,Python分布式爬虫打造搜索引擎Scrapy精讲—craw母版l创建自动爬虫文件—以及 scrapy item loader机制 用命令创建自动爬虫文件 创建爬虫文件是根据scrap ...

  7. 引入css文件时,css link和@import区别

    这里link与@import介绍的是html引入css的语法单词.两者均是引入css到html的单词. 一.了解基本 1.link语法结构 <link href="CSSurl路径&q ...

  8. Linux实战教学笔记12:linux三剑客之sed命令精讲

    第十二节 linux三剑客之sed命令精讲 标签(空格分隔): Linux实战教学笔记-陈思齐 ---更多资料点我查看 1,前言 我们都知道,在Linux中一切皆文件,比如配置文件,日志文件,启动文件 ...

  9. (转)不看绝对后悔的Linux三剑客之grep实战精讲

    不看绝对后悔的Linux三剑客之grep实战精讲 原文:http://blog.51cto.com/hujiangtao/1923675 https://www.cnblogs.com/peida/a ...

随机推荐

  1. Intellij IDEA通过SVN导入基于Springboot的maven项目以及对已有项目做更新

    一.导入外部maven项目 点击“+”,输入SVN地址并下载项目 弹出窗口,选择new window(自己觉得哪个好就选哪个) 等待执行完毕,执行完后会出现以下情况,就需要配置一下你的maven库了 ...

  2. Maximum Product Subarray 最大连续乘积子集

    Find the contiguous subarray within an array (containing at least one number) which has the largest ...

  3. Eclipse 分屏显示同一个代码文件

    描述: 今天在使用Eclipse开发的时候不知按错哪个键,出现编辑框分屏显示同一个代码,由于之前没有使用过这一功能,所以就去查了一下,原来是Eclipse的分屏功能. 快捷键: 方式一:Window ...

  4. Python爬虫教程-07-post介绍(百度翻译)(上)

    Python爬虫教程-07-post介绍(百度翻译)(上) 访问网络两种方法 get: 利用参数给服务器传递信息 参数为dict,使用parse编码 post :(今天给大家介绍的post) 一般向服 ...

  5. Python爬虫教程-04-response简介

    Spider-04-response简介 本小节介绍urlopen的返回对象,和简单调试方法 案例v3 研究request的返回值,输出返回值类型,打印内容 geturl:返回请求对象的url inf ...

  6. volley5--Request<T>类的介绍

    源码: /* * Copyright (C) 2011 The Android Open Source Project * * Licensed under the Apache License, V ...

  7. 在 O(1) 时间删除链表结点(C 和 Python 实现)

    (说明:本博客中的题目.题目详细说明及参考代码均摘自 “何海涛<剑指Offer:名企面试官精讲典型编程题>2012年”) 题目 给定单向链表的头指针和一个结点指针,定义一个函数在 O(1) ...

  8. 适用于 Windows 的虚拟机扩展和功能

    Azure 虚拟机扩展是小型应用程序,可在Azure 虚拟机上提供部署后配置和自动化任务. 例如,如果虚拟机要求安装软件.防病毒保护或进行 Docker 配置,便可以使用 VM 扩展来完成这些任务. ...

  9. SSH 无法启动的原因分析及解决方法

    简介 Secure Shell(缩写为 SSH),由 IETF 的网络工作小组(Network Working Group)所制定:SSH 为一项创建在应用层和传输层基础上的安全协议,为计算机上的 S ...

  10. zabbix_3.0安装部署与中文支持

    Zabbix 3.0界面焕然一新,一改10多年的老面孔,alpha4的更新具体记录如下:http://www.zabbix.com/rn3.0.0alpha4.php What's New in 3. ...