共有 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;

}

}

05-27 07:48