通过cmakelist生成与调用C++动态链接库

2024-05-14 21:28

本文主要是介绍通过cmakelist生成与调用C++动态链接库,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • 前言
  • 生成动态链接库
    • 样例项目说明
    • 修改cmakelist
  • 调用动态链接库
    • 修改配置文件
    • 修改原来外层的CMakeLists.txt
  • 总结


前言

此前有写过用编译工具链直接通过命令行的方式生成与调用C++动态链接库的方法,本文记录下通过cmake来实现so的生成。


生成动态链接库

样例项目说明

以下笔者通过具体的开源项目代码进行演示。
在这里插入图片描述

通过以上项目的层级结构,我们可以看到有两个CMakeLists.txt。
其中外层CMakeLists.txt内容为:

# Copyright (c) Huawei Technologies Co., Ltd. 2021. All rights reserved.# 最低CMake版本
cmake_minimum_required(VERSION 3.5.1)# 项目名
project(HelmetIdentification)# 配置环境变量MX_SDK_HOME,如:/home/xxxxxxx/MindX_SDK/mxVision,可在远程环境中用指令env查看
set(MX_SDK_HOME $ENV{MX_SDK_HOME})if (NOT DEFINED ENV{MX_SDK_HOME})set(MX_SDK_HOME "/usr/local/Ascend/mindx_sdk")message(STATUS "set default MX_SDK_HOME: ${MX_SDK_HOME}")
else ()message(STATUS "env MX_SDK_HOME: ${MX_SDK_HOME}")
endif()add_subdirectory("./src")

src中的CMakeLists.txt内容为:

# CMake lowest version requirement
cmake_minimum_required(VERSION 3.5.1)
# project information
project(Individual)# Compile options
add_definitions(-D_GLIBCXX_USE_CXX11_ABI=0 -Dgoogle=mindxsdk_private)
add_compile_options(-std=c++11 -fPIC -fstack-protector-all -Wall -D_FORTIFY_SOURCE=2 -O2)set(CMAKE_RUNTIME_OUTPUT_DIRECTORY  "../../")
set(CMAKE_CXX_FLAGS_DEBUG "-g")
set(CMAKE_EXE_LINKER_FLAGS "${CMAKE_EXE_LINKER_FLAGS} -Wl,-z,relro,-z,now,-z,noexecstack -s -pie -pthread")
set(CMAKE_SKIP_RPATH TRUE)SET(CMAKE_BUILD_TYPE "Debug")
SET(CMAKE_CXX_FLAGS_DEBUG "$ENV{CXXFLAGS} -O0 -Wall -g2 -ggdb")
SET(CMAKE_CXX_FLAGS_RELEASE "$ENV{CXXFLAGS} -O3 -Wall")# Header path
include_directories(${MX_SDK_HOME}/include/${MX_SDK_HOME}/opensource/include/${MX_SDK_HOME}/opensource/include/opencv4//home/HwHiAiUser/Ascend/ascend-toolkit/latest/include/./
)# add host lib path
link_directories(${MX_SDK_HOME}/lib/${MX_SDK_HOME}/lib/modelpostprocessors${MX_SDK_HOME}/opensource/lib/${MX_SDK_HOME}/opensource/lib64//usr/lib/aarch64-linux-gnu//home/HwHiAiUser/Ascend/ascend-toolkit/latest/lib64//usr/local/Ascend/driver/lib64/./
)aux_source_directory(. sourceList)add_executable(main ${sourceList})target_link_libraries(main mxbase opencv_world boost_filesystem glog avformat avcodec avutil cpprest yolov3postprocess ascendcl acl_dvpp_mpi)install(TARGETS main DESTINATION ${CMAKE_RUNTIME_OUTPUT_DIRECTORY})

项目的主要目的是把src中的main.cpp文件编译成可执行文件输出到指定的地址。我们需要做的是把除main.cpp以外的文件编译成一个so,通过so来编译main.cpp,而不是通过源码的方式。(为了方便,我下面实现的时候把main.cpp也打包进了so)

修改cmakelist

src中的cmakelist,将原来:

aux_source_directory(. sourceList)add_executable(main ${sourceList})target_link_libraries(main mxbase opencv_world boost_filesystem glog avformat avcodec avutil cpprest yolov3postprocess ascendcl acl_dvpp_mpi)install(TARGETS main DESTINATION ${CMAKE_RUNTIME_OUTPUT_DIRECTORY})

修改为:

aux_source_directory(. sourceList)# add_executable(main ${sourceList})add_library(ai_detection310 SHARED ${sourceList})target_link_libraries(ai_detection310 mxbase opencv_world boost_filesystem glog avformat avcodec avutil cpprest yolov3postprocess ascendcl acl_dvpp_mpi)install(TARGETS ai_detection310 DESTINATION ${CMAKE_RUNTIME_OUTPUT_DIRECTORY})

在这里插入图片描述

编译运行后将会得到ai_detection310.so

调用动态链接库

修改配置文件

将生成的ai_detection310.so移动台指定的位置,为了方便我在原来的同级目录下,新建了test_so的文件夹,把需要用的文件都复制了过来,结构如下:
在这里插入图片描述
其中test_so中的CMakeLists.txt为src中的文件的拷贝,然后再对test_so中的CMakeLists.txt进行修改:
将CMakeLists.txt中:

aux_source_directory(. sourceList)# add_executable(main ${sourceList})add_library(ai_detection310 SHARED ${sourceList})target_link_libraries(ai_detection310 mxbase opencv_world boost_filesystem glog avformat avcodec avutil cpprest yolov3postprocess ascendcl acl_dvpp_mpi)install(TARGETS ai_detection310 DESTINATION ${CMAKE_RUNTIME_OUTPUT_DIRECTORY})

修改为:

include_directories(${PROJECT_SOURCE_DIR}/include/)message(STATUS "Include directories: ${PROJECT_SOURCE_DIR}")add_executable(main ${PROJECT_SOURCE_DIR}/main.cpp)target_link_libraries(main mxbase opencv_world boost_filesystem glog avformat avcodec avutil cpprest yolov3postprocess ascendcl acl_dvpp_mpi)target_link_libraries(main ${PROJECT_SOURCE_DIR}/lib/libai_detection310.so)install(TARGETS main DESTINATION ${CMAKE_RUNTIME_OUTPUT_DIRECTORY})

修改原来外层的CMakeLists.txt

将原来外层CMakeLists.txt中:

add_subdirectory("./src")

修改为:

add_subdirectory("./test_so")

编译运行后将会得到main的可执行文件。


总结

通过cmake进行so编译还是比较简单的,主要是用add_library(ai_detection310 SHARED ${sourceList})生成so替换原来add_executable(main ${sourceList})生成的可执行文件,然后对main.cpp进行配置,能够正确调用生成的so。这样我们就实现了通过cmakelist生成与调用C++动态链接库。
如果阅读本文对你有用,欢迎一键三连呀!!!
2024年5月14日16:25:40在这里插入图片描述

这篇关于通过cmakelist生成与调用C++动态链接库的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/989899

相关文章

深入理解C++ 空类大小

《深入理解C++空类大小》本文主要介绍了C++空类大小,规定空类大小为1字节,主要是为了保证对象的唯一性和可区分性,满足数组元素地址连续的要求,下面就来了解一下... 目录1. 保证对象的唯一性和可区分性2. 满足数组元素地址连续的要求3. 与C++的对象模型和内存管理机制相适配查看类对象内存在C++中,规

Java调用Python代码的几种方法小结

《Java调用Python代码的几种方法小结》Python语言有丰富的系统管理、数据处理、统计类软件包,因此从java应用中调用Python代码的需求很常见、实用,本文介绍几种方法从java调用Pyt... 目录引言Java core使用ProcessBuilder使用Java脚本引擎总结引言python

在 VSCode 中配置 C++ 开发环境的详细教程

《在VSCode中配置C++开发环境的详细教程》本文详细介绍了如何在VisualStudioCode(VSCode)中配置C++开发环境,包括安装必要的工具、配置编译器、设置调试环境等步骤,通... 目录如何在 VSCode 中配置 C++ 开发环境:详细教程1. 什么是 VSCode?2. 安装 VSCo

详解Java中如何使用JFreeChart生成甘特图

《详解Java中如何使用JFreeChart生成甘特图》甘特图是一种流行的项目管理工具,用于显示项目的进度和任务分配,在Java开发中,JFreeChart是一个强大的开源图表库,能够生成各种类型的图... 目录引言一、JFreeChart简介二、准备工作三、创建甘特图1. 定义数据集2. 创建甘特图3.

java如何调用kettle设置变量和参数

《java如何调用kettle设置变量和参数》文章简要介绍了如何在Java中调用Kettle,并重点讨论了变量和参数的区别,以及在Java代码中如何正确设置和使用这些变量,避免覆盖Kettle中已设置... 目录Java调用kettle设置变量和参数java代码中变量会覆盖kettle里面设置的变量总结ja

C++11的函数包装器std::function使用示例

《C++11的函数包装器std::function使用示例》C++11引入的std::function是最常用的函数包装器,它可以存储任何可调用对象并提供统一的调用接口,以下是关于函数包装器的详细讲解... 目录一、std::function 的基本用法1. 基本语法二、如何使用 std::function

【C++ Primer Plus习题】13.4

大家好,这里是国中之林! ❥前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站。有兴趣的可以点点进去看看← 问题: 解答: main.cpp #include <iostream>#include "port.h"int main() {Port p1;Port p2("Abc", "Bcc", 30);std::cout <<

AI一键生成 PPT

AI一键生成 PPT 操作步骤 作为一名打工人,是不是经常需要制作各种PPT来分享我的生活和想法。但是,你们知道,有时候灵感来了,时间却不够用了!😩直到我发现了Kimi AI——一个能够自动生成PPT的神奇助手!🌟 什么是Kimi? 一款月之暗面科技有限公司开发的AI办公工具,帮助用户快速生成高质量的演示文稿。 无论你是职场人士、学生还是教师,Kimi都能够为你的办公文

C++包装器

包装器 在 C++ 中,“包装器”通常指的是一种设计模式或编程技巧,用于封装其他代码或对象,使其更易于使用、管理或扩展。包装器的概念在编程中非常普遍,可以用于函数、类、库等多个方面。下面是几个常见的 “包装器” 类型: 1. 函数包装器 函数包装器用于封装一个或多个函数,使其接口更统一或更便于调用。例如,std::function 是一个通用的函数包装器,它可以存储任意可调用对象(函数、函数

C++11第三弹:lambda表达式 | 新的类功能 | 模板的可变参数

🌈个人主页: 南桥几晴秋 🌈C++专栏: 南桥谈C++ 🌈C语言专栏: C语言学习系列 🌈Linux学习专栏: 南桥谈Linux 🌈数据结构学习专栏: 数据结构杂谈 🌈数据库学习专栏: 南桥谈MySQL 🌈Qt学习专栏: 南桥谈Qt 🌈菜鸡代码练习: 练习随想记录 🌈git学习: 南桥谈Git 🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈🌈�