QSqlRelationalTableModel继承自QSqlTableModel,并且对其进行了扩展,提供了对外键的支持。一个外键就是一个表中的一个属性和其他表中的主键属性之间的一对一的映射。例如,student表中的course属性对应的是course表中的id属性,那么就称属性course是一个外键。因为这里的course属性的值是一些数字,这样的显示很不友好,使用关系表格模型,就可以将它显示为course表中的name属性的值。
1.1. 新建Qt Gui 应用,名称为relationalTableModel,基类为QMainWindow,类名为MainWindow。完成后打开relationalTableModel.pro项目文件,将第一行改为:
QT += coregui sql
然后保存该文件。
1.2.下面向项目中添加新的C++头文件connection.h,并更改其内容如下:
本文福利,莬费领取Qt开发学习资料包、技术视频,内容包括(C++语言基础,Qt编程入门,QT信号与槽机制,QT界面开发-图像绘制,QT网络,QT数据库编程,QT项目实战,QSS,OpenCV,Quick模块,面试题等等)↓↓↓↓↓↓见下面↓↓文章底部点击莬费领取↓↓
#ifndef CONNECTION_H
#define CONNECTION_H
#include
#include
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 primarykey, 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 表进行关联,它们的类型就必须相同,一定要注意这一点。
1.3. 更改main.cpp 文件内容如下:
#include "mainwindow.h"
#include
#include "connection.h"
int main(int argc, char *argv[])
{
QApplication a(argc, argv);
if(!createConnection()) return 1;
MainWindow w;
w.show();
return a.exec();
}
1.4. 然后到mainwindow.h 文件中,先包含头文件:
#include
然后添加private 类型对象声明:
QSqlRelationalTableModel *model;
1.5. 到设计模式,往界面上拖放一个Table View 部件。
1.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() 函数实现了创建外键,注意它的格式就行了。
1.7. 运行程序,效果如下图所示。
可以看到Course 属性已经不再是编号,而是具体的课程了。关于外键,大家也应该有一定的认识了吧,说简单点就是将两个相关的表建立一个桥梁,让它们关联起来。
有时我们也希望,如果用户更改课程属性,那么只能在课程表中有的课程中进行选择,而不能随意填写课程。Qt 中还提供了一个QSqlRelationalDelegate 委托类,它可以为QSqlRelationalTableModel 显示和编辑数据。这个委托为一个外键提供了一个QComboBox 部件来显示所有可选的数据,这样就显得更加人性化了。使用这个委托是很简单的,我们先在mainwindow.cpp 文件中添加头文件#include
ui->tableView->setItemDelegate(
new QSqlRelationalDelegate(ui->tableView));
运行程序。
我们可以根据自己的需要来选择使用哪个模型。如果熟悉SQL语法,又不需要将所有的数据都显示出来,那么只需要使用QSqlQuery就可以了。对于QSqlTableModel,它主要是用来显示一个单独的表格的,而QSqlQueryModel可以用来显示任意一个结果集,如果想显示任意一个结果集,而且想使其可读写,那么建议子类化QSqlQueryModel,然后重新实现flags()和setData()函数。
本文福利,莬费领取Qt开发学习资料包、技术视频,内容包括(C++语言基础,Qt编程入门,QT信号与槽机制,QT界面开发-图像绘制,QT网络,QT数据库编程,QT项目实战,QSS,OpenCV,Quick模块,面试题等等)↓↓↓↓↓↓见下面↓↓文章底部点击莬费领取↓↓