共有 main.cpp, Widget.h, Widget.cpp, Widget.ui, MyModel.h, MyModel.cpp 六個(gè)文件。
可從此下載整個(gè)工程文件: /Files/biao/ModelDemo.zip
關(guān)鍵在于Model中的數(shù)據(jù)是動(dòng)態(tài)加載的,而不是一次性全加載到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是頁(yè)碼值, value是此頁(yè)的第一項(xiàng)數(shù)據(jù)在dataList中的下標(biāo)值
QList<int> m_dataList; // 存儲(chǔ)數(shù)據(jù)的容器
int m_pageSize; // 每頁(yè)顯示的數(shù)據(jù)量
int m_rowCount; // 要顯示的行數(shù)
// 這個(gè)下標(biāo)是有規(guī)律可尋的, 但是是無(wú)序的.
// 同一頁(yè)碼中的數(shù)值的下標(biāo)是有序的, 不同頁(yè)碼的數(shù)據(jù)之間是無(wú)序的.
int indexOfRow(int row) const;
// 讀取數(shù)據(jù), 可是以模擬生成的, 實(shí)際工程中很可能是從數(shù)據(jù)庫(kù)中動(dòng)態(tài)讀取
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;
}
int MyModel::columnCount(const QModelIndex &parent) const {
Q_UNUSED(parent);
return 1; // 以一列為示例, 如果是多列, 則data list保存的應(yīng)該是對(duì)象
}
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; // 每頁(yè)顯示pageSize條記錄
if (!m_pageHash.contains(page)) {
// 如果此面的數(shù)據(jù)不存在, 則讀取數(shù)據(jù)到 data list里
(const_cast<MyModel *>(this))->fetchData(page);
}
return m_pageHash.value(page) + row % m_pageSize;
}
// 每次加載數(shù)據(jù)時(shí),如果是耗時(shí)任務(wù), 可以使用進(jìn)度條顯示加載進(jìn)度
// 也可以取消加載, 但是這個(gè)時(shí)候data()函數(shù)中返回一個(gè)非有效的index
void MyModel::fetchData(int page) {
int pageStartIndex = m_dataList.count(); // 存儲(chǔ)此頁(yè)碼與其所對(duì)應(yīng)的開始下標(biāo)值
m_pageHash.insert(page, pageStartIndex);
// 例如在這里使用分頁(yè)查詢, 從數(shù)據(jù)庫(kù)里讀取數(shù)據(jù)
for (int i = 0; i < m_pageSize; ++i) {
int data = page * m_pageSize + i;
m_dataList.append(data);
// 快速拖動(dòng)滾動(dòng)條, 可以看到中間很多沒(méi)必要的數(shù)據(jù)沒(méi)有產(chǎn)生.
qDebug() << data;
}
}
@import url(http://m.shnenglu.com/CuteSoft_Client/CuteEditor/Load.ashx?type=style&file=SyntaxHighlighter.css);@import url(/css/cuteeditor.css);