日期:2014-05-16  浏览次数:20487 次

二十七、Qt数据库(七)QSqlRelationalTableModel(转)

QSqlRelationalTableModel,该类为单张的数据库表提供了一个可编辑的数据模型,它支持外键。
我们还是新建Qt4 Gui Application工程,我这里工程名为relationalTableModel ,然后选中QtSql模块,Base class选QWidget。工程建好后,添加C++ Header File ,命名为database.h,更改其内容如下:
#ifndef DATABASE_H
#define DATABASE_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 primary key, 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 // DATABASE_H
我们在这里建立了两个表,student表中有一项是course,它是int型的,而course表的主键也是int型的。如果要将course项和course表进行关联,它们的类型就必须相同,一定要注意这一点。
然后将main.cpp中的内容更改如下:
#include <QtGui/QApplication>
#include "widget.h"
#include "database.h"
int main(int argc, char *argv[])
{
??? QApplication a(argc, argv);
??? if(!createConnection()) return 1;
??? Widget w;
??? w.show();
??? return a.exec();
}


我们在widget.h中添加头文件: #include <QSqlRelationalTableModel>
然后在private中声明对象:??? QSqlRelationalTableModel *model;
我们在widget.ui中添加一个Table View部件到窗体上,然后到widget.cpp中的构造函数里添加如下代码:
??? model = new QSqlRelationalTableModel(this);
??? model->setEditStrategy(QSqlTableModel::OnFieldChange); //属性变化时写入数据库
??? model->setTable("student");
??? model->setRelation(2,QSqlRelation("course","id","name"));
??? //将student表的第三个属性设为course表的id属性的外键,并将其显示为course表的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()函数实现了创建外键,注意它的格式就行了。
运行效果如下:


可以看到Course属性已经不再是编号,而是具体的课程了。关于外键,你也应该有一定的认识了吧,说简单点就是将两个相关的表建立一个桥梁,让它们关联起来。
那么我们也希望,如果用户更改课程属性,那么他只能在课程表中有的课程中进行选择,而不能随意填写课程。在Qt中的QSqlRelationalDelegate委托类就能实现这个功能。我们只需在上面的构造函数的最后添加一行代码: