zoukankan      html  css  js  c++  java
  • QT开发——动态库/静态库的生成与调用(Qmake和Cmake方式)

    1.理解动态库与静态库区别

    链接:https://blog.csdn.net/wonengguwozai/article/details/93195827

    静态库和动态库最本质的区别就是:该库是否被编译进目标(程序)内部。

    1.1 静态(函数)库   

    一般扩展名为(.a或.lib),这类的函数库通常扩展名为libxxx.a或xxx.lib 。   

            这类库在编译的时候会直接整合到目标程序中,所以利用静态函数库编译成的文件会比较大,这类函数库最大的优点就是编译成功的可执行文件可以独立运行,而不再需要向外部要求读取函数库的内容;但是从升级难易度来看明显没有优势,如果函数库更新,需要重新编译。  

    1.2 动态函数库   

    动态函数库的扩展名一般为(.so或.dll),这类函数库通常名为libxxx.so或xxx.dll 。
           与静态函数库被整个捕捉到程序中不同,动态函数库在编译的时候,在程序里只有一个“指向”的位置而已,也就是说当可执行文件需要使用到函数库的机制时,程序才会去读取函数库来使用;也就是说可执行文件无法单独运行。这样从产品功能升级角度方便升级,只要替换对应动态库即可,不必重新编译整个可执行文件。

    1.3总结

        从产品化的角度,发布的算法库或功能库尽量使动态库,这样方便更新和升级,不必重新编译整个可执行文件,只需新版本动态库替换掉旧动态库即可。
        从函数库集成的角度,若要将发布的所有子库(不止一个)集成为一个动态库向外提供接口,那么就需要将所有子库编译为静态库,这样所有子库就可以全部编译进目标动态库中,由最终的一个集成库向外提供功能。

    2.qmake方式生成和调用动态/静态库

    链接:https://blog.csdn.net/lywzgzl/article/details/42805991

    2.1 生成库

    QT -= gui
    TARGET = laser_lib
    TEMPLATE = lib
    CONFIG   += staticlib  #加这句是生成静态库,不加则是动态库
    
    DEFINES += LASER_LIB_LIBRARY
    DEFINES += QT_DEPRECATED_WARNINGS
    LIBS += /usr/lib/x86_64-linux-gnu/libboost_thread.so
            /usr/lib/x86_64-linux-gnu/libboost_system.so
    
    SOURCES += 
            laser_lib.cpp 
    
    HEADERS += 
            laser_lib.h 
            laser_lib_global.h 
    
    include(LMS1xx/LMS1xx.pri)
    
    DESTDIR = $$PWD/../Lib
    
    unix {
        target.path = /usr/lib
        INSTALLS += target
    }

    2.2 调用库

    QT -= gui
    
    CONFIG += c++11 console
    CONFIG -= app_bundle
    DEFINES += QT_DEPRECATED_WARNINGS
    
    #增加系统库的依赖
    LIBS +=/usr/lib/x86_64-linux-gnu/libboost_thread.so
            /usr/lib/x86_64-linux-gnu/libboost_system.so
    #增加自定义库的依赖
    LIBS +=  -L$$PWD/../Lib -llaser_lib  #$$PWD获取当前pro文件的目录
    INCLUDEPATH += ../laser_lib
    
    SOURCES += main.cpp
    DESTDIR = $$PWD/../Lib

    3.cmake方式生成和调用动态库

    3.1创建共享库项目

    Cmake新建一个Qt Creator项目,在CMakeLists.txt中添加如下代码

    #-------------------------------------- 搜索软件包 --------------------------------------
    find_package(Qt5Widgets REQUIRED)
    find_package(Qt5Network REQUIRED)
    set(CMAKE_AUTOMOC ON)
    #-------------------------------------- 包含头文件 --------------------------------------
    include_directories(${Qt5Widgets_INCLUDE_DIRS})
    include_directories(${Qt5Network_INCLUDE_DIRS})
    include_directories(${CMAKE_CURRENT_SOURCE_DIR}/include)
    #-------------------------------------- -添加项目- --------------------------------------
    FILE(GLOB_RECURSE HEADER_FILES ${CMAKE_CURRENT_SOURCE_DIR}/*.h*)
    FILE(GLOB_RECURSE SOURCE_FILES ${CMAKE_CURRENT_SOURCE_DIR}/*.c*)
    
    add_library(${PROJECT_NAME} SHARED
    ${HEADER_FILES}
    ${SOURCE_FILES}
    ) #STATIC or SHARED 对应静态库 或者 动态库
    
    target_link_libraries(${PROJECT_NAME}
    ${Qt5Widgets_LIBRARIES}
    ${Qt5Network_LIBRARIES}
    )
    #-------------------------------------- -设置输出- --------------------------------------
    set(OUTPUT_LIB_DIR ${PROJECT_BINARY_DIR}/lib
        CACHE PATH "Output directory for libraries")
    file(MAKE_DIRECTORY ${OUTPUT_LIB_DIR})
    file(MAKE_DIRECTORY ${PROJECT_BINARY_DIR}/${PROJECT_NAME})
    set(LIBRARY_OUTPUT_PATH     ${OUTPUT_LIB_DIR})
    #-------------------------------------- -安装项目- --------------------------------------
    install(TARGETS ${PROJECT_NAME}
    EXPORT ${PROJECT_NAME}
    LIBRARY DESTINATION ${LIBRARY_OUTPUT_PATH}
    )

    按需编辑plugintest.h和plugintest.cpp,这个就是该共享库项目的plugintest类了,我加入了一个int sum(int input1, int input2);公共函数,内容为:

    int Plugintest::sum(int input1, int input2)
    {
        std::cout<<"Hello World!"<<std::endl;
        int sum = input1+input2;
        return sum;
    }

    保存并编译项目吧,没问题的话会在plugintest-build/lib目录里生成libplugintest.so。

    3.2调用共享库的项目

    3.2.1再次新建Cmake新建一个Qt Creator项目,用来调用刚刚生成的库

    在项目根目录下新建一个plugintest文件夹,把生成的库和头文件include文件夹全复制进去

    或者把库生成安装到计算机系统文件夹下,这样每次更新安装更加方便

    3.2.2在CMakeLists.txt中添加如下代码

    #-------------------------------------- 包含头文件 --------------------------------------
    include_directories(${CMAKE_CURRENT_SOURCE_DIR}/pludintest/include)
    
    #-------------------------------------- 引入库文件 --------------------------------------
    link_directories(${CMAKE_CURRENT_SOURCE_DIR}/pludintest)
    
    #-------------------------------------- -添加项目- --------------------------------------
    target_link_libraries(
    ${PROJECT_NAME}
    ${Qt5Widgets_LIBRARIES}
    ${Qt5Network_LIBRARIES}
    libplugintest.so
    )

     3.2.3修改一下main.cpp调用pluginTest的sum()试试看吧

    我的main.cpp内容如下:

    #include <QApplication>
    #include "plugintest.h"
    int main(int argc, char* argv[])
    {
      QApplication a(argc, argv);
      Plugintest demo;
      int sum = demo.sum(3,5);
      std::cout<<sum<<std::endl;
      return a.exec();
    }

     3.2.4保存编译项目然后运行。

    不出意外就会得到Hello World!和求和8的输出了,修改pluginTest项目的sum()函数重新编译该共享库,运行主程序就会得到不同的输出结果(不用重新编译主程序),这就是共享库(动态链接库)的好处所在。

    4.常见问题解决

    4.1 Qt项目在Qt Creator下编译运行正常,但是直接执行应用程序,内建的动态库找不到...

    Qt默认检索的动态库路径是/usr/lib和/lib,

    保证所有的动态库生成后都在同一个目录下即可,手动拖也行。

    然后把整个bin路径里所有动态库,拷贝到/usr/lib目录下即可





  • 相关阅读:
    24节气冬至
    最佳人体舒适温度是多少?
    常用正则
    点击按钮后的Loading处理
    支付时过渡动画
    npm ci 和 npm install
    Vue批量上传文件及实时进度
    HTML DOM classList 属性的使用
    Electron Uncaught ReferenceError: require is not defined
    javascript 深拷贝的问题
  • 原文地址:https://www.cnblogs.com/zx-hit/p/11977720.html
Copyright © 2011-2022 走看看