共有 main.cpp, Widget.h, Widget.cpp, Widget.ui, MyModel.h, MyModel.cpp 六个文件。
可从此下载整个工程文件: /Files/biao/ModelDemo.zip
关键在于Model中的数据是动态加载的,而不是一次性全加载到Model中。
/***************************************************
* main.cpp
***************************************************/
#include <QtGui/QApplication>
#include "Widget.h"
int main(int argc, char *argv[]) {
QApplication a(argc, argv);
Widget w;
w.show();
return a.exec();
}
/***************************************************
* Widget.h
***************************************************/
#ifndef WIDGET_H
#define WIDGET_H
#include <QWidget>
namespace Ui {
class Widget;
}
class Widget : public QWidget {
Q_OBJECT
public:
explicit Widget(QWidget *parent = 0);
~Widget();
private:
Ui::Widget *ui;
};
#endif // WIDGET_H
/***************************************************
* Widget.cpp
***************************************************/
#include "Widget.h"
#include "ui_Widget.h"
#include "MyModel.h"
Widget::Widget(QWidget *parent) : QWidget(parent), ui(new Ui::Widget) {
ui->setupUi(this);
MyModel *model = new MyModel(1000, 20, this);
ui->tableView->setModel(model);
ui->tableView->horizontalHeader()->setStretchLastSection(true);
}
Widget::~Widget() {
delete ui;
}
/***************************************************
* MyModel.h
***************************************************/
#ifndef MYMODEL_H
#define MYMODEL_H
#include <QAbstractTableModel>
#include <QHash>
#include <QList>
class MyModel : public QAbstractTableModel {
public:
explicit MyModel(int rowCount = 1000, int pageSize = 10, QObject *parent = 0);
virtual int columnCount(const QModelIndex &parent = QModelIndex()) const;
virtual int rowCount(const QModelIndex &parent = QModelIndex()) const;
virtual QVariant data( const QModelIndex &index, int role = Qt::DisplayRole) const;
private:
QHash<int, int> m_pageHash; // key是页码值, value是此页的第一项数据在dataList中的下标值
QList<int> m_dataList; // 存储数据的容器
int m_pageSize; // 每页显示的数据量
int m_rowCount; // 要显示的行数
// 这个下标是有规律可寻的, 但是是无序的.
// 同一页码中的数值的下标是有序的, 不同页码的数据之间是无序的.
int indexOfRow(int row) const;
// 读取数据, 可是以模拟生成的, 实际工程中很可能是从数据库中动态读取
void fetchData(int page);
};
#endif // MYMODEL_H
/***************************************************
* MyModel.cpp
***************************************************/
#include "MyModel.h"
#include <QDebug>
MyModel::MyModel(int rowCount, int pageSize, QObject *parent) : QAbstractTableModel(parent) {
this->m_pageSize = pageSize;
this->m_rowCount = rowCount;
}
intMyModel::columnCount(const QModelIndex &parent) const {
Q_UNUSED(parent);
return 1; // 以一列为示例, 如果是多列, 则data list保存的应该是对象
}
int MyModel::rowCount (const QModelIndex &parent) const {
Q_UNUSED(parent);
return m_rowCount;
}
QVariant MyModel::data( const QModelIndex &index, int role) const {
if (!index.isValid()) { return QVariant(); }
if (Qt::DisplayRole == role) {
int i = indexOfRow(index.row());
return m_dataList.at(i);
}
return QVariant();
}
int MyModel::indexOfRow(int row) const {
int page = row / m_pageSize; // 每页显示pageSize条记录
if (!m_pageHash.contains(page)) {
// 如果此面的数据不存在, 则读取数据到 data list里
(const_cast<MyModel *>(this))->fetchData(page);
}
return m_pageHash.value(page) + row % m_pageSize;
}
// 每次加载数据时,如果是耗时任务, 可以使用进度条显示加载进度
// 也可以取消加载, 但是这个时候data()函数中返回一个非有效的index
void MyModel::fetchData(int page) {
int pageStartIndex = m_dataList.count(); // 存储此页码与其所对应的开始下标值
m_pageHash.insert(page, pageStartIndex);
// 例如在这里使用分页查询, 从数据库里读取数据
for (int i = 0; i < m_pageSize; ++i) {
int data = page * m_pageSize + i;
m_dataList.append(data);
// 快速拖动滚动条, 可以看到中间很多没必要的数据没有产生.
qDebug() << data;
}
}