C++ 使用g+从cpp文件和静态库创建共享库+;
正如title所说,我想从三个cpp文件和一些静态库创建共享库 基本上我想这样做C++ 使用g+从cpp文件和静态库创建共享库+;,c++,linker,g++,shared-libraries,static-libraries,C++,Linker,G++,Shared Libraries,Static Libraries,正如title所说,我想从三个cpp文件和一些静态库创建共享库 基本上我想这样做 g++ libProject.so file1.cpp file2.cpp file3.cpp -I /usr/local/include -L /usr/local/lib -lAlgatorc 这是我的文件1.cpp: #include <iostream> #include <TestSetIterator.hpp> class SortingTestSetIterator :
g++ libProject.so file1.cpp file2.cpp file3.cpp -I /usr/local/include -L /usr/local/lib -lAlgatorc
这是我的文件1.cpp:
#include <iostream>
#include <TestSetIterator.hpp>
class SortingTestSetIterator : public TestSetIterator {
public:
TestCase *get_current(){
//TODO: implement method
return nullptr;
}
};
当我有file.o时,我想我需要说:
g++ -shared -o libProject.so file.o
但我不知道如何将.cpp文件编译成.o文件。
我知道我可以这样做
g++ -std=gnu++11 -o file1.o -fPIC -c file1.cpp -I /usr/local/include
但我还必须为这个命令提供静态库,因为file1.cpp(和其他文件)继承了在/usr/local/include/TestSetIterator.hpp中定义的类,但在/usr/local/lib/libAlgatorc.a中实现,因为-c我不能说是-L/usr/local/lib-lAlgatorc
最后,我想要这三个类的共享库,以便在main函数中,我可以将这个库加载到我的程序中,并且可以从这三个类中调用方法。因此,我希望共享库包含来自静态库(libAlgatorc.a)和所有三个cpp文件(file1.cpp、file2.cpp和file3.cpp)的所有符号
我使用的是Ubuntu 12.04 x64
> g++ --version
g++ (Ubuntu 4.8.1-2ubuntu1~12.04) 4.8.1
Copyright (C) 2013 Free Software Foundation, Inc.
This is free software; see the source for copying conditions. There is NO
warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
ld-r file1.o file2.o file3.o-o file.o
您可能应该使用g++
链接,而不是直接调用ld
。将.o
文件传递给GCC将导致它为您调用链接器,因此这是有效的:
g++ file1.o file2.o file3.o -o file.o
你说:
但我不知道如何将.cpp文件编译成.o文件
您只需使用-c
进行编译(如果您想将目标文件放入共享库中,那么也可以使用-fPIC
):
但我还必须为这个命令提供静态库
否,因为带有-c
的命令正在编译,而不是链接,所以您不能提供库,因为它们仅在链接时使用
因此,我希望共享库包含来自静态库(libAlgatorc.a)和所有三个cpp文件(file1.cpp、file2.cpp和file3.cpp)的所有符号
然后你需要阅读我已经给你指出的答案:
也许您应该阅读一篇关于在类unix环境中构建软件的教程,或者类似的教程,以便了解所需的单独步骤以及每个命令的作用
要编译每个文件,请执行以下操作:
g++ -c -fPIC file1.cpp
g++ -c -fPIC file2.cpp
g++ -c -fPIC file3.cpp
(此处不需要-o
选项,默认情况下,当您使用-c
选项时,GCC将file1.cpp
等文件编译为file1.o
。)
或者,您可以通过一个步骤完成:
g++ -c -fPIC file1.cpp file2.cpp file3.cpp
这里不能使用-o
选项,因为作为-c
步骤的输出,有三个不同的.o
文件,因此不能指定单个输出文件
要将它们全部链接到一个共享库中,该库还包括libAlgatorc.a
中的符号,请执行以下操作:
g++ file1.o file2.o file3.o -shared -o libProject.so -Wl,--whole-archive libAlgatorc.a -Wl,--no-whole-archive
或者在一个命令中编译所有三个文件,然后链接它们(注意这里没有-c
选项):
注意:说-I/usr/local/include
或-L/usr/local/include
是多余的,因为默认情况下总是搜索这些路径
更简单的方法是编写makefile:
libProjects.so: file1.o file2.o file3.o
$(CXX) -shared $^ -o $@ -Wl,--whole-archive -lAlgatorc -Wl,--no-whole-archive
file1.o file2.o file3.o : CXXFLAGS+=-fPIC
这就是makefile中所需的全部内容,因为Make已经知道如何从输入的
file1.cpp
创建file1.o
(并且为.o
目标设置cxflags
,可以确保编译这些文件时使用-fPIC
)。由于您对执行此操作的正确命令没有信心,我建议您使用Make来完成此操作。这不是我已经向您指出的问题的重复吗?否。没有从静态库和cpp文件生成共享库。只有从静态库生成共享库。我想把这两者结合起来,基本上还是一样的。只需将.o
文件添加到该答案中的命令中即可。关键是,另一个答案说明了如何确保静态库中的所有符号都在共享库中结束,这正是您的问题。事实上,这完全相同,如果您阅读问题,您将看到链接器命令包含以下文件:some.o other.o
当我来到这里时,我在寻找g++./libsomething.cpp-fPIC-shared-o./libsomething。所以如果我像你那样做,我会得到这个:/usr/bin/ld:/usr/local/lib/libAlgatorc.a(ParameterSet.o):在创建共享对象时不能使用针对“.rodata.str1”的R_X86_64_32重定位;使用-fPIC/usr/local/lib/libAlgator.a(ParameterSet.o)重新编译:无法读取符号:错误值集合2:错误:ld返回1个退出状态,这意味着libAlgator.ca
中的对象不可重定位,因此无法在共享库中使用。您可能需要获得一个新版本的libAlgatorc
,其中的内容是用-fPIC
构建的(可能需要一个.so
intead),或者您无法实现这一点,必须将主可执行文件链接到-lAlgatorc
,而不是将其包含在libProject中。因此更好!这个命令g++-fPIC file1.cpp file2.cpp file3.cpp-shared-o libProject.so-Wl,--whole archive-lAlgatorc-Wl,--whole archive
工作起来很有魅力!问题也出现在静态库中。静态库不是用-fPIC编译的,但它必须是,因为我们正在将这些符号添加到共享库中。完美答案真的!:)谢谢你的帮助!
g++ -c -fPIC file1.cpp
g++ -c -fPIC file2.cpp
g++ -c -fPIC file3.cpp
g++ -c -fPIC file1.cpp file2.cpp file3.cpp
g++ file1.o file2.o file3.o -shared -o libProject.so -Wl,--whole-archive libAlgatorc.a -Wl,--no-whole-archive
g++ -fPIC file1.cpp file2.cpp file3.cpp -shared -o libProject.so -Wl,--whole-archive -lAlgatorc -Wl,--no-whole-archive
libProjects.so: file1.o file2.o file3.o
$(CXX) -shared $^ -o $@ -Wl,--whole-archive -lAlgatorc -Wl,--no-whole-archive
file1.o file2.o file3.o : CXXFLAGS+=-fPIC