Qt应用程序连接达梦数据库-飞腾PC麒麟V10

目录

      • 前言
      • 1 安装ODBC
        • 1.1 下载unixODBC源码
        • 1.2 编译安装
        • 1.4 测试
      • 2 编译QODBC
        • 2.1 修改 qsqldriverbase.pri 文件
        • 2.2 修改 odbc.pro 文件
        • 2.3 编译并安装QODBC
      • 3 Qt应用程序连接达梦数据库测试
      • 4 优化ODBC配置,方便程序部署
        • 4.1 修改pro文件,增加DESTDIR 变量配置
        • 4.2 新建lib目录,拷贝驱动到工程目录下
        • 4.3 增加config目录,存放ODBC配置文件
        • 4.3 设置LD_LIBRARY_PATH 环境变量
        • 4.4 打包部署
      • 总结

前言

本文主要记录了在飞腾架构麒麟V10系统下开发Qt应用程序时,如何通过ODBC连接达梦数据库,将ODBC的两个配置文件作为应用程序配置文件的一部分随程序打包发布,并在程序中动态设置ODBC需要的环境变量。

1 安装ODBC

1.1 下载unixODBC源码

下载地址 https://www.unixodbc.org/download.html
解压 gunzip unixODBC*.tar.gz 或 tar xvf unixODBC*.tar

1.2 编译安装

打开终端,cd到unixODBC源码所在目录下,执行编译安装命令:

./configure
make
make install

默认安装路径为/usr/local ,当然了也可以通过./configure --prefix=/usr/local/unixODBC 来指定安装目录。默认安装完成如下图所示。
在这里插入图片描述

1.4 测试

首先添加两个环境变量,用终端打开.profile 文件,在最后增加两行

export ODBCINI=/usr/local/etc/odbc.ini
export ODBCSYSINI=/usr/local/etc

在终端执行source .profile命令使环境变量生效,飞腾PC麒麟V10需要重启生效。

修改odbc.ini文件,配置达梦数据库信息

[MAIN]
Description=DM ODBC DSN
Driver=MAIN ODBC DRIVER
SERVER=192.168.1.2
UID=SYSDBA
PWD=123456789
TCP_PORT=5236

[MAIN] <MAIN对应达梦数据库的表空间名>
Driver=<驱动程序名称,与odbcinst.ini文件中[MAIN ODBC DRIVER]名称对应>
SERVER=<数据库服务器地址>
UID=<数据库用户名>
PWD=<数据库用户密码>
TCP_PORT=<数据库服务器端口>

修改odbcinst.ini,配置达梦驱动

[MAIN ODBC DRIVER]
Description=ODBC DRIVER FOR MAIN
Driver=/dm/dmdbms/drivers/odbc/libdodbc.so

然后打开终端,将达梦数据库的odbc驱动和odbc动态库所在路径添加到LD_LIBRARY_PATH环境变量中,如果没有安装达梦数据库驱动,可以参考这篇文章来安装 飞腾架构麒麟V10安装达梦数据库客户端。

export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:/dm/dmdbms/drivers/odbc:/usr/local/lib

设置好环境变量后,下面测试连接达梦数据库,cd到/usr/local/bin目录下,执行下面的命令,如果显示如下图所示的信息则表示连接成功。

isql -v MAIN 

在这里插入图片描述

2 编译QODBC

这里使用之前搭建麒麟开发环境时的Qt源码及编译好的qmake来编译QODBC,在麒麟V10环境下编译Qt,可以参考这篇文章,博客链接。qodbc在源码中的目录为 qt5.12.7/qtbase/src/plugins/sqldrivers/odbc

2.1 修改 qsqldriverbase.pri 文件

将原来的include 一行用#注释,改为下图红框中第二行所示的代码,因为qtsqldrivers-config.pri 文件中把odbc的编译设置为disable了
在这里插入图片描述

2.2 修改 odbc.pro 文件

把QMAKE_USE一行用#注释掉
在这里插入图片描述

2.3 编译并安装QODBC

可以使用QtCreator,也可以直接用命令编译,这里演示用命令编译QODBC。从 1.2 节知道unixODBC安装在/usr/local 目录下,编译qodbc需要用到unixODBC的头文件和动态库。在qodbc源码目录下打开终端,执行下面的命令后可生成libqsqlodbc.so库,并安装到qt安装目录的plugins/sqldrviers目录下。

../../../../bin/qmake "INCLUDEPATH+=/usr/local/include" "LIBS+=-L/usr/local/lib -lodbc" odbc.promake
make install

在这里插入图片描述

3 Qt应用程序连接达梦数据库测试

新建Qt工程,pro文件如下

QT       += core gui sql network
greaterThan(QT_MAJOR_VERSION, 4): QT += widgets
CONFIG += c++11
DEFINES += QT_DEPRECATED_WARNINGS
SOURCES += \main.cpp \mainwindow.cpp
HEADERS += \mainwindow.h
FORMS += \mainwindow.ui

在mainwindow.cpp中添加测试代码

#include "mainwindow.h"
#include "ui_mainwindow.h"#include <QSqlDatabase>
#include <QSqlError>
#include <QDebug>
#include <QSqlQuery>MainWindow::MainWindow(QWidget *parent): QMainWindow(parent), ui(new Ui::MainWindow)
{ui->setupUi(this);qDebug() << QSqlDatabase::drivers();QSqlDatabase db = QSqlDatabase::addDatabase("QODBC");db.setHostName("192.168.1.2");db.setPort(5236);QString dsn=QString::fromLocal8Bit("DRIVER={MAIN ODBC DRIVER};SERVER=192.168.1.2;DATABASE=MAIN");db.setDatabaseName(dsn);//db.setDatabaseName("MAIN");   //这里数据库名要设置成上面odbc.ini配置文件中的MAIN,即表空间名db.setUserName("SYSDBA");db.setPassword("123456789");if (db.open()){qDebug() << "connect ok!";}else{qDebug() << "connect fail! " << db.lastError().text();}QSqlQuery query;QString strsql = "select name from Category";if (query.exec(strsql)){qDebug() << "select ok!";}else{qDebug() << "select fail! " << query.lastError().text();}while (query.next()){qDebug() << query.value(0).toString();}query.clear();
}

运行程序,通过观察打印日志,可以看到已经成功连接到达梦数据库,并查询到了数据。

4 优化ODBC配置,方便程序部署

为了程序部署时不用在每台电脑安装ODBC环境,现对ODBC的配置进行优化。先删除之前添加到.profile文件中的环境变量ODBCINI、ODBCSYSINI,以免影响测试结果。

4.1 修改pro文件,增加DESTDIR 变量配置

DESTDIR = $$PWD/bin

4.2 新建lib目录,拷贝驱动到工程目录下

然后在bin目录下新建lib目录,将ODBC的动态库及达梦的odbc驱动都复制到lib目录下
在这里插入图片描述

4.3 增加config目录,存放ODBC配置文件

在bin目录下新建config目录,并将odbc.ini 和 odbcinst.ini复制到config目录下,因为odbcinst.ini文件中指定了ODBC驱动的绝对路径,因此要在程序中动态修改odbcinst.ini文件内容。下面的代码演示了在main.cpp中动态设置环境变量 ODBCINI、ODBCSYSINI。

#include "mainwindow.h"#include <QApplication>
#include <QProcessEnvironment>
#include <QFileInfo>
#include <QLibrary>
#include <QSettings>
#include <QFile>
#include <QtDebug>
#include <QByteArray>void envConfig(const QString &appPath)
{//设置ODBC环境变量
#ifdef linux//这里的设置都是打包后路径的设置QString odbcdriver = "libdodbc.so";QString odbcIni = appPath + "/config/odbc.ini";QString odbcsysini = appPath + "/config";QString librarypath = appPath + "/lib";//重新设置ODBC驱动路径QString odbcInstIni = appPath + "/config/odbcinst.ini";QFile odbcinstFile(odbcInstIni);if(odbcinstFile.open(QIODevice::ReadOnly)){QByteArray byteArr = odbcinstFile.readAll();int index = byteArr.indexOf("Driver") + sizeof ("Driver");int len = byteArr.size() - index;byteArr.replace(index, len, odbcdriver.prepend(librarypath + "/").toLocal8Bit().data());odbcinstFile.close();if(odbcinstFile.open(QIODevice::WriteOnly)){odbcinstFile.write(byteArr);odbcinstFile.close();}}//settings.setValue("ZODA ODBC DRIVER/Driver", odbcdriver.prepend(librarypath + "/"));//设置ODBC环境变量setenv("ODBCINI", odbcIni.toLocal8Bit().data(), 0);setenv("ODBCSYSINI", odbcsysini.toLocal8Bit().data(), 0);
#endif
}int main(int argc, char *argv[])
{QString appFilePath = QString(argv[0]);QFileInfo appFileInfo(appFilePath);qDebug() <<  appFileInfo.absolutePath();envConfig(appFileInfo.absolutePath());QApplication a(argc, argv);MainWindow w;w.show();return a.exec();
}

4.3 设置LD_LIBRARY_PATH 环境变量

经过上面的调整之后,要想成功连接数据库,需要将bin/lib目录添加到LD_LIBRARY_PATH 变量中,以便程序能正确找到驱动所在位置。设置好环境变量再次运行程序,就能正常连接达梦数据库了。
在这里插入图片描述

4.4 打包部署

这里使用linuxdeployqt进行打包,使用默认命令打包不会将/bin/lib目录下的odbc相关动态库拷贝到程序包的lib目录下的,这里需要手动拷贝bin/config 和 /bin/lib文件到程序包目录下。这时可以双击程序图标运行程序。如何在飞腾PC麒麟V10系统下安装使用linuxdeployqt打包工具可以参考这篇文章linuxdeployqt打包参考

总结

以上就是本文的所有内容了,对本文内容有疑问的朋友欢迎留言讨论!可在此处下载本文demo源码连接。

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

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

相关文章

高可用Kuberbetes部署Prometheus + Grafana

概述 阅读官方文档部署部署Prometheus Grafana GitHub - prometheus-operator/kube-prometheus at release-0.10 环境 步骤 下周官方github仓库 git clone https://github.com/prometheus-operator/kube-prometheus.git git checkout release-0.10 进入工作目录 cd kube…

聚观早报|华为Mate 60 Pro支持面容支付;特斯拉重回底特律车展

【聚观365】9月8日消息 华为Mate 60 Pro已支持面容支付 特斯拉将重回底特律车展 iPhone在美国有1.67亿用户 韩国半导体8月份出口85.6亿美元 比亚迪元PLUS冠军版将于9月15日上市 华为Mate 60 Pro已支持面容支付 毫无预热的华为Mate 60 Pro突然在华为商城首批开售&#xf…

老站长带你全面认识基站和天线

认识基站 作为数量最多的移动通信设备 基站几乎是随处可见 其实 基站也分为很多种 基站的天线&#xff0c;也分为很多种&#xff0c;真正都能区分清楚的人其实不多。 什么是基站 Base Station 一般特指“公用移动通信基站” 大家都知道&#xff0c;基站就是给手机提供信…

【Vue】vue2使用pdfjs预览pdf文件,在线预览方式一,pdfjs文件包打开新窗口预览pdf文件

系列文章目录 【Vue】vue2预览显示quill富文本内容&#xff0c;vue-quill-editor回显页面&#xff0c;v-html回显富文本内容 【Vue】vue2项目使用swiper轮播图2023年8月21日实战保姆级教程 【Vue】vue2使用pdfjs预览pdf文件&#xff0c;在线预览方式一&#xff0c;pdfjs文件包…

Convai:让虚拟游戏角色更智能的对话AI人工智能平台

【产品介绍】​ 名称 Convai​ 具体描述​ Convai是一款专为虚拟世界而设计的对话人工智能平台&#xff0c;它可以让你为你的游戏或应用中的角色 赋予人类般的对话能力。Convai利用了最先进的生成式对话人工智能技术&#xff0c;让你的角色可以…

数学建模__动态规划

动态规划就是&#xff0c;将任务每一步均记录下来&#xff0c;以便将来重复使用时能够直接调用 问题描述&#xff1a;给定n个物品&#xff0c;每个物品的重量是Wi,价值是Vi&#xff0c;但是背包最多能装下capacity重量的物品&#xff0c;问我们如何选择才能利益最大化。 这里涉…

R语言分析糖尿病数据:多元线性模型、MANOVA、决策树、典型判别分析、HE图、Box's M检验可视化...

全文链接&#xff1a;https://tecdat.cn/?p33609 Reaven和Miller&#xff08;1979&#xff09;研究了145名非肥胖成年人的葡萄糖耐量和胰岛素血液化学指标之间的关系。他们使用斯坦福线性加速器中心的PRIM9系统将数据可视化为3D&#xff0c;并发现了一个奇特的图案&#xff0c…

SQlite操作后如何正确退出

在 C 语言中&#xff0c;使用 SQLite 库进行数据库操作后&#xff0c;可以通过以下步骤来正常退出和关闭 SQLite 连接&#xff1a; 关闭数据库连接&#xff1a;在完成数据库操作后&#xff0c;使用 sqlite3_close() 函数来关闭 SQLite 连接。该函数接受一个指向 sqlite3 数据库…

一文详解TCP/IP协议栈的心跳、丢包重传、连接超时机制实例

1、问题概述 虽然软件底层模块在网络恢复后能自动重连上服务器&#xff0c;但会议因为网络问题已经退出&#xff0c;需要重新加入会议。因为客户特殊的网络运行环境&#xff0c;会频繁出现网络抖动不稳定的情况&#xff0c;客户要求必须要实现60秒内网络恢复后能依然保持在会议…

Unity中Shader的屏幕抓取 GrabPass

文章目录 前言一、抓取1、抓取指令2、在使用抓取的屏幕前&#xff0c;需要像使用属性一样定义一下,_GrabTexture这个名字是Unity定义好的 前言 Unity中Shader的屏幕抓取 GrabPass 一、抓取 1、抓取指令 屏幕的抓取需要使用一个Pass GrabPass{} GrabPass{“NAME”} 2、在使用…

752. 打开转盘锁

链接&#xff1a; 752. 打开转盘锁 题解&#xff1a; class Solution { public:int openLock(vector<string>& deadends, string target) {std::unordered_set<std::string> table(deadends.begin(), deadends.end());if (table.find("0000") ! t…

基础算法---区间合并

直接上题目,不废话! 题目 给定 n 个区间 [l,r]&#xff0c;要求合并所有有交集的区间。 注意如果在端点处相交&#xff0c;也算有交集。 输出合并完成后的区间个数。 例如&#xff1a;[1,3] 和 [2,6] 可以合并为一个区间 [1,6]。 输入格式 第一行包含整数 n。 接下来 n 行&am…

python-xpath语法-爬取彼岸图4k高清动漫壁纸

安装 pip install lxml导入 from lxml import etreexpath使用路径表达式提取html文档中的元素或元素集&#xff0c;然后元素通过沿路径path或步steps来选取数据 XPath常用语法格式 表达式描述div选取div元素的所有子元素/div选取根元素divul//li选取ul元素下的所有li子元素…

二蛋赠书二期:《Python机器学习项目实战》

文章目录 前言活动规则参与方式本期赠书《Python机器学习项目实战》作者介绍内容简介读者对象获奖名单 结语 前言 大家好&#xff01;我是二蛋&#xff0c;一个热爱技术、乐于分享的工程师。在过去的几年里&#xff0c;我一直通过各种渠道与大家分享技术知识和经验。我深知&am…

2327. 知道秘密的人数;1722. 执行交换操作后的最小汉明距离;2537. 统计好子数组的数目

2327. 知道秘密的人数 核心思想&#xff1a;动态规划&#xff0c;每天的人可以分为三种&#xff0c;可分享秘密的人&#xff0c;不可分享秘密的人&#xff0c;忘记秘密的人。定义f[i]为第i天可分享秘密的人&#xff0c;那么第(idelay ,iforget)天&#xff0c;会增加f[i]个可分…

C++算法进阶系列之倍增算法 ST 表

1. 引言 前文使用倍增算法实现了快速求幂的运算&#xff0c;本文继续讲解ST表&#xff0c;ST表即倍增表&#xff0c;本质就是动态规划表&#xff0c;记忆化了不同子问题域中的结果&#xff0c;用于实时查询。只是动态规划过程和传统的稍有点不一样&#xff0c;采用了倍增思想。…

h5开发网站-页面内容不够高时,如何定位footer始终位于页面的最底部

一、问题描述&#xff1a; 在使用h5开发页面时&#xff0c;会遇到这个情况&#xff1a;当整个页面高度不足以占满显示屏一屏&#xff0c;页脚不是在页面最底部&#xff0c;影响用户视觉。想让页脚始终在页面最底部&#xff0c;我们可能会想到用&#xff1a; 1.min-height来控…

初识Mybatis(二)动态SQL、缓存和逆向工程

动态SQL Mybatis框架的动态SQL技术是一种根据特定条件动态拼装SQL语句的功能&#xff0c;它存在的意义是为了解决拼接SQL语句字符串时的痛点问题。 &#xff08;比如多条件查询的设置&#xff0c;实现判空等&#xff09; 1、if 创建DynamicSQLMapper接口&#xff0c;DynamicSQL…

小程序中如何查看指定会员的付款记录

在小程序中&#xff0c;我们可以通过一些简单的步骤来查看指定会员的付款记录。下面是具体的操作流程&#xff1a; 1. 找到指定的会员卡。在管理员后台->会员管理处&#xff0c;找到需要查看付款记录的会员卡。也支持对会员卡按卡号、手机号和等级进行搜索。 2. 查看会员卡…

springboot3 + java虚拟线程初体验

java虚拟线程介绍 虚拟线程是 Java 19 的 预览特性&#xff0c;估计会在Java 21被纳入 JDK 的正式版本中&#xff0c;会在2023年9月发布&#xff0c;目前springboot 3 已经提供了对虚拟线程的支持。 虚拟线程和平台线程主要区别在于&#xff0c;虚拟线程在运行周期内不依赖操…