QT学习笔记-Linux ARM环境下实现QT程序通过ODBC驱动访问SQLServer数据库

QT学习笔记-Linux ARM环境下实现QT程序通过ODBC驱动访问SQLServer数据库

  • 0、背景
  • 1、基本环境
  • 2、搭建交叉编译环境
  • 3、在交叉编译服务器上交叉编译安装unixODBC
    • 3.1 下载unixODBC
    • 3.2 交叉编译unixODBC
      • 3.2.1 基本编译说明
      • 3.2.2 交叉编译说明
      • 3.2.3 ./configure -build,-host,-target参数说明
      • 3.2.4 交叉编译unixODBC的具体过程
  • 4、交叉编译QT源码中的odbc.pro
  • 5、在交叉编译服务器上交叉编译安装freeTDS
    • 5.1 下载freeTDS
    • 5.2 交叉编译freeTDS
  • 6、把交叉编译生成的库部署到Linux Arm目标系统
    • 6.1 部署unixODBC
    • 6.2 部署freeTDS
    • 6.3 部署QT ODBC驱动
  • 7、编写ODBC配置
    • 7.1 配置freetds
    • 7.2 配置unixODBC
  • 8、基本工具测试
    • 8.1 使用freeTDS的tsql测试SQL Server的连通性
    • 8.2 使用unixODBC的isql测试数据源配置的正确性
  • 9、程序测试

0、背景

在嵌入式系统中使用QT开发上位机应用时不可避免的会涉及访问各种数据库的场景,而服务端数据库的种类则有多种可能(Oracle、Postgresql、MySql、SQLServer),本文就介绍一下如何实现在Linux Arm环境下实现QT程序通过ODBC驱动访问SQLServer数据库的。

1、基本环境

开发环境操作系统:windows10专业版
QT版本:5.15.2
QT安装目录:D:\Qt
交叉编译环境操作系统:Ubuntu18.4
QT版本:5.15.2
QT安装目录:/opt/Qt
目标环境操作系统:Linux4.19 Arm 64
QT版本:5.15.2

2、搭建交叉编译环境

由于ARM开发板的内存资源和存储资源以及嵌入式操作系统的限制,我们不大可能在ARM开发板的系统上安装QT开发环境以及源码,ARM开发板通常作为最终的运行环境。我们用的笔记本或个人PC往往是X86处理器,因此在笔记本系统或个人PC中编译的QT程序是无法直接运行在ARM开发板上的。从开发电脑到ARM开发板是有2个关键的变化,1是操作系统从Windows变为了Linux,2是从X86处理器变为了ARM处理器。这就需要搭建一台交叉编译服务器(可以是一台独立的PC或者是虚拟机),交叉编译服务器的操作系统要为Linux,示意如下:
在这里插入图片描述

交叉编译环境的搭建主要是获得对应的交叉编译工具链,就是在X86处理器上运行的Linux系统中使用交叉编译工具链对X86的Windows下开发的源码进行交叉编译生成基于Arm处理器的Linux系统能够运行的目标程序。

通常我们管X86的Windows开发环境叫开发环境,X86的Linux环境叫交叉编译环境,Arm的Linux环境叫目标环境或运行环境。最简单搭建交叉编译环境的方式就是直接使用目标开发版操作系统的编译服务器作为交叉编译服务器。不过这个环境一般比较大,因为操作系统的源码非常大。要想用一个轻量级的交叉编译环境,还是建议自己搭建。

本文后面介绍的是直接采用的操作系统编译服务器作为交叉编译环境的。

3、在交叉编译服务器上交叉编译安装unixODBC

3.1 下载unixODBC

1、unixODBC的官网地址为:https://www.unixodbc.org/
如下图:
在这里插入图片描述
2、如要下载最新版本,则点击左侧菜单中的Download,如下图:
在这里插入图片描述
3、如果要下载历史版本,可以通过ftp://ftp.unixodbc.org/pub/unixODBC网址下载需要的版本,如下图:
在这里插入图片描述
4、因为Linux Arm目标环境中安装的版本是2.3.4,为了一致,所以在交叉编译服务器上我也下载这个版本,如下:

cd /usr/local/src
wget ftp://ftp.unixodbc.org/pub/unixODBC/unixODBC-2.3.4.tar.gz

如下图:
在这里插入图片描述
5、然后就可以对其进行解压了,如下:

tar -xzvf unixODBC-2.3.4.tar.gz
ls

如下图:
在这里插入图片描述

3.2 交叉编译unixODBC

3.2.1 基本编译说明

1、在官网的下载页面已经介绍了基本的安装步骤,如下:
1.1 默认安装如下:

./configure
make
make install

1.2 指定目录安装如下:

./configure --prefix=/usr/local/unixODBC
make
make install

1.3 指定配置路径安装如下:

./configure --sysconfdir=/etc
make
make install

当然,1.2 1.3可以结合在一起。
在这里插入图片描述

3.2.2 交叉编译说明

1、要想对c/c++源码进行交叉编译,首先要在交叉编译服务器上安装配置交叉编译工具链。这里就不介绍安装过程了,为了能够保证与目标系统一致,我使用的就是构建目标操作系统时生成的交叉编译环境,目录为:LinuxSDK/buildroot/output/rockchip_rk3568/host/bin/。如下图:
在这里插入图片描述
上图中可以看出:

交叉编译工具链所在目录为:/home/zhenglb/Linux/wg_rk356x_linux_sdk/buildroot/output/rockchip_rk3568/host/bin
交叉编译工具链名称为:aarch64-buildroot-linux-gnu

2、需要把交叉编译器所在的路径添加到PATH环境变量中,方便系统能找到交叉编译器,如下:

echo $PATH
export PATH=$PATH:/home/zhenglb/Linux/wg_rk356x_linux_sdk/buildroot/output/rockchip_rk3568/host/bin
echo $PATH

如下图:
在这里插入图片描述

3.2.3 ./configure -build,-host,-target参数说明

在对c/c++源码进行交叉编译的时候,关键的一步是设置configure的参数,其中有三个参数-build、-host、-target的区别如下:

  • build:执行代码编译的主机,正常的话就是你的主机系统。这个参数一般由config.guess来猜就可以。当然自己指定也可以。
  • host:编译出来的二进制程序所执行的主机,因为绝大多数是本机编译,本机执行,所以这个值就等于build。只有交叉编译的时候(也就是本机编译,其他系统机器执行)才会build和host不同。用host指定运行主机。
  • target:这个选项只有在建立交叉编译环境的时候用到,正常编译和交叉编译都不会用到。他用build主机上的编译器,编译一个新的编译器(binutils, gcc,gdb等),这个新的编译器将来编译出来的其他程序将运行在target指定的系统上。

3.2.4 交叉编译unixODBC的具体过程

1、下面就看一下在交叉编译服务器上的交叉编译安装步骤,如下:

cd unixODBC-2.3.4
./configure --prefix=/usr/local/arm/unixODBC --host=aarch64-buildroot-linux-gnu
make
make install

日志太长,就不截图了。–prefix指定了安装位置,–host指定了交叉编译工具链

3、安装完毕后,就会在/usr/local/arm目录下看到unixODBC目录了,如下图:
在这里插入图片描述

  • 在/usr/local/arm/unixODBC/include目录下是我们交叉编译odbc.pro所依赖的unixODBC的头文件
  • 在/usr/local/arm/unixODBC/lib目录下是我们交叉编译odbc.pro所依赖的库

如下图:
在这里插入图片描述

4、交叉编译QT源码中的odbc.pro

1、首先在交叉编译服务器上进入odbc.pro源码目录,如下:

cd /opt/Qt/5.15.2/Src/qtbase/src/plugins/sqldrivers/odbc
ls

如下图:
在这里插入图片描述
2、修改odbc.pro文件添加指定unixODBC头文件及库文件的位置的相关代码,如下:

vim odbc.pro

odbc.pro修改的内容

#注释下行
#QMAKE_USE += odbc
#增加以下2行
INCLUDEPATH += /usr/local/arm/unixODBC/include
LIBS += -L/usr/local/arm/unixODBC/lib -lodbc

如下图:
在这里插入图片描述
3、按:wq,保存退出odbc.pro文件的编辑
4、使用交叉编译工具链下的qmake生成Makefile,如下:
这里我使用的是目标Linux操作系统构建时生成的交叉编译工具链,位置为:
LinuxSDK/buildroot/output/rockchip_rk3568/host/bin/qmake,如下图:
在这里插入图片描述
(如果在此终端下已配置了交叉编译工具链的PATH环境变量,可以直接qmake)具体使用如下:
在这里插入图片描述
5、然后输入make命令进行交叉编译,交叉编译完毕后会在原木odbc目录同级目录生成plugins/sqldrivers目录,交叉编译生成的库文件就这这个目录中,如下图:
在这里插入图片描述

5、在交叉编译服务器上交叉编译安装freeTDS

5.1 下载freeTDS

1、freeTDS的官网地址为:https://www.freetds.org/
如下图:
在这里插入图片描述
2、通过ftp://ftp.freetds.org/pub/freetds/可以查看历史版本,如下图:
在这里插入图片描述
3、在开发环境下可以直接点击右侧菜单中的Stable Release进行下载。或者通过右键查看属性获取到下载地址,如下图:
在这里插入图片描述
4、在交叉编译服务器上通过wget直接下载freetds到交叉编译服务器中,如下:

cd /usr/local/src
wget ftp://ftp.freetds.org/pub/freetds/stable/freetds-1.3.19.tar.gz

如下图:

在这里插入图片描述
5、然后就可以解压了,如下:

tar -xzvf freetds-1.3.19.tar.gz
ls

在这里插入图片描述

5.2 交叉编译freeTDS

1、首先,如果PATH环境变量中没有指定交叉编译器的路径,则需要把交叉编译器所在的路径添加到PATH环境变量中,方便系统能找到交叉编译器,如下:

echo $PATH
export PATH=$PATH:/home/zhenglb/Linux/wg_rk356x_linux_sdk/buildroot/output/rockchip_rk3568/host/bin
echo $PATH

1、下面就看一下在交叉编译服务器上的交叉编译安装步骤,如下:

cd freetds-1.3.19
./configure --prefix=/usr/local/arm/freetds --host=aarch64-buildroot-linux-gnu --with-tdsver=7.1 --disable-libiconv
make
make install

日志太长,就不截图了。–prefix指定了安装位置,–host指定了交叉编译工具链

3、安装完毕后,就会在/usr/local/arm目录下看到freetds目录了,如下图:
在这里插入图片描述

6、把交叉编译生成的库部署到Linux Arm目标系统

6.1 部署unixODBC

如果目标系统中没有unixODBC,则部署方式与6.2 部署freeTDS类似。因为我的目标系统已经有unixODBC了,并且安装时没有指定安装目录,是属于默认安装的。
因此配置文件路径为/etc/odbc.ini、/etc/odbcinst.ini。so库在/usr/lib目录下

6.2 部署freeTDS

1、首先把第5步在交叉编译服务器上交叉编译freeTDS后的安装目录/usr/local/arm/freetds压缩一下,如下:

cd /usr/local/aram
tar -cvf freetds.tar freetds

2、把压缩包freetds.tar从交叉编译服务器下载后,传到目标系统(Linux Arm)中。

注意:一定要相同目录

如下图:
在这里插入图片描述
3、然后在目标系统中对其进行解压。

6.3 部署QT ODBC驱动

把第4步在交叉编译服务器上交叉编译QT源码中的odbc.pro生成的libqsqlodbc.so复制到目标系统(Linux Arm)的qt的插件目录下,如下图:
复制前:
在这里插入图片描述
复制后:
在这里插入图片描述

7、编写ODBC配置

7.1 配置freetds

修改/usr/local/aram/freetds/etc/freetds.conf,添加如下内容:

[SQLSERVER]host = xxx.xxx.xxx.xxxport = 1433tds version = 7.0client charset = UTF-8

freetds.conf是用于管理SQL Server服务信息的
[SQLSERVER] #定义服务名称
host = xxx.xxx.xxx.xxx #指定SQL Server服务器的IP地址或者主机名称/域名
port = 1433 #指定SQL Server服务器对外服务的端口号
tds version = 7.0 #指定tds的版本
client charset = UTF-8 #指定客户端字符集

7.2 配置unixODBC

修改/etc/odbcinst.ini,添加如下内容:

[FreeTDS]
Description = FreeTDS unixODBC Driver
Driver = /usr/local/arm/freetds/lib/libtdsodbc.so
UsageCount = 1

odbcinst.ini是unixODBC管理数据访问驱动的配置文件,通过ODBC访问不同种类的数据库会用到不同的驱动库
[FreeTDS] #定义驱动名称
Description = FreeTDS unixODBC Driver #对驱动的描述,可以是任意值
Driver = /usr/local/arm/freetds/lib/libtdsodbc.so #指定驱动库文件的完整路径
UsageCount = 1

修改/etc/odbc.ini,添加如下内容:

[testdsn]
Driver = FreeTDS
Description = MS SQL Server
Servername = SQLSERVER
Database = test

odbc.ini是unixODBC管理数据源名称的配置文件,在程序中指定的数据源名称就是在这个文件中定义的
[testdsn] #是指数据源名称
Driver = FreeTDS #指定访问数据库所使用的驱动,驱动的值(FreeTDS)必须为odbcinst.ini中的配置的中括号中的值[FreeTDS]一致
Description = MS SQL Server #对当前数据源的描述,可以是任意值
Servername = SQLSERVER #指定服务名称,服务名称(SQLSERVER)必须跟freetds.conf中配置的中括号中的值[SQLSERVER]一致
Database = test #指定默认数据库,数据库的名称必须是服务器上有的数据库

8、基本工具测试

8.1 使用freeTDS的tsql测试SQL Server的连通性

1、在freetds的部署目录(/usr/local/arm/freetds)的bin目录下有一个工具tsql,如下:
在这里插入图片描述
2、查看FreeTDS状态
命令如下:

cd /usr/local/arm/freetds/bin
./tsql -C

执行结果如下:
在这里插入图片描述
3、传入IP地址,端口号、用户名、密码测试SQL Server的连通性

命令格式:tsql -Hip地址 -p端口号 -U 用户名 -P 密码

测试如下:
在这里插入图片描述
4、对freetds.conf中配置的服务名进行连通性测试。

命令格式:tsql -S 服务名 -U 用户名 -P 密码

在这里插入图片描述
通过服务名能连通,说明我们的freetds.conf配置是正确的。

8.2 使用unixODBC的isql测试数据源配置的正确性

1、我的目标系统是按默认配置安装的unixODBC,因此他的工具默认都是安装到了/usr/bin目录下。
2、可以通过odbcinst查看unixODBC的基本信息,如下:

cd /usr/bin
odbcinst -j

执行如下图:
在这里插入图片描述
3、通过isql工具对odbc.ini中配置的数据源是否正确进行验证,如下:

命令格式:isql 数据源名称 数据库登录名 数据库登录密码

测试如下图:
在这里插入图片描述
如果提示Connected!,并且能够正确执行SQL语句,说明odbc.ini中的数据源配置是正确的。

9、程序测试

1、QT关键的代码如下:

void Widget::on_btnDbTest_clicked()
{QStringList driverList = QSqlDatabase::drivers();qDebug() << driverList;//以下代码是测试通过ODBC访问MSSQLServer数据库QSqlDatabase db = QSqlDatabase::addDatabase("QODBC");db.setDatabaseName("testdsn");			//这里指定的是配置好的数据源名称db.setUserName("sa");db.setPassword("xxxxxx");				//密码根据实际进行修改if (!db.open()){qDebug() << "数据库连接失败!";QSqlError lastError = db.lastError();qCritical() << lastError.nativeErrorCode() << ", " << lastError.text();}else{qDebug() << "数据库连接成功!";QSqlQuery query(db);if (query.exec("select * from UserInfo")){qDebug() << "查询表格UserInfo成功!";while(query.next()){qDebug() << query.value(0).toInt() << ", " << query.value(1).toString() << ", " << query.value(2).toString() << "," << query.value(3).toString();}}}
}

2、对QT程序进行交叉编译后,部署到目标系统进行执行测试,运行效果如下:
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/45840.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

不知道开黑语音哪个软件好?

黑盒语音官方网站&#xff1a;https://chat.top 免费支持AI降噪免费支持高品质立体声免费支持码率128Kbps,192Kbps免费支持上传100M文件免费支持动态房间头像和横幅支持更多自定义动态表情即将支持更多免费功能

IP 地址监控工具

地址监控实用程序是一套 IP 工具&#xff0c;包括 IP 地址监控工具、流氓检测工具和 MAC 地址解析器&#xff0c;用于日常监控和管理 DNS 名称、IP和 MAC 地址。地址监控工具用于 IP监控&#xff0c;用于管理 DNS 名称、网络的 IP 和 MAC 地址&#xff0c;并跟踪 IP 地址。 IP…

JavaScript中的this关键字的作用,以及它如何确定其值

聚沙成塔每天进步一点点 ⭐ 专栏简介⭐ this关键字的作用⭐ this的值取决于执行上下文⭐ 示例⭐ 总结⭐ 写在最后 ⭐ 专栏简介 前端入门之旅&#xff1a;探索Web开发的奇妙世界 记得点击上方或者右侧链接订阅本专栏哦 几何带你启航前端之旅 欢迎来到前端入门之旅&#xff01;这…

MySQL数据库第十四课--------sql优化---------层层递进

作者前言 &#x1f382; ✨✨✨✨✨✨&#x1f367;&#x1f367;&#x1f367;&#x1f367;&#x1f367;&#x1f367;&#x1f367;&#x1f382; ​&#x1f382; 作者介绍&#xff1a; &#x1f382;&#x1f382; &#x1f382; &#x1f389;&#x1f389;&#x1f389…

极光笔记 | 如何为您的业务开发和训练一个AI-BOT

生成式AI&#xff08;Generative AI&#xff09;是当今科技领域的前沿技术之一。随着数据量的不断增加和计算能力的不断提升&#xff0c;AI技术在企业和个人生活中的应用越来越广泛。AI-BOT&#xff08;以下简称BOT&#xff09;是生成式AI技术的其中一种重要的应用形式&#xf…

最小二乘拟合圆柱

目录 一、算法原理二、代码实现 本文由CSDN点云侠原创&#xff0c;原文链接。如果你不是在点云侠的博客中看到该文章&#xff0c;那么此处便是不要脸的爬虫。 一、算法原理 由圆柱面的几何特性可得&#xff0c;圆柱面上的点到其轴线的距离恒等于半径 r 0 r_0 r0​&#xff0c;…

Java自学到什么程度就可以去找工作了?

引言 Java作为一门广泛应用于软件开发领域的编程语言&#xff0c;对于初学者来说&#xff0c;了解到什么程度才能开始寻找实习和入职机会是一个常见的问题。 本文将从实习和入职这两个方面&#xff0c;分点详细介绍Java学习到什么程度才能够开始进入职场。并在文章末尾给大家安…

基于springboot+vue的博物馆藏品平台(前后端分离)

博主主页&#xff1a;猫头鹰源码 博主简介&#xff1a;Java领域优质创作者、CSDN博客专家、公司架构师、全网粉丝5万、专注Java技术领域和毕业设计项目实战 主要内容&#xff1a;毕业设计(Javaweb项目|小程序等)、简历模板、学习资料、面试题库、技术咨询 文末联系获取 项目介绍…

剑指 Offer 48. 最长不含重复字符的子字符串(C++实现)

剑指 Offer 48. 最长不含重复字符的子字符串https://leetcode.cn/problems/zui-chang-bu-han-zhong-fu-zi-fu-de-zi-zi-fu-chuan-lcof/ dp 注意&#xff1a;缩小 不含重复字符子串 时的写法 dp_1 min(i - charToIndex[s[i]], dp_0 1); int lengthOfLongestSubstring(string s…

python中使用xml快速创建Caption和URL书签管理器应用程序

导语&#xff1a; 本文介绍如何使用wxPython库创建一个Caption和URL管理器应用程序。该应用程序具有图形用户界面&#xff0c;允许用户输入Caption和URL&#xff0c;并将其保存到XML文件中。此外&#xff0c;还提供了浏览文件夹并选择HTML文件的功能&#xff0c;并可以运行另一…

深入解析淘宝API,实现高效商务应用

淘宝API的基本调用 1. API文档与SDK 淘宝API官方提供了详细的API文档&#xff0c;包含了API的使用说明、参数列表、示例代码等内容。开发者可以通过文档了解每个API接口的具体功能和使用方法。此外&#xff0c;淘宝API还提供了多种编程语言的SDK&#xff0c;方便开发者进行快速…

jupyter notebook出现ERR_SSL_VERSION_OR_CIPHER_MISMATCH解决方案

大家好,我是爱编程的喵喵。双985硕士毕业,现担任全栈工程师一职,热衷于将数据思维应用到工作与生活中。从事机器学习以及相关的前后端开发工作。曾在阿里云、科大讯飞、CCF等比赛获得多次Top名次。现为CSDN博客专家、人工智能领域优质创作者。喜欢通过博客创作的方式对所学的…

c#扩展方法的使用

扩展方法可以向现有类型“添加”方法&#xff0c;无需创建新的派生类型、重新编译或以其他方式修改原始类型&#xff0c;用起来很方便&#xff0c;下面是我写的例子&#xff0c;为string这个常用的类型添加一个showmes方法&#xff0c;以下是扩展方法的代码&#xff1a; public…

python爬虫9:实战2

python爬虫9&#xff1a;实战2 前言 ​ python实现网络爬虫非常简单&#xff0c;只需要掌握一定的基础知识和一定的库使用技巧即可。本系列目标旨在梳理相关知识点&#xff0c;方便以后复习。 申明 ​ 本系列所涉及的代码仅用于个人研究与讨论&#xff0c;并不会对网站产生不好…

Linux中shell脚本——for、while循环及脚本练习

目录 一.for循环 1.1.基本格式 1.2.类C语言格式 二.while循环 2.1.基本格式 2.2.死循环语句 三.跳出循环 3.1.continue跳出循环 3.2.break跳出循环 四.常用循环 4.1.循环打印九九乘法表 4.2.循环ping测试某个网段网络连通性 4.3.while死循环实现猜数字游戏 4.4.数…

Linux的热拔插UDEV机制

文章目录 UDEV简介守护进程基本特点 守护进程和后台进程的区别开发守护进程结束 UDEV简介 udev是一个设备管理工具&#xff0c;udev以守护进程的形式运行&#xff0c;通过侦听内核发出来的uevent来管理/dev目录下的设备文件。 udev在用户空间运行&#xff0c;而不在内核空间 …

硬编码基础一(经典定长指令,寄存器相关)

硬编码基础一&#xff08;定长指令&#xff09; push/pop 通用寄存器 50~57是push8个32位通用寄存器 58~5f是pop8个32位通用寄存器 inc/dec 通用寄存器 40~47是inc8个32位通用寄存器 47~4f是dec8个32位通用寄存器 八位通用寄存器的立即数赋值 b0~b3 {立即数} 是低八位(…

【使用Zookeeper当作注册中心】自己定制负载均衡常见策略

自己定制负载均衡常见策略 一、前言随机&#xff08;Random&#xff09;策略的实现轮询&#xff08;Round Robin&#xff09;策略的实现哈希&#xff08;Hash&#xff09;策略 一、前言 大伙肯定知道&#xff0c;在分布式开发中&#xff0c;目前使用较多的注册中心有以下几个&…

Typescript基础知识(类型拓宽、类型缩小)

系列文章目录 引入一&#xff1a;Typescript基础引入&#xff08;基础类型、元组、枚举&#xff09; 引入二&#xff1a;Typescript面向对象引入&#xff08;接口、类、多态、重写、抽象类、访问修饰符&#xff09; 第一章&#xff1a;Typescript基础知识&#xff08;Typescri…

elementPlus——图标引入+批量注册全局组件——基础积累

因为我们要根据路由配置对应的图标&#xff0c;也要为了后续方便更改。因此我们将所有的图标注册为全局组件。&#xff08;使用之前将分页器以及矢量图注册全局组件的自定义插件&#xff09;&#xff08;所有图标全局注册的方法element-plus文档中已给出&#xff09; 全局注册…