【Qt官方例程学习笔记】Address Book Example(代理模型)
地址簿示例展示了如何使用代理模型在单个模型的数据上显示不同的视图。
本例提供了一个地址簿,允许按字母顺序将联系人分组为9组:ABC、DEF、GHI、…,VW,…XYZ。这是通过在同一个模型上使用多个视图实现的,每个视图都使用QSortFilterProxyModel类的一个实例进行过滤。
综述
地址簿包含5个类:MainWindow、AddressWidget、TableModel、NewAddressTab和AddDialog。MainWindow类使用AddressWidget作为其中心小部件,并提供文件和工具菜单。
AddressWidget类是一个QTabWidget子类,用于操作示例中显示的10个选项卡:9个字母组选项卡和NewAddressTab的一个实例。NewAddressTab类是QWidget的一个子类,它只在地址簿为空时使用,提示用户添加一些联系人。AddressWidget还与TableModel实例交互,以添加、编辑和删除地址簿中的条目。
TableModel是QAbstractTableModel的子类,它提供了访问数据的标准模型/视图API。它包含一个已添加的联系人列表。然而,这些数据并不是都显示在单个选项卡中。相反,根据字母表组,QTableView被用来提供相同数据的9个不同视图。
QSortFilterProxyModel是负责从所有联系人中过滤每组联系人的类。每个代理模型都使用一个QRegExp来过滤不属于相应字母组的联系人。AddDialog类用于从用户获取地址簿的信息。这个QDialog子类由NewAddressTab实例化来添加联系人,并由AddressWidget实例化来添加和编辑联系人。
我们从TableModel实现开始。0
TableModel Class Definition
TableModel类通过子类化QAbstractTableModel提供标准API来访问联系人列表中的数据。为此必须实现的基本函数是:rowCount()、columnCount()、data()、headerData()。要使TableModel可编辑,它必须提供insertRows()、removeRows()、setData()和flags()函数的实现。
struct Contact
{
QString name;
QString address; bool operator==(const Contact &other) const
{
return name == other.name && address == other.address;
}
}; inline QDataStream &operator<<(QDataStream &stream, const Contact &contact)
{
return stream << contact.name << contact.address;
} inline QDataStream &operator>>(QDataStream &stream, Contact &contact)
{
return stream >> contact.name >> contact.address;
} class TableModel : public QAbstractTableModel
{
Q_OBJECT public:
TableModel(QObject *parent = );
TableModel(QList<Contact> contacts, QObject *parent = ); int rowCount(const QModelIndex &parent) const override;
int columnCount(const QModelIndex &parent) const override;
QVariant data(const QModelIndex &index, int role) const override;
QVariant headerData(int section, Qt::Orientation orientation, int role) const override;
Qt::ItemFlags flags(const QModelIndex &index) const override;
bool setData(const QModelIndex &index, const QVariant &value, int role = Qt::EditRole) override;
bool insertRows(int position, int rows, const QModelIndex &index = QModelIndex()) override;
bool removeRows(int position, int rows, const QModelIndex &index = QModelIndex()) override;
QList<Contact> getContacts() const; private:
QList<Contact> contacts;
};
使用了两个构造函数,一个默认构造函数使用TableModel自己的QList<Contact>,另一个构造函数使用QList<Contact>作为参数,以方便使用。
TableModel Class Implementation
我们实现头文件中定义的两个构造函数。第二个构造函数使用参数值初始化模型中的联系人列表。
TableModel::TableModel(QObject *parent)
: QAbstractTableModel(parent)
{
} TableModel::TableModel(QList<Contact> contacts, QObject *parent)
: QAbstractTableModel(parent)
, contacts(contacts)
{
}
The rowCount() and columnCount()返回模型的维数。然而,rowCount()的值将根据添加到地址簿的联系人数量而变化,columnCount()的值总是2,因为我们只需要名称和地址列的空间。
注意:q_unavailable()宏防止编译器生成关于未使用参数的警告。
int TableModel::rowCount(const QModelIndex &parent) const
{
Q_UNUSED(parent);
return contacts.size();
} int TableModel::columnCount(const QModelIndex &parent) const
{
Q_UNUSED(parent);
return ;
}
data()函数的作用是:根据提供的模型索引的内容返回名称或地址。模型索引中存储的行号用于引用联系人列表中的项。选择操作由QItemSelectionModel处理,该模型将用AddressWidget解释。
QVariant TableModel::data(const QModelIndex &index, int role) const
{
if (!index.isValid())
return QVariant(); if (index.row() >= contacts.size() || index.row() < )
return QVariant(); if (role == Qt::DisplayRole) {
const auto &contact = contacts.at(index.row()); if (index.column() == )
return contact.name;
else if (index.column() == )
return contact.address;
}
return QVariant();
}
headerData()函数的作用是:显示表的标题、名称和地址。如果您的地址簿需要编号的条目,可以使用我们在本例中隐藏的垂直标题(请参阅AddressWidget实现)。
QVariant TableModel::headerData(int section, Qt::Orientation orientation, int role) const
{
if (role != Qt::DisplayRole)
return QVariant(); if (orientation == Qt::Horizontal) {
switch (section) {
case :
return tr("Name"); case :
return tr("Address"); default:
return QVariant();
}
}
return QVariant();
}
在添加新数据之前调用insertRows()函数,否则将不会显示数据。调用beginInsertRows()和endInsertRows()函数,以确保所有连接的视图都知道这些更改。
bool TableModel::insertRows(int position, int rows, const QModelIndex &index)
{
Q_UNUSED(index);
beginInsertRows(QModelIndex(), position, position + rows - ); for (int row = ; row < rows; ++row)
contacts.insert(position, { QString(), QString() }); endInsertRows();
return true;
}
removeRows()函数的作用是:删除数据。再次调用beginRemoveRows()和endRemoveRows(),以确保所有连接的视图都知道这些更改。
bool TableModel::removeRows(int position, int rows, const QModelIndex &index)
{
Q_UNUSED(index);
beginRemoveRows(QModelIndex(), position, position + rows - ); for (int row = ; row < rows; ++row)
contacts.removeAt(position); endRemoveRows();
return true;
}
setData()函数的作用是将数据逐项而不是逐行插入表中。这意味着要填充地址簿中的一行,必须调用两次setData(),因为每一行有两列。发出dataChanged()信号很重要,因为它告诉所有连接的视图更新其显示。
bool TableModel::setData(const QModelIndex &index, const QVariant &value, int role)
{
if (index.isValid() && role == Qt::EditRole) {
int row = index.row(); auto contact = contacts.value(row); if (index.column() == )
contact.name = value.toString();
else if (index.column() == )
contact.address = value.toString();
else
return false; contacts.replace(row, contact);
emit dataChanged(index, index, {role}); return true;
} return false;
}
flag()函数的作用是:返回给定索引的项标志。
Qt::ItemFlags TableModel::flags(const QModelIndex &index) const
{
if (!index.isValid())
return Qt::ItemIsEnabled; return QAbstractTableModel::flags(index) | Qt::ItemIsEditable;
}
我们设置Qt::ItemIsEditable标志,因为我们希望允许编辑TableModel。虽然在这个例子中我们没有使用QTableView对象的编辑特性,但是我们在这里启用了它们,这样我们就可以在其他程序中重用这个模型。
TableModel中的最后一个函数getContacts()返回QList<Contact>对象,该对象保存通讯录中的所有联系人。稍后,我们将使用此函数获取联系人列表,以检查现有条目,将联系人写入文件并将其读取回来。AddressWidget给出了进一步的解释。
QList<Contact> TableModel::getContacts() const
{
return contacts;
}
AddressWidget Class Definition
AddressWidget类在技术上是本例中涉及的主要类,因为它提供了添加、编辑和删除联系人、将联系人保存到文件中以及从文件中加载联系人的功能。
class AddressWidget : public QTabWidget
{
Q_OBJECT public:
AddressWidget(QWidget *parent = );
void readFromFile(const QString &fileName);
void writeToFile(const QString &fileName); public slots:
void showAddEntryDialog();
void addEntry(QString name, QString address);
void editEntry();
void removeEntry(); signals:
void selectionChanged (const QItemSelection &selected); private:
void setupTabs(); TableModel *table;
NewAddressTab *newAddressTab;
QSortFilterProxyModel *proxyModel;
};
AddressWidget扩展了QTabWidget,以便容纳10个标签(NewAddressTab和9个字母组标签),并操作标签、TableModel对象、proxyModel、QSortFilterProxyModel对象(我们用来过滤条目)和tableView、QTableView对象。
AddressWidget Class Implementation
AddressWidget构造函数接受父小部件并实例化NewAddressTab、TableModel和QSortFilterProxyModel。添加NewAddressTab对象(用于指示地址簿为空),其余9个选项卡使用setupTabs()设置。
AddressWidget::AddressWidget(QWidget *parent)
: QTabWidget(parent)
{
table = new TableModel(this);
newAddressTab = new NewAddressTab(this);
connect(newAddressTab, &NewAddressTab::sendDetails,
this, &AddressWidget::addEntry); addTab(newAddressTab, "Address Book"); setupTabs();
}
setupTabs()函数的作用是:在AddressWidget中设置9个字母组选项卡、表视图和代理模型。每个代理模型依次设置为使用不区分大小写的QRegExp对象根据相关字母表组过滤联系人名称。表视图也使用相应代理模型的sort()函数按升序排序。
将每个表视图的selectionMode设置为QAbstractItemView::SingleSelection,将selectionBehavior设置为QAbstractItemView::SelectRows,允许用户同时选择一行中的所有项。每个QTableView对象都会自动被赋予一个QItemSelectionModel来跟踪所选索引。
void AddressWidget::setupTabs()
{
QStringList groups;
groups << "ABC" << "DEF" << "GHI" << "JKL" << "MNO" << "PQR" << "STU" << "VW" << "XYZ"; for (int i = ; i < groups.size(); ++i) {
QString str = groups.at(i);
QString regExp = QString("^[%1].*").arg(str); proxyModel = new QSortFilterProxyModel(this);
proxyModel->setSourceModel(table);
proxyModel->setFilterRegExp(QRegExp(regExp, Qt::CaseInsensitive));
proxyModel->setFilterKeyColumn(); QTableView *tableView = new QTableView;
tableView->setModel(proxyModel); tableView->setSelectionBehavior(QAbstractItemView::SelectRows);
tableView->horizontalHeader()->setStretchLastSection(true);
tableView->verticalHeader()->hide();
tableView->setEditTriggers(QAbstractItemView::NoEditTriggers);
tableView->setSelectionMode(QAbstractItemView::SingleSelection); tableView->setSortingEnabled(true); connect(tableView->selectionModel(),
&QItemSelectionModel::selectionChanged,
this, &AddressWidget::selectionChanged); connect(this, &QTabWidget::currentChanged, this, [this](int tabIndex) {
auto *tableView = qobject_cast<QTableView *>(widget(tabIndex));
if (tableView)
emit selectionChanged(tableView->selectionModel()->selection());
}); addTab(tableView, str);
}
}
QItemSelectionModel类提供了一个selectionChanged信号,该信号连接到AddressWidget的selectionChanged()信号。我们还将QTabWidget::currentChanged()信号连接到发出AddressWidget的selectionChanged()的lambda表达式。这些连接对于启用编辑条目是必要的…并删除主窗口“工具”菜单中的输入操作。在MainWindow的实现中有进一步的解释。
地址簿中的每个表视图都作为一个选项卡添加到QTabWidget,并带有从组的QStringList中获得的相关标签。
我们提供了2个addEntry()函数:1是用来接受用户输入的,另一个函数执行向地址簿添加新条目的实际任务。我们将添加条目的职责划分为两部分,以允许newAddressTab无需弹出对话框即可插入数据。
第一个addEntry()函数是一个插槽,连接到主窗口的Add条目…行动。这个函数创建一个AddDialog对象,然后调用第二个addEntry()函数来实际将联系人添加到表中。
void AddressWidget::showAddEntryDialog()
{
AddDialog aDialog; if (aDialog.exec()) {
QString name = aDialog.nameText->text();
QString address = aDialog.addressText->toPlainText(); addEntry(name, address);
}
}
基本验证在第二个addEntry()函数中完成,以防止地址簿中的重复条目。正如在TableModel中提到的,这是我们需要getter方法getContacts()的部分原因。
void AddressWidget::addEntry(QString name, QString address)
{
if (!table->getContacts().contains({ name, address })) {
table->insertRows(, , QModelIndex()); QModelIndex index = table->index(, , QModelIndex());
table->setData(index, name, Qt::EditRole);
index = table->index(, , QModelIndex());
table->setData(index, address, Qt::EditRole);
removeTab(indexOf(newAddressTab));
} else {
QMessageBox::information(this, tr("Duplicate Name"),
tr("The name \"%1\" already exists.").arg(name));
}
}
如果模型还没有包含具有相同名称的条目,则调用setData()将名称和地址插入第一列和第二列。否则,我们将显示一个QMessageBox来通知用户。
注意:一旦添加了联系人,newAddressTab将被删除,因为地址簿不再为空。
编辑条目只是更新联系人地址的一种方法,因为示例不允许用户更改现有联系人的名称。
首先,我们使用QTabWidget::currentWidget()获取活动选项卡的QTableView对象。然后我们从tableView中提取selectionModel来获得所选的索引。
void AddressWidget::editEntry()
{
QTableView *temp = static_cast<QTableView*>(currentWidget());
QSortFilterProxyModel *proxy = static_cast<QSortFilterProxyModel*>(temp->model());
QItemSelectionModel *selectionModel = temp->selectionModel(); QModelIndexList indexes = selectionModel->selectedRows();
QString name;
QString address;
int row = -; foreach (QModelIndex index, indexes) {
row = proxy->mapToSource(index).row();
QModelIndex nameIndex = table->index(row, , QModelIndex());
QVariant varName = table->data(nameIndex, Qt::DisplayRole);
name = varName.toString(); QModelIndex addressIndex = table->index(row, , QModelIndex());
QVariant varAddr = table->data(addressIndex, Qt::DisplayRole);
address = varAddr.toString();
}
接下来,我们从用户打算编辑的行中提取数据。此数据显示在具有不同窗口标题的AddDialog实例中。只有对aDialog中的数据进行了更改,才会更新该表。
AddDialog aDialog;
aDialog.setWindowTitle(tr("Edit a Contact")); aDialog.nameText->setReadOnly(true);
aDialog.nameText->setText(name);
aDialog.addressText->setText(address); if (aDialog.exec()) {
QString newAddress = aDialog.addressText->toPlainText();
if (newAddress != address) {
QModelIndex index = table->index(row, , QModelIndex());
table->setData(index, newAddress, Qt::EditRole);
}
}
}
使用removeEntry()函数删除条目。通过QItemSelectionModel对象selectionModel访问所选行,可以删除所选行。只有当用户删除通讯录中的所有联系人时,才会将newAddressTab重新添加到AddressWidget。
void AddressWidget::removeEntry()
{
QTableView *temp = static_cast<QTableView*>(currentWidget());
QSortFilterProxyModel *proxy = static_cast<QSortFilterProxyModel*>(temp->model());
QItemSelectionModel *selectionModel = temp->selectionModel(); QModelIndexList indexes = selectionModel->selectedRows(); foreach (QModelIndex index, indexes) {
int row = proxy->mapToSource(index).row();
table->removeRows(row, , QModelIndex());
} if (table->rowCount(QModelIndex()) == ) {
insertTab(, newAddressTab, "Address Book");
}
}
writeToFile()函数的作用是:保存一个包含通讯录中所有联系人的文件。该文件以自定义.dat格式保存。联系人列表的内容使用QDataStream写入文件。如果文件无法打开,则会显示一个QMessageBox,并显示相关的错误消息。
void AddressWidget::writeToFile(const QString &fileName)
{
QFile file(fileName); if (!file.open(QIODevice::WriteOnly)) {
QMessageBox::information(this, tr("Unable to open file"), file.errorString());
return;
} QDataStream out(&file);
out << table->getContacts();
}
readFromFile()函数的作用是:加载一个包含地址簿中所有联系人的文件,该文件以前使用writeToFile()保存。QDataStream用于将.dat文件的内容读入联系人列表,每个联系人都使用addEntry()添加。
void AddressWidget::readFromFile(const QString &fileName)
{
QFile file(fileName); if (!file.open(QIODevice::ReadOnly)) {
QMessageBox::information(this, tr("Unable to open file"),
file.errorString());
return;
} QList<Contact> contacts;
QDataStream in(&file);
in >> contacts; if (contacts.isEmpty()) {
QMessageBox::information(this, tr("No contacts in file"),
tr("The file you are attempting to open contains no contacts."));
} else {
for (const auto &contact: qAsConst(contacts))
addEntry(contact.name, contact.address);
}
}
NewAddressTab Class Definition
NewAddressTab类提供一个提供信息的选项卡,告诉用户地址簿是空的。它根据地址簿的内容出现和消失,如AddressWidget实现中所述。
NewAddressTab类扩展了QWidget,并包含QLabel和QPushButton。
class NewAddressTab : public QWidget
{
Q_OBJECT public:
NewAddressTab(QWidget *parent = ); public slots:
void addEntry(); signals:
void sendDetails(QString name, QString address); private:
QLabel *descriptionLabel;
QPushButton *addButton;
QVBoxLayout *mainLayout; };
NewAddressTab Class Implementation
构造函数实例化addButton、descriptionLabel并将addButton的信号连接到addEntry()插槽。
NewAddressTab::NewAddressTab(QWidget *parent)
{
Q_UNUSED(parent); descriptionLabel = new QLabel(tr("There are currently no contacts in your address book. "
"\nClick Add to add new contacts.")); addButton = new QPushButton(tr("Add")); connect(addButton, &QAbstractButton::clicked, this, &NewAddressTab::addEntry); mainLayout = new QVBoxLayout;
mainLayout->addWidget(descriptionLabel);
mainLayout->addWidget(addButton, , Qt::AlignCenter); setLayout(mainLayout);
}
addEntry()函数与AddressWidget的addEntry()类似,这两个函数都实例化了一个AddDialog对象。通过发出sendDetails()信号,从对话框中提取数据并发送到AddressWidget的addEntry()插槽。
void NewAddressTab::addEntry()
{
AddDialog aDialog; if (aDialog.exec()) {
QString name = aDialog.nameText->text();
QString address = aDialog.addressText->toPlainText(); emit sendDetails(name, address);
}
}
AddDialog Class Definition
AddDialog类扩展了QDialog,并向用户提供一个QLineEdit和一个QTextEdit来将数据输入地址簿。
class AddDialog : public QDialog
{
Q_OBJECT public:
AddDialog(QWidget *parent = );
QLineEdit *nameText;
QTextEdit *addressText; private:
QLabel *nameLabel;
QLabel *addressLabel;
QPushButton *okButton;
QPushButton *cancelButton;
};
AddDialog Class Implementation
AddDialog的构造函数设置用户界面,创建必要的小部件并将它们放置到布局中。
AddDialog::AddDialog(QWidget *parent)
: QDialog(parent)
{
nameLabel = new QLabel("Name");
addressLabel = new QLabel("Address");
okButton = new QPushButton("OK");
cancelButton = new QPushButton("Cancel"); nameText = new QLineEdit;
addressText = new QTextEdit; QGridLayout *gLayout = new QGridLayout;
gLayout->setColumnStretch(, );
gLayout->addWidget(nameLabel, , );
gLayout->addWidget(nameText, , ); gLayout->addWidget(addressLabel, , , Qt::AlignLeft|Qt::AlignTop);
gLayout->addWidget(addressText, , , Qt::AlignLeft); QHBoxLayout *buttonLayout = new QHBoxLayout;
buttonLayout->addWidget(okButton);
buttonLayout->addWidget(cancelButton); gLayout->addLayout(buttonLayout, , , Qt::AlignRight); QVBoxLayout *mainLayout = new QVBoxLayout;
mainLayout->addLayout(gLayout);
setLayout(mainLayout); connect(okButton, &QAbstractButton::clicked, this, &QDialog::accept);
connect(cancelButton, &QAbstractButton::clicked, this, &QDialog::reject); setWindowTitle(tr("Add a Contact"));
}
为了给对话框提供所需的行为,我们将OK和Cancel按钮连接到对话框的accept()和reject()插槽。由于对话框只充当名称和地址信息的容器,因此我们不需要为它实现任何其他功能。
MainWindow Class Definition
MainWindow类扩展了QMainWindow,并实现了操作地址簿所需的菜单和操作。
class MainWindow : public QMainWindow
{
Q_OBJECT public:
MainWindow(); private slots:
void updateActions(const QItemSelection &selection);
void openFile();
void saveFile(); private:
void createMenus(); AddressWidget *addressWidget;
QMenu *fileMenu;
QMenu *toolMenu;
QAction *openAct;
QAction *saveAct;
QAction *exitAct;
QAction *addAct;
QAction *editAct;
QAction *removeAct;
};
MainWindow类使用AddressWidget作为其中心小部件,并提供具有打开、关闭和退出操作的文件菜单,以及具有添加条目的工具菜单……、编辑条目…并删除输入操作。
MainWindow Class Implementation
MainWindow的构造函数实例化AddressWidget,将其设置为其中心小部件,并调用createMenus()函数。
MainWindow::MainWindow()
{
addressWidget = new AddressWidget;
setCentralWidget(addressWidget);
createMenus();
setWindowTitle(tr("Address Book"));
}
createMenus() 函数的作用是:设置文件和工具菜单,将操作连接到它们各自的槽。两个编辑条目…删除条目操作在默认情况下是禁用的,因为此类操作不能在空地址簿上执行。只有在添加一个或多个联系人时才启用它们。
void MainWindow::createMenus()
{
fileMenu = menuBar()->addMenu(tr("&File")); openAct = new QAction(tr("&Open..."), this);
fileMenu->addAction(openAct);
connect(openAct, &QAction::triggered, this, &MainWindow::openFile);
... editAct = new QAction(tr("&Edit Entry..."), this);
editAct->setEnabled(false);
toolMenu->addAction(editAct);
connect(editAct, &QAction::triggered, addressWidget, &AddressWidget::editEntry); toolMenu->addSeparator(); removeAct = new QAction(tr("&Remove Entry"), this);
removeAct->setEnabled(false);
toolMenu->addAction(removeAct);
connect(removeAct, &QAction::triggered, addressWidget, &AddressWidget::removeEntry); connect(addressWidget, &AddressWidget::selectionChanged,
this, &MainWindow::updateActions);
}
除了将所有操作的信号连接到它们各自的插槽之外,我们还将AddressWidget的selectionChanged()信号连接到它的updateActions()插槽。
openFile()函数的作用是:允许用户在打开文件对话框中选择一个文件。所选文件必须是包含地址簿联系人的自定义.dat文件。此函数是文件菜单中连接到openAct的插槽。
void MainWindow::openFile()
{
QString fileName = QFileDialog::getOpenFileName(this);
if (!fileName.isEmpty())
addressWidget->readFromFile(fileName);
}
The saveFile() function allows the user to save a file with the save file dialog. This function is a slot connected to saveAct in the File menu.
void MainWindow::saveFile()
{
QString fileName = QFileDialog::getSaveFileName(this);
if (!fileName.isEmpty())
addressWidget->writeToFile(fileName);
}
updateActions()函数的作用是:启用和禁用编辑条目…并根据地址簿的内容删除条目。如果地址簿为空,则禁用这些操作;否则,将启用它们。这个函数是一个插槽连接到AddressWidget的selectionChanged()信号。
void MainWindow::updateActions(const QItemSelection &selection)
{
QModelIndexList indexes = selection.indexes(); if (!indexes.isEmpty()) {
removeAct->setEnabled(true);
editAct->setEnabled(true);
} else {
removeAct->setEnabled(false);
editAct->setEnabled(false);
}
}
main() Function
The main function for the address book instantiates QApplication and opens a MainWindow before running the event loop.
int main(int argc, char *argv[])
{
QApplication app(argc, argv);
MainWindow mw;
mw.show();
return app.exec();
}
【Qt官方例程学习笔记】Address Book Example(代理模型)的更多相关文章
- 【Qt官方例程学习笔记】Application Example(构成界面/QAction/退出时询问保存/用户偏好载入和保存/文本文件的载入和保存/QCommandLineParser解析运行参数)
The Application example shows how to implement a standard GUI application with menus, toolbars, and ...
- 【Qt官方例程学习笔记】Raster Window Example(画笔的平移/旋转/缩放应用)
这个例子显示了如何使用QPainter渲染一个简单的QWindow. 值得学习的内容 <QtGui>头文件 #include <QtGui>就可以使用Qt GUI模块中的所有类 ...
- 【Qt官方例程学习笔记】Analog Clock Window Example (画笔的平移/旋转/缩放应用)
这个例子演示了如何使用QPainter的转换和缩放特性来简化绘图. 值得学习的: 定时器事件ID检查: 在定时器事件中检查定时器id是比较好的实践. QPainter抗锯齿: We call QPai ...
- 【Qt官方例程学习笔记】Getting Started Programming with Qt Widgets
创建一个QApplication对象,用于管理应用程序资源,它对于任何使用了Qt Widgets的程序都必要的.对于没有使用Qt Widgets 的GUI应用,可以使用QGuiApplication代 ...
- (转)Qt Model/View 学习笔记 (七)——Delegate类
Qt Model/View 学习笔记 (七) Delegate 类 概念 与MVC模式不同,model/view结构没有用于与用户交互的完全独立的组件.一般来讲, view负责把数据展示 给用户,也 ...
- (转)Qt Model/View 学习笔记 (五)——View 类
Qt Model/View 学习笔记 (五) View 类 概念 在model/view架构中,view从model中获得数据项然后显示给用户.数据显示的方式不必与model提供的表示方式相同,可以与 ...
- GIS案例学习笔记-多边形内部缓冲区地理模型
GIS案例学习笔记-多边形内部缓冲区地理模型 联系方式:谢老师,135-4855-4328,xiexiaokui#qq.com 目的:对于多边形,建立内部缓冲区. 问题:ArcGIS缓冲工具不支持内部 ...
- Spring学习笔记之aop动态代理(3)
Spring学习笔记之aop动态代理(3) 1.0 静态代理模式的缺点: 1.在该系统中有多少的dao就的写多少的proxy,麻烦 2.如果目标接口有方法的改动,则proxy也需要改动. Person ...
- 《Master Bitcoin》学习笔记02——比特币的交易模型
比特币的交易模型 模型基本描述 前面一篇学习笔记01提到了一个交易模型(第三章的内容),在第五章中,除了对这个模型做个详细介绍之外,其实和我上一篇理解的交易模型差不多,一个交易包含输入与输出,比特币是 ...
随机推荐
- Codeforces 372B Counting Rectangles is Fun:dp套dp
题目链接:http://codeforces.com/problemset/problem/372/B 题意: 给你一个n*m的01矩阵(1 <= n,m <= 40). 然后有t组询问( ...
- 分享知识-快乐自己:mybatis 主键回调
以下两种方式实现 主键回掉方式. <!--添加用户信息:主键回调--> <insert id="insertUser" useGeneratedKeys=&quo ...
- linux应用之xampp集成环境的安装及配置(centos)
1.xampp集成环境的下载 在xampp的官网上选择对应系统的版本进行下载,官网地址:https://www.apachefriends.org/zh_cn/index.html #wget htt ...
- poj 1080 Human Gene Functions(lcs,较难)
Human Gene Functions Time Limit: 1000MS Memory Limit: 10000K Total Submissions: 19573 Accepted: ...
- struts2--标签取值
OGNL的概念: OGNL是Object-Graph Navigation Language的缩写,全称为对象图导航语言,是一种功能强大的表达式语言,它通过简单一致的语法,可以任意存取对象的属性或者调 ...
- 用截取的部分图像创建新图像--关于cvGetSubRect,cvGetImage的用法
CvMat* cvGetSubRect(const CvArr* arr, CvMat* submat, CvRect rect)可以把截取图像中需要的区域存入矩阵.把IplImage *传给arr, ...
- linux下导入导出oracle的dmp文件
1.导出dmp件 命令:exp QGTG/\"QGTG@orcl\" file=/usr/fuck.dmp exp QGTG/\"QGTG@orcl\" fil ...
- ES+open-falcon之报警自动发送请求信息
当我们监控nginx的状态码出现错误状态码的时候, 一般的处理方法是通过kibana查询是哪个接口导致从而确定是哪个服务,再进一步登录业务机器查询业务日志确定原因. 我们现在要做的事情就是将 人为的通 ...
- bzoj 1006: 神奇的国度 MCS
题目大意: 弦图的最小染色. 题解: 裸题. #include <vector> #include <cstdio> #include <cstring> #inc ...
- Brunch with a Friend 与朋友共进午餐
brownies 核仁巧克力饼 toast 烤面包 dining room 餐厅 practical 实用的 meal 一餐 combination 组合 pancake 薄煎饼 waffle 华夫饼 ...