共享库和静态库的使用区别

Posted

技术标签:

【中文标题】共享库和静态库的使用区别【英文标题】:Difference of usage between shared and static library 【发布时间】:2016-12-23 16:20:31 【问题描述】:

我有一个非常简单的代码示例:

#include<iostream>
#include "libMPSSE_spi.h"

int main() 
   uint32 channels = 0;
   std::cout << "erreur = " <<  SPI_GetNumChannels(&channels) << std::endl;
   std::cout << "Number of available SPI channels = " << (int)channels << std::endl;
   return 0;

当我链接静态 libMPSSE 时效果很好,但共享库有问题。

我构建文件的两个命令是:

g++ test.cpp -o test.o -I../../libs/MPSSE -I../../libs/FTDI/linux -L../../libs/MPSSE/linux/x64 -lMPSSE

g++ test.cpp -o test.o -I../../libs/MPSSE -I../../libs/FTDI/linux ../../libs/MPSSE/linux/x64/libMPSSE.a -ldl

两种编译都可以,但是执行输出不一样。静态链接库工作正常,动态链接返回“其他错误”。

注意:我自己使用提供的 makefile 构建了这两个库(对象是在两个库之前以相同的方式构建的):

libMPSSE:   $(OBJECTS)
    $(CC)  -o libMPSSE.so -shared $(OBJECTS) -L /MinGW/lib -ldl
    $(AR) rcs libMPSSE.a $(OBJECTS)

什么可以解释这两个库的不同行为?

注意:在他们提供的库示例中,他们使用dlopen 的共享库,但如果我在链接时有库和标题,我不应该这样做吗?

注意2:他们都使用libftd2xx.so,我使用这个命令来运行可执行文件(动态和静态)

LD_LIBRARY_PATH=~/WS/Qt/main/LC4/main/libs/FTDI/linux/x64:~/WS/Qt/main/LC4/main/libs/MPSSE/linux/x64 ./test.o
LD_LIBRARY_PATH=~/WS/Qt/main/LC4/main/libs/FTDI/linux/x64 ./test.o

编辑:使用共享库的代码(来自他们的示例)

#include <iostream>
#include "libMPSSE_spi.h"
#include <dlfcn.h>


int main() 

    typedef FT_STATUS (*pfunc_SPI_GetNumChannels)(uint32 *numChannels);
    pfunc_SPI_GetNumChannels p_SPI_GetNumChannels;
    void *h_libMPSSE;

    h_libMPSSE = dlopen("../../libs/MPSSE/linux/x64/libMPSSE.so",RTLD_LAZY);
    if(!h_libMPSSE)
    
        std::cout <<"Failed loading libMPSSE.so. Please check if the file exists in ";
        std::cout << "the shared library folder(/usr/lib or /usr/lib64)" << std::endl;
        return 1;
    
    p_SPI_GetNumChannels = (pfunc_SPI_GetNumChannels)dlsym(h_libMPSSE, "SPI_GetNumChannels");

   uint32 channels = 0;
   std::cout << "erreur = " <<  p_SPI_GetNumChannels(&channels) << std::endl;
   std::cout << "Number of available SPI channels = " << (int)channels << std::endl;
   return 0;

【问题讨论】:

【参考方案1】:

当您的程序被加载和初始化时,操作系统会在适当的目录(例如 /usr/lib 或 %WINDIR%\system32)* 中搜索所需的共享库。

如果这些目录中不存在该库,则应用程序将不会运行,因为不满足依赖关系。您的示例可能建议您使用 dlopen 从应用程序所在的同一目录加载共享库 - 如果没有管理员权限,您可能无法将共享库复制到默认路径中。

* 通常您可以添加更多要搜索的路径,但这些是默认路径

【讨论】:

我最后在我的问题中添加了一条注释,可执行文件找到了库。如果它没有找到它,我会有一个 shell 错误告诉我它没有找到它

以上是关于共享库和静态库的使用区别的主要内容,如果未能解决你的问题,请参考以下文章

动态链接库和静态链接库的区别简述

静态库和共享库

C语言中静态库和动态库的区别,如何使用它们

详解Linux下静态库/动态库的生成和使用(含代码示例和操作流程)&&动态库和静态库的区别

单例模式 静态库和动态库的区别

Linux 静态库和共享(动态)库的创建与使用详解