yafeilinux 发表于 2017-8-7 23:57:24

第26篇 Qt5之数据库(六)SQL关系表格模型QSqlRelationalTableModel

版权声明该文章原创于Qt开源社区(www.qter.org),作者yafeilinux,转载请注明出处!

导语       QSqlRelationalTableModel继承自QSqlTableModel,并且对其进行了扩展,提供了对外键的支持。一个外键就是一个表中的一个字段和其他表中的主键字段之间的一对一的映射。例如,student表中的course字段对应的是course表中的id字段,那么就称字段course是一个外键。因为这里的course字段的值是一些数字,这样的显示很不友好,使用关系表格模型,就可以将它显示为course表中的name字段的值。


环境:Windows 7 + Qt 5.8.0(包含QtCreator 4.2.1)

目录一、使用外键二、使用委托


正文
一、使用外键

1.新建Qt Widgets应用,名称为relationaltablemodel,基类为QMainWindow类名为MainWindow保持不变。完成后打开relationaltablemodel.pro项目文件,将第一行改为:
QT       += core gui sql
       然后保存该文件。
2.下面向项目中添加新的C++头文件connection.h,并更改其内容如下:
#ifndef CONNECTION_H
#define CONNECTION_H
#include <QSqlDatabase>
#include <QSqlQuery>
static bool createConnection()
{
   QSqlDatabase db = QSqlDatabase::addDatabase("QSQLITE");
   db.setDatabaseName("database.db");
   if(!db.open()) return false;
   QSqlQuery query;
   query.exec("create table student (id int primarykey, name vchar,course int)");
   query.exec("insert into student values(1,'yafei0',1)");
   query.exec("insert into student values(2,'yafei1',1)");
   query.exec("insert into student values(3,'yafei2',2)");
   
   query.exec("create table course (id int primary key, name vchar, teacher vchar)");
   query.exec("insert into course values(1,'Math','yafeilinux1')");
   query.exec("insert into course values(2,'English','yafeilinux2')");
   query.exec("insert into course values(3,'Computer','yafeilinux3')");
   return true;
}
#endif // CONNECTION_H

       在这里建立了两个表,student表中有一字段是course,它是int型的,而course表的主键也是int型的。如果要将course字段和course表进行关联,它们的类型就必须相同,一定要注意这一点。

3.更改main.cpp文件内容如下:#include "mainwindow.h"
#include <QApplication>
#include "connection.h"

int main(int argc, char *argv[])
{
    QApplication a(argc, argv);
    if(!createConnection()) return 1;
    MainWindow w;
    w.show();
   
    return a.exec();
}


4.然后到mainwindow.h文件中,先包含头文件:
#include<QSqlRelationalTableModel>
然后添加private类型对象定义:
QSqlRelationalTableModel *model;


5.到设计模式,往界面上拖放一个Table View部件。

6.到mainwindow.cpp文件中,在构造函数里添加如下代码:
model = new QSqlRelationalTableModel(this);
//字段变化时写入数据库
model->setEditStrategy(QSqlTableModel::OnFieldChange);
model->setTable("student");
//将student表的第三个字段设为course表的id字段的外键,
//并将其显示为course表的name字段的值
model->setRelation(2,QSqlRelation("course","id","name"));
model->setHeaderData(0, Qt::Horizontal, QObject::tr("ID"));
model->setHeaderData(1, Qt::Horizontal, QObject::tr("Name"));
model->setHeaderData(2, Qt::Horizontal, QObject::tr("Course"));
model->select();
ui->tableView->setModel(model);


       这里修改了model的提交策略,OnFieldChange表示只要字段被改动就马上写入数据库,这样就不需要我们再执行提交函数了。setRelation()函数实现了创建外键,注意它的格式就行了。

7.运行程序,效果如下图所示。



       可以看到Course字段已经不再是编号,而是具体的课程名称了。关于外键,大家也应该有一定的认识了吧,说简单点就是将两个相关的表建立一个桥梁,让它们关联起来。

二、使用委托

       有时我们也希望,如果用户更改课程字段,那么只能在课程表中已有的课程中进行选择,而不能随意填写课程。Qt中还提供了一个QSqlRelationalDelegate委托类,它可以为QSqlRelationalTableModel显示和编辑数据。这个委托为一个外键提供了一个QComboBox部件来显示所有可选的数据,这样就显得更加人性化了。使用这个委托很简单,我们先在mainwindow.cpp文件中添加头文件:#include <QSqlRelationalDelegate>
然后继续在构造函数中添加如下一行代码:
ui->tableView->setItemDelegate(new QSqlRelationalDelegate(ui->tableView));


       运行程序,效果如下图所示。


结语
我们可以根据自己的需要来选择使用哪个模型。如果熟悉SQL语法,又不需要将所有的数据都显示出来,那么只需要使用QSqlQuery就可以了。对于QSqlTableModel,它主要是用来显示一个单独的表格,而QSqlQueryModel可以用来显示任意一个结果集,如果想显示任意一个结果集,而且想使其可读写,那么建议子类化QSqlQueryModel,然后重新实现flags()和setData()函数。

源码下载:
返回主目录

langziyang 发表于 2018-8-30 07:33:11

同时多表查询是否model只否只能使用QSqlQueryModel来操作?其它两种model都不行?

staticnull 发表于 2019-3-4 19:52:35

//并将其显示为course表的name字段的值
model->setRelation(2,QSqlRelation("course","id","name"));
老师您能帮忙再详细解释一下这个函数吗?看了帮助手册,还是有点不懂,就这个例子。谢谢

yafeilinux 发表于 2019-3-5 00:10:56

staticnull 发表于 2019-3-4 19:52
//并将其显示为course表的name字段的值
model->setRelation(2,QSqlRelation("course","id","name"));
老师 ...

第一次看都会有点懵,最好的办法是自己去尝试,多改改后面的参数,摸索下就明白了。

chanlon 发表于 2019-8-2 10:37:36

使用QSqlRelationalDelegate实现Course的编辑时希望可以设置Course的值为null,该如何实现?
页: [1]
查看完整版本: 第26篇 Qt5之数据库(六)SQL关系表格模型QSqlRelationalTableModel