- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
假设共享库A 依赖于另一个共享库B。
在这种情况下,当我构建A时,是否只需要B的头文件?因为只有当我构建一些需要 A 的程序时才需要 B。这样对吗?你能解释一下细节吗?
最佳答案
你是对的。这是一个具体的例子。
啊.h
#ifndef A_H
#define A_H
extern void aa(void);
#endif
交流
#include "a.h"
#include "b.h"
void aa(void)
{
bb();
}
b.h
#ifndef B_H
#define B_H
extern void bb(void);
#endif
BC
#include "b.h"
#include <stdio.h>
void bb(void)
{
puts(__func__);
}
main.c
#include "a.h"
int main(void)
{
aa();
return 0;
}
我们制作了一个共享库liba.so
。先编译一个PIC (Position Independent)目标文件。
$ gcc -Wall -Wextra -o a.o -c -fPIC a.c
现在目标文件 a.o
包含对 bb
的 undefined reference :
$ readelf -s a.o
Symbol table '.symtab' contains 11 entries:
Num: Value Size Type Bind Vis Ndx Name
...
10: 0000000000000000 0 NOTYPE GLOBAL DEFAULT UND bb
链接共享库:
$ gcc -shared -o liba.so a.o
现在共享库还对 bb
进行了 undefined reference :
$ readelf --dyn-syms liba.so
Symbol table '.dynsym' contains 12 entries:
Num: Value Size Type Bind Vis Ndx Name
...
2: 0000000000000000 0 NOTYPE GLOBAL DEFAULT UND bb
...
没关系。 链接器将创建一个包含 undefined reference 的共享库。
以同样的方式制作另一个共享库libb.so
:
$ gcc -Wall -Wextra -o b.o -c -fPIC b.c
$ gcc -shared -o libb.so b.o
这个共享库定义了bb
:
$ readelf --dyn-syms libb.so
Symbol table '.dynsym' contains 12 entries:
Num: Value Size Type Bind Vis Ndx Name
...
11: 000000000000060a 19 FUNC GLOBAL DEFAULT 12 bb
接下来尝试做一个程序:
$ gcc -Wall -Wextra -o main.o -c main.c
$ gcc -o prog main.o
main.o: In function `main':
main.c:(.text+0x5): undefined reference to `aa'
collect2: error: ld returned 1 exit status
当我们既不链接 liba
也不链接 libb
时就会发生这种情况。链接器将不要创建包含 undefined reference 的程序。所以:
$ gcc -o prog main.o liba.so
liba.so: undefined reference to `bb'
collect2: error: ld returned 1 exit status
这就是我们链接 liba
而不是 libb
时发生的情况。如果我们链接两者都像:
$ gcc -o prog main.o liba.so libb.so
成功了!但是要小心。如果我们交换库的顺序:
$ gcc -o prog main.o libb.so liba.so
liba.so: undefined reference to `bb'
collect2: error: ld returned 1 exit status
联动再次失败。也是如此:
$ gcc -o prog libb.so liba.so main.o
main.o: In function `main':
main.c:(.text+0x5): undefined reference to `aa'
collect2: error: ld returned 1 exit status
链接器需要在其他之后看到一个库依赖于它的库或目标文件。所以main.o
必须链接在liba
之前,和 liba
必须在 libb
之前链接。
还有最后一个问题。
$ gcc -o prog main.o liba.so libb.so
或等同于:
$ gcc -o prog main.o -L. -la -lb
成功链接程序 prog
,但是:
$ ./prog
./prog: error while loading shared libraries: liba.so: cannot open shared object file: No such file or directory
它不会运行。因为运行时加载器仍然不知道在哪里寻找 liba
或 libb
加载器知道 prog
需要一些名为 liba.so
和 libb.so
的共享库,因为链接器已经写好了信息到 prog
:
$ readelf -d prog
Dynamic section at offset 0xda8 contains 29 entries:
Tag Type Name/Value
0x0000000000000001 (NEEDED) Shared library: [liba.so]
0x0000000000000001 (NEEDED) Shared library: [libb.so]
0x0000000000000001 (NEEDED) Shared library: [libc.so.6]
...
就像链接器一样,有directories where the loader will search for shared libraries by default .它将在这些默认目录之一中找到 libc.so.6
(GNU C 库)。但它不会找到我的 liba.so
或 libb.so
只是内置其中任何一个。
但是我可以告诉链接器为加载器提供缺少的信息,通过链接程序是这样的:
$ gcc -o prog main.o -L. -la -lb -Wl,-rpath=$PWD
使用 -Wl,-rpath=$PWD
,我告诉 gcc
传递(扩展的)选项 -rpath=$PWD
到链接器,如果我们这样做,我们将看到:
$ readelf -d prog
Dynamic section at offset 0xd98 contains 30 entries:
Tag Type Name/Value
0x0000000000000001 (NEEDED) Shared library: [liba.so]
0x0000000000000001 (NEEDED) Shared library: [libb.so]
0x0000000000000001 (NEEDED) Shared library: [libc.so.6]
0x000000000000001d (RUNPATH) Library runpath: [/home/imk/develop/so/scrap1]
...
现在当加载器加载 prog
时,它会看到 RUNPATH=/home/imk/develop/so/scrap1
是一个非默认目录,它还应该在其中搜索任何 NEEDED
共享库。 编程
然后可以成功加载其所有运行时依赖项:
$ ./prog
bb
关于build - 构建共享库时,是否只需要依赖库的头文件?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54779000/
我有一位客户希望她的网站适合 iPad/iPhone。该网站有一个用 Flash 制作的菜单标题。我使用 Google 的 Swiffy 将文件转换为 flash_1.html 文件。 当网站检测到
我注意到 boost 库使用了 (.hpp) 的头文件。 我很好奇,因为我看到的大多数源文件都使用普通的 .h 头文件。 是否有任何特殊情况需要使用 .hpp 而不是 .h ? 谢谢 最佳答案 只是约
C需要不同的头文件,如stdio.h、stdlib.h、fcntl.h等,对于不同的函数和结构定义,这是为什么?在幕后,一切都归结为 libc 那么为什么不创建一个包含所有定义和原型(prototyp
我是一名工科学生,不是一个非常强大的程序员。我的一项作业包括使用 openGL 创建 VR 程序。我得到了一个使用 gmtl 的模板(我真的不想重写)标题广泛。该作业需要实现一些 sixense模拟中
我正在尝试编译涉及C和C++文件的应用程序。使用一个特定的 header ,我遇到了问题。有问题的文件(C++头文件)如下所示: #ifndef TASK_H #define TASK_H #incl
我编写了我的项目,将main和c源代码保存在一个文件中,并将头文件保存在codeblocks的include目录中。当我从项目主函数中调用我的函数时,它编译得很好......但是当我将 header
我正在尝试将 .so 库导入到 python 代码中以使用 c 函数。我认为使用 from ctypes import * import ctypes lib = CDLL('./libcaenhvw
我有 2 个重叠的头文件,如下所示: header1.h ... __declspec(dllexport) void abc(); __declspec(dllexport) void xyz(
我有一个大型 Java 库,我想开发几个与该库接口(interface)的较小应用程序。该库将作为 JAR 出现在目标设备的类路径中,但我想尽可能避免在编译时出现整个库(JAR 或源代码)。 (如果重
我必须在我的项目中使用相机制造商提供的库。我正在使用 Visual Studio 2015。我在编写#include 后合并了VS 2015 建议的所有头文件。 我已经在VS 2015中指定了包含文件
我在编写我的项目时遇到了这个问题。我有 2 个 header ,每个 header 都有一个类,它们需要另一个,如下所示。 我以为这只是需要使用前向声明,但仍然不起作用。我没有想法。 寻求帮助:D H
我在一个项目中工作,我想在C++中做反射,所以经过研究我发现最好的方法是解析头文件以获得XML格式的抽象语法树并在反射中使用它。我尝试了很多工具,但没有一个与 visual c++ 2008 或 vi
我尝试从 BufferedImage 制作一个 BMP 文件。这是我尝试在 bmp 文件中写入标题和像素的函数。 我有一个错误,但我找不到那个。我需要你的帮助。 static void writeTo
这个问题在这里已经有了答案: 关闭 10 年前。 Possible Duplicate: undefined reference to `WinMain@16' 我一直在研究循环双链表。决定创建一个
处理这种情况的最佳做法是什么? class A { private: std::vector derp; public: struct B { ...
上下文:我正在为嵌入式板构建一些代码。它需要安装 Xilinx 工具、Linaro 工具链,然后调用开发板构建目录中的设置 bash 脚本(我们称之为 setup.sh)。 如果我不运行 setup.
我尝试使用头文件和源文件,但遇到了问题。因此,我对我正在尝试做的事情做了一个简化版本,我在 CodeBlocks 中遇到了同样的错误(undefined reference to add(double
当我包含用于将某些程序的整数类型转换为字符串类型的#include 头文件时,我的编译器(GCC for C++)抛出错误。谁能帮我解决这个问题? 这是一个C++的小代码,我是第一次尝试。 #incl
我的头文件中有一些错误,我不知道如何修复,因为我是 C++ 的新手。 这是头文件的代码: #pragma once typedef unsigned int uint; class DCEncrypt
按照目前的情况,这个问题不适合我们的问答形式。我们希望答案得到事实、引用或专业知识的支持,但这个问题可能会引发辩论、争论、投票或扩展讨论。如果您觉得这个问题可以改进并可能重新打开,visit the
我是一名优秀的程序员,十分优秀!