[Linux]从零开始的ARM Linux交叉编译与.so文件链接教程
一、前言
最近在项目需要将C++版本的opencv集成到原本的代码中从而进行一些简单的图像处理。但是在这其中遇到了一些问题,首先就是原本的opencv我们需要在x86的架构上进行编译然后将其集成到我们的项目中,这里我们到底应该将opencv编译为x86架构的还是编译成ARM架构,其次就是,编译完成以后,我们得到的都是.so的二进制文件,我们应该怎么将其链接到我们自己的项目中呢?带着这些问题,所以才有了这一篇教程。所以本次教程就是为了教大家如何将一个开源的项目交叉编译,并且将其链接到自己的项目中,如果你准备好了,就让我们开始吧!
二、谁适合本次教程
因为本次教程已经涉及到Linux C应用开发了所以并不适用小白,所以请具备一定的Linux基础以后,再阅读本次教程。在教程中很多关于Linux的基础操作我并不会细讲,甚至有的简单的步骤我会直接省略。让我们开始吧!
三、动态链接的概念
首先我们来讲讲什么是动态链接,以及什么是动态链接库。首先来讲讲什么是动态链接,动态链接是一种在程序运行时解析外部函数和变量引用的机制。简单来说,我们在编译程序时,并不需要将这部分程序编译到二进制文件中而是在程序运行时进行调用。然后就是动态连接库,动态链接库是包含可被多个程序共享的代码和数据的文件。动态连接库往往被我们编译成二进制文件(.dll或.so)然后在外部通过.h或者其它接口进行调用,这也是目前最主流的编程方式。总的来说动态连接库解决了代码重复依赖的问题,我们将一部分公共的代码编译到一个二进制文件中并且对外开放接口,外部程序通过预留接口访问动态连接库从而实现一套代码被多个程序使用。其次,如果我们的代码涉及机密,可以将其编译成动态链接库并且开放接口给用户使用,用户在可以使用完整函数功能的同时又不能获取源代码。了解了这些,下面就来带大家看看如何交叉编译动态链接库并且将其链接到自己的项目中!
四、动态链接库的编译与链接
这里大家需要注意,我们后面所说的编译都是指交叉编译,也就是说我们会在x86的设备上交叉编译动态链接库,并且将其链接到自己的代码中再将其放到开发板端运行。如果你只想在x86设备上完成此操作其实也可以直接看本次教程,因为不管需不需要交叉编译,道理都是一样的,唯一不一样的就是交叉编译后的程序需要放到开发板端运行。
这里我们首先需要安装交叉编译环境,交叉编译器的安装与环境变量的添加在之前交叉调试的文章中已经讲过了,大家可以直接参考:
vs code交叉调试教程:[Linux]从零开始的vs code交叉调试arm Linux程序教程-CSDN博客
这里我就默认大家已经安装好了Ubuntu并且已经安装好了交叉编译器,需要像这样能够输出版本号,如下图所示:

输入命令以后,能够有上面的输出就表示交叉编译环境没有问题,就可以进行下一步了。
这里我们同样使用图形化中的vs code进行操作,首先我们需要新建一个文件夹,直接使用下面的命令即可:
mkdir Project

然后我们再用vs code打开这个文件夹,这个文件夹后面也会作为我们的工程文件夹:

然后我们新建一个用于编译动态链接库的.c文件,这里我就直接叫“lib.c”了:

下面我们可以将下方的测试代码拷贝到这个.c的文件中:
#include "stdio.h"void Hello_World()
{printf("Hello World\n");
}void Hello_Gcc()
{printf("Hello Gcc\n");
}void Hello_Arm()
{printf("Hello Arm\n");
}
因为我们使用这个.c的文件编译动态链接库,所以并不需要写主函数。又因为是用于测试,所以写的函数非常简单,写好以后,如图所示:

大家将代码写入文件以后,记得保存。这里我们还需要一个.h文件来调用动态链接库中的函数,因为在动态链接库中只包含了函数的定义,没有包含声明,我们需要在.h文件中声明这些函数,这也是为了给外部一个接口供外部调用。这里我直接新建了了一个名为“lib.h”的文件:

我们在.h文件中写入下面的代码声明我们在.c文件中定义的函数:
#ifndef __LIB_H__
#define __LIB_H__void Hello_World();
void Hello_Gcc();
void Hello_Arm();#endif
写入以后,如图所示:

这里同样的,写入以后记得保存,然后我们在.c文件中引用这个.h文件:

这里我们修改完以后,我们将.c和.h文件都保存好,然后我们在项目目录下使用下面的命令将我们刚刚的代码编译成动态链接库:
aarch64-linux-gnu-gcc -fPIC -shared -o lib.so lib.c
这里还是来简单解释一下这段代码,首先是“aarch64-linux-gnu-gcc”,这就是我们编译时使用的交叉编译器,这里就不多说了,然后“-fPIC”是为了生成位置无关代码,这是动态库的要求。然后是“-shared”是为了告诉编译器要将这个文件编译成动态链接库,“-o lib.so”是为了指定生成的二进制文件的名字,这里生成的二进制名字就叫“lib.so”,最后“lib.c”就是我们输入的源文件了。
编译完成以后,就可以看到我们的项目目录下多了一个.so的文件,这个就是我们通过.c文件编译出的动态链接库:

这里我们可以使用“nm”工具来查看这个动态链接库中是否包含了我们写的函数,先使用下面的命令安装一个“nm”的工具库:
sudo apt install binutils
安装完成以后,我们使用下面的命令来检查我们的.so文件,这里lib.so就是我们编译出来的动态链接文件:
nm -D lib.so
输入命令以后,可以看到许多关于这个动态链接库的信息,看不懂没关系,我们只需要找输出的信息中有没有我们刚刚写的函数,这里可以看到,我们写的函数已经成功的编译到动态链接库中了:

这一步一般不会出错,就不多说了。
下面我们来使用一下这个被我们编译出来的动态链接库,这里我们直接在原本的项目文件夹中直接新建一个名为“main.c”的文件:

下面我们在main.c中直接输入下面的代码:
#include "lib.h"int main()
{Hello_World();Hello_Gcc();Hello_Arm();
}
这里我们只需要引用我们动态链接库的头文件即可。
写入完成以后,如图所示:

然后我们使用下面的命令来编译这个main.c文件:
aarch64-linux-gnu-gcc main.c -o main -L./ lib.so
这里还是来简单解释一下命令,首先就是“aarch64-linux-gnu-gcc”这就不多说了,然后是“main.c”这是我们编译时输入的源文件,同样不多说了,“-o main”表示我们要输出的二进制文件名为“main,”
这里的“-L”表示自己指定.so文件路径,这里我写的“./”表示在当前目录下搜索.so文件,最后“lib.so”表示要链接的库的名称。
比那一完成以后,就可以看到项目目录下多了一个名为“main”的可执行文件:

我们下面再使用“nm”工具来查看一下我们编译出来的可执行文件:

这里可以看到,我们的函数已经被编译到这个可执行文件中了,但是可以看到这些函数的前面都有一个U,这里的U表示“Undefined”,这也证实了这些函数未在我们的可执行文件中定义,需要从别的库链接。
因为这个可执行文件我们是使用交叉编译器编译的,所以肯定是不能在X86的主机上运行的:

下面我们就来测试一下这个可执行文件是否可以正常运行,这里我们需要将可执行文件和动态链接库文件发送到开发板端,这里我直接使用sftp发送,大家可以选择自己熟悉的方式去传输文件:

在开发板端,我们有一个可执行文件和一个动态链接库文件,如下:

这里我们需要指定一个环境变量LD_LIBRARY_PATH,这个环境变量会指定除了在标准路径以外的路径中寻找链接库文件,我们直接使用下面的命令将这个环境变量设置为当前目录,表示在当前目录寻找动态链接库:
export LD_LIBRARY_PATH=./
路径设置完成以后,我们直接运行可执行文件即可:

这里我们可以看到,我们的函数可以正常打印。
如果我们不指定LD_LIBRARY_PATH路径的话,运行可执行文件时就会提示库找不到:

至此,我们的动态链接库已经正常的编译并且正常的链接到了我们的可执行文件中。
五、编译与链接opencv
有了上面的经验以后,我们就可以来实战一下,这里就来教大家如何交叉编译opencv库并且链接到自己的项目中。
这里我们首先使用下面的命令来下载一下opencv的源码:
wget https://github.com/opencv/opencv/archive/refs/tags/4.5.5.tar.gz
如果这里下载卡住的话,就使用下面的命令配置一下代理,大家根据自己的情况自行配置即可:
export http_proxy=http://192.168.112.10:7890
export https_proxy=http://192.168.112.10:7890
拉取到opencv的源码压缩包以后,如图所示:
我们使用下面的命令解压opencv的源码压缩包:
tar -xvf 4.5.5.tar.gz
解压以后得到了下面的文件夹:

我们进入这个文件夹可以看到下面的文件夹:

下面我们准备编译,首先在opencv项目目录下新建一个名为“build”的目录,并且进入:
mkdir build

然后我们在build目录下新建一个名为build的构建文件:
touch build

下面我们在build构建文件中写入下面的构建脚本:
cmake-DCMAKE_SYSTEM_NAME=Linux \-DCMAKE_SYSTEM_PROCESSOR=aarch64 \-DCMAKE_C_COMPILER=aarch64-linux-gnu-gcc \-DCMAKE_CXX_COMPILER=aarch64-linux-gnu-g++ \-DCMAKE_INSTALL_PREFIX="/home/chulingxiao/Opencv/install" \-DBUILD_LIST=core,imgproc,highgui \-DBUILD_EXAMPLES=OFF \-DBUILD_TESTS=OFF \-DWITH_JPEG=ON \-DWITH_PNG=ON \..make -j4
make install
因为我们已经将交叉编译器的可执行文件的路径添加到环境变量了,所以这里直接写交叉编译器的名字即可。写入以后我们保存退出即可。
DCMAKE_INSTALL_PREFIX变量可以设置我们编译后安装的路径。这里大家自己写安装的路径即可。
我们再使用下面的命令给这个构建脚本可执行权限:
chmod +x build
然后我们使用下面的命令安装一下cmake:
sudo apt install cmake
然后我们直接执行这个可执行文件即可:
./build
随后就开始编译了:

我们等待makefile生成完成即可。生成完成makefile以后编译就开始了:

编译完成以后,可以看到我们的文件被安装到了如下目录:

我们打开安装的目录,可以看到以下文件夹:

这里的“include”文件夹里面放了所有的头文件:

在lib目录下放了所有的动态链接库文件:

下面来教大家如何将我们编译出来的内容链接到我们自己的项目中,这里我们首先回到项目文件夹中,然后将下面的内容写入main.c中用于测试我们opencv的功能,这是一个使用opencv将图片二值化的程序,可以将传入的图片二值化:
#include <opencv2/opencv.hpp>
#include <iostream>
int main()
{cv::Mat image = cv::imread("test.jpg", cv::IMREAD_GRAYSCALE);if (image.empty()) {std::cerr << "无法读取图片!" << std::endl;return -1;}// 二值化处理cv::Mat binary_image;double thresh_value = 128; // 设定阈值cv::threshold(image, binary_image, thresh_value, 255, cv::THRESH_BINARY);// 保存结果cv::imwrite("test_output.jpg", binary_image);std::cout << "二值化完成,结果保存在 test_output.jpg" << std::endl;
}
这里因为头文件以及库的引用比较复杂,我们写一个makefile来帮助我们编译文件,在项目目录下新建一个makefile文件,将下面的内容复制到文件中:
# 编译器设置
CXX = aarch64-linux-gnu-g++
CXXFLAGS = -std=c++11# OpenCV 路径(改成你的安装路径)
OPENCV_PATH = /home/chulingxiao/Opencv/install
OPENCV_INC = /home/chulingxiao/Opencv/install/include/opencv4/
OPENCV_LIB = /home/chulingxiao/Opencv/install/lib# 程序名称
TARGET = main
SRC = main.cOPENCV_LIBS = -lopencv_core -lopencv_imgcodecs -lopencv_imgproc $(TARGET): $(SRC)$(CXX) $(CXXFLAGS) -I$(OPENCV_INC) $< -o $@ -L$(OPENCV_LIB) $(OPENCV_LIBS)
复制以后如图所示:
这里大家只需要修改几个地方即可,首先就是OPENCV_PATH,这里大家将路径改为我们一开始构建opencv时DCMAKE_INSTALL_PREFIX变量的路径,也就是一开始设置的opencv的安装路径。
然后OPENCV_INC 路径大家写到opencv安装路径下的头文件路径,这里可以参考我写的。
最后OPENCV_LIB 路径大家写到opencv安装路径下的动态链接库路径,这里同样参考我写的。
修改完以上内容以后,就没有什么需要改了,我们直接在项目目录下输入“make”即可开始编译:

这里没有输出别的错误并且生成可执行文件就表示编译没有问题。
下面我们将这个可执行文件通过sftp传输到开发板端:

然后我们在开发板端运行这个可执行文件,发现缺少库:

我们在opencv安装目录中,将这个库拷贝到开发板:

然后再在开发板端指定一下寻找库的路径:
export LD_LIBRARY_PATH=./
然后再次执行可执行文件,发现还缺少了一个名为“libopencv_imgcodecs.so.405”的库:

我们再次使用sftp传输这个库到开发板:

我们再次运行可执行文件,发现还缺少了一个名为“libopencv_imgproc.so.405”的库:

我们再次使用sftp将这个库传输到开发板:

我们再次运行可执行文件,发现,可执行文件已经不提示找不到库了,提示的是找不到文件:

大家还记得我们的程序是做什么的吗?是的,这是一个将图像二值化的程序,要求我们传入一个图像,然而我们的目录下没有图像,这里我们传输一张图片到当前目录下并且将名字改为“test.jpg”这也和我们程序中的名称一样:

我们再次执行可执行文件,可以看到,图像已经正常被处理了,并且输出为了“test_output.jpg”:

我们将其传输到可视化界面中,可以看到图像被正常二值化:

这也证明了我们的opencv在正常工作,表示我们的交叉编译以及so文件的链接都是成功的。
六、结语
尽管我们在这个过程中遇到了很多问题,但我教给大家的是解决问题的方法,这些方法也包括了如果我们在运行可执行文件缺少库我们应该怎么办编译时怎样链接库不会出错。当然,做完上面的步骤,相信大家对嵌入式Linux开发多少有一定的了解了,但这也只是学习嵌入式Linux开发的一个开始。那么最后,感谢大家的观看!
相关文章:
[Linux]从零开始的ARM Linux交叉编译与.so文件链接教程
一、前言 最近在项目需要将C版本的opencv集成到原本的代码中从而进行一些简单的图像处理。但是在这其中遇到了一些问题,首先就是原本的opencv我们需要在x86的架构上进行编译然后将其集成到我们的项目中,这里我们到底应该将opencv编译为x86架构的还是编译…...
【模板】缩点
洛谷p3387 思路: 算法:tarjan算法 根据题意,我们只要找到一个路径,使得最终权重最大即可,首先,根据题目可知,如果一个点在一个环上,那么我们就将这整个环都选上,题目上允许我们能够重复走,因此,我们可以将环缩成点,将环所称点后,就可以转换成树,从没有父节点的结点开始,我们向…...
Rag实现流程
Rag实现流程 目录 Rag实现流程1. 加载问答链代码解释`chain_type="stuff"` 的含义其他 `chain_type` 参数选项及特点1. `map_reduce`2. `refine`3. `map_rerank`示例代码展示不同 `chain_type` 的使用其他参数类型2. 提出问题3. 检索相关文档代码解释其他参数类型4. …...
计算机网络- 传输层安全性
传输层安全性 7. 传输层安全性7.1 传输层安全基础7.1.1 安全需求机密性(Confidentiality)完整性(Integrity)真实性(Authenticity)不可否认性(Non-repudiation) 7.1.2 常见安全威胁窃…...
常青藤快速选择系统介绍
功能特点 支持多种属性和特性:可依据实体属性(如实体类型、图层、颜色、线宽等)以及实体特性(如直线长度、圆面积、文字内容等)进行筛选。多过滤条件与运算符号:支持多个过滤条件组合,基本涵盖实…...
【c语言】指针习题
练习一:使用指针打印数组内容 #include <stdio.h> void print(int* p, int sz) {int i 0;for (i 0; i < sz; i) {printf("%d ", *p);//printf("%d ", *(p i));} } int main() {int arr[] { 1,2,3,4,5,6,7,8,9,10 };int sz sizeof…...
KWDB创作者计划—KWDB认知引擎:数据流动架构与时空感知计算的范式突破
引言:数据智能的第三范式 在数字化转型进入深水区的2025年,企业数据系统正面临三重悖论:数据规模指数级增长与实时决策需求之间的矛盾、多模态数据孤岛与业务连续性要求之间的冲突、静态存储范式与动态场景适配之间的鸿沟。KWDB(K…...
Sqoop常用指令
Sqoop(SQL-to-Hadoop)是一个开源工具,旨在将关系型数据库中的数据导入到Hadoop的HDFS中,或者从HDFS导出到关系型数据库中。以下是一些常用的Sqoop命令: 导入数据到HDFS 1. 基本导入 sqoop import \ --connect jdbc:mys…...
银行业务知识序言
银行业务知识体系全景解析 第一章 金融创新浪潮下的银行业务知识革命 1.1 数字化转型驱动金融业态重构 在区块链、人工智能、物联网等技术的叠加作用下,全球银行业正经历着"服务无形化、流程智能化、风控穿透化"的深刻变革。根据麦肯锡《2023全球银行业…...
智慧水务项目(八)基于Django 5.1 版本PyScada详细安装实战
一、说明 PyScada,一个基于Python和Django框架的开源SCADA(数据采集与监视控制系统)系统,采用HTML5技术打造人机界面(HMI)。它兼容多种工业协议,如Modbus TCP/IP、RTU、ASCII等,并具…...
畅游Diffusion数字人(23):字节最新表情+动作模仿视频生成DreamActor-M1
畅游Diffusion数字人(0):专栏文章导航 前言:之前有很多动作模仿或者表情模仿的工作,但是如果要在实际使用中进行电影级的复刻工作,仅仅表情或动作模仿还不够,需要表情和动作一起模仿。最近字节跳动提出了一个表情+动作模仿视频生成DreamActor-M1。 目录 贡献概述 核心动…...
【Unity网络编程知识】C#的 Http相关类学习
1、搭建HTTP服务器 使用别人做好的HTTP服务器软件,一般作为资源服务器时使用该方式(学习阶段建议使用)自己编写HTTP服务器应用程序,一般作为Web服务器或者短连接游戏服务器时使用该方式(工作后由后端程序员来做&#…...
Python operator 模块介绍
operator 模块是 Python 标准库中的一个模块,它提供了一系列与 Python 内置运算符对应的函数。这些函数可以用于替代一些常见的运算符操作,在某些场景下能让代码更加简洁、高效,还能方便地用于函数式编程。以下是对 operator 模块的详细介绍: 1. 导入模块 使用 operator …...
SpringBoot企业级开发之【用户模块-更新用户头像】
功能如下所示: 我们先看一下接口文档: 为什么头像是一串字符串呢?因为我们是将头像图片放到第三方去存储,比如:阿里云等 开发思路: 实操: 1.controller 注意!这里使用【PatchMapping】注解…...
DAPP实战篇:使用ethersjs连接智能合约并输入地址查询该地址余额
本系列目录 专栏:区块链入门到放弃查看目录-CSDN博客文章浏览阅读400次。为了方便查看将本专栏的所有内容列出目录,按照顺序查看即可。后续也会在此规划一下后续内容,因此如果遇到不能点击的,代表还没有更新。声明:文中所出观点大多数源于笔者多年开发经验所总结,如果你…...
网络流量管理-流(Flow)
1. 传统网络的问题:快递员送信模式 想象你每天要寄100封信给同一个朋友,传统网络的处理方式就像一个固执的快递员: 每封信都单独处理:检查地址、规划路线、盖章、装车…即使所有信的目的地、收件人都相同,也要重复100…...
每日文献(十一)——Part two
今天从第四章:快速RCNN,方法细节开始介绍。 目录 四、快速RCNN:方法细节 4.1 快速R-CNN回顾 4.2 对抗网络设计 4.2.1 遮挡的对抗空间信息损失 4.2.2 对抗空间Transformer网络 4.2.3 对抗融合 五、实验 5.1 实验设置 5.2 PASCAL VOC…...
Laravel 实现 队列 发送邮件功能
一. 什么是队列 在构建 Web 应用程序时,你可能需要执行一些任务,例如解析文件,发送邮件,大量的数据计算等等,这些任务在典型的 Web 请求期间需要很长时间才能执行。 庆幸的是,Laravel 可以创建在后台运行…...
一、绪论(Introduction of Artificial Intelligence)
写在前面: 老师比较看重的点:对问题的概念本质的理解,不会考试一堆运算的东西,只需要将概念理解清楚就可以,最后一个题会出一个综合题,看潜力,前面的部分考的不是很深,不是很难&…...
Web攻防—SSRF服务端请求伪造Gopher伪协议无回显利用
前言 重学Top10的第二篇,希望各位大佬不要见笑。 SSRF原理 SSRF又叫服务端请求伪造,是一种由服务端发起的恶意请求,SSRF发生在应用程序允许攻击者诱使服务器向任意域或资源发送未经授权的请求时。服务器充当代理,执行攻击者构造…...
2025蓝桥杯python A组题解
真捐款去了,好长时间没练了,感觉脑子和手都不转悠了。 B F BF BF 赛时都写假了, G G G 也只写了爆搜。 题解其实队友都写好了,我就粘一下自己的代码,稍微提点个人的理解水一篇题解 队友题解 B 思路: 我…...
使用Python建模量子隧穿
引言 量子隧穿是量子力学中的一个非常有趣且令人神往的现象。在经典物理学中,我们通常认为粒子必须克服一个势垒才能通过它。但是,在量子力学中,粒子有时可以“穿越”一个势垒,即使它的能量不足以克服这个势垒。这种现象被称为“量子隧穿”。今天,我们将通过 Python 来建…...
微信小程序开发常用语法和api
vue写习惯了,小程序太久不做,一些语法和api都忘记。本文总结下小程序常用的语法和api 语法 绑定事件和传参 绑定事件还有很多,触摸反馈事件,表单事件,媒体事件后续更新细说。 <!-- 绑定事件 bindtap 事件传参 da…...
【时时三省】(C语言基础)选择结构程序综合举例
山不在高,有仙则名。水不在深,有龙则灵。 ----CSDN 时时三省 下面综合介绍几个包含选择结构的应用程序。 例题1: 写一程序,判断某一年是否为闰年。 程序1: 先画出判别闰年算法的流程图,见下图用变量le…...
Redis实现分布式定时任务
设计思路 任务表示:每个任务通过一个特定格式的键来表示。键名可以包含任务ID等信息,值可以是任务的具体内容或指向任务详情的引用。过期机制:利用Redis的EXPIRE命令为任务设置过期时间,当到达设定的时间点时,Redis会…...
File 类 (文件|文件夹操作)
一、File 类 1.1 前言 在 JDK 中 通过 java.io.File 类,可以实现操作系统重文件|文件夹的创建、删除、查看、重命名等操作。 1.2 File 类构造方法 File 一共提供了四个构造方法,都是有参构造。其中最常使用的是 File(String) 和 File(String, String)…...
Html页面Table表格导出导入Excel文件 xlsx.full
Html页面Table表格导出Excel文件 引用 xlsx.full.min.js 文件 导出 <!DOCTYPE html> <html> <head><meta charset"utf-8" /><title></title><script src"https://cdn.bootcdn.net/ajax/libs/jquery/3.6.3/jquery.min.j…...
【资料分享】瑞芯微RK3576,8核2.2GHz+6T算力NPU工业核心板说明书
核心板简介 创龙科技SOM-TL3576-S是一款基于瑞芯微RK3576J/RK3576高性能处理器设计的4核ARM Cor...
埃隆·马斯克如何通过开源创新塑造未来
李升伟 编译 埃隆马斯克的名字在多个行业回响——从电动汽车、太空探索到人工智能及更多领域。虽然许多人关注他革命性的公司(如特斯拉、SpaceX、Neuralink和The Boring Company),但较少有人意识到他在开源软件运动中悄然却深远的影响力。本…...
ALOPS智能化运维管理平台
AIOps(Artificial Intelligence for IT Operations)即智能运维,是将人工智能技术应用于 IT 运维管理领域,以实现自动化、智能化的运维决策和管理。以下是关于 AIOps 的详细介绍: 核心能力 数据收集与整合:…...
