7.4 KiB
从头开始完成GraphBuilder工程
实验要求
- 以邻接多重表为存储结构,实现联通无向图的深度优先和广度优先遍历。以指定的节点为起点,分别输出每种遍历下的节点访问序列和相应生成树的边集。
- 以邻接表为存储结构,建立深度优先生成树和广度优先生成树,并以树型输出生成树。
- 测试数据如图:

分析需求
在本节中将分析上述题目要求,大致明确程序所需要完成的模块及功能
由于需要以树型输出生成树,因此最好能够实现可视化控件,其需要支持显示图结构
为了便于添加点和线,故可视化控件需要支持添加及删除点和线的功能
从测试数据图中看出,输入的数据为带权图,因此对于该控件还需要实现添加及修改权重的功能
由于需要进行遍历,为了体现遍历过程,该控件需要实现创建节点和边的访问动画功能
在数据结构层面,需要实现邻接表和邻接多重表两种数据结构
为了避免演示时不同数据结构需要分开建立测试数据,因此需要为两种不同的数据结构实现一个虚基类,并使两种不同的结构均从该虚基类继承并添加转换函数,实现两种不同数据结构的互相转换
程序设计
在本节中,将依据上述需求,对各个需要实现的模块进行更进一步的分析,并给出其实现的过程
从上述需求可以看出,实现最基本的功能大致需要将程序分为两个主要部分:
- 可视化控件:类画板控件,负责处理图中点和线的元素对象的显示、修改等操作,并为数据结构提供功能接口,诸如调用动画等
- 数据结构:需要使用要求的数据结构类将上述可视化控件中的元素对象组织起来,并调用其提供的接口实现如深度优先遍历等功能
下面将依次完成这两个主要部分
可视化控件(画板)
由于需要实现的这个可视化控件的主要功能是对图元对象进行管理并使它们可视化。
QT中已有的QGraphicsView具有相似的功能,在文档中有如下描述:
The QGraphicsView class provides a widget for displaying the contents of a QGraphicsScene
QGraphicsView类是一个将QGraphicsScene场景中的内容可视化显示出来的控件
其中,QGraphicsScene是管理所有图元的场景,其在文档中的描述如下:
The QGraphicsScene class provides a surface for managing large number of 2D graphical items
QGraphicsScene类提供了一个管理大量2D图形项的平台
而QGraphicsScene所管理的图形项均继承自QGraphicsItem,也就是说对于所有继承自QGraphicsItem的图形项目,都可以通过QGraphicsScene进行管理,并使用QGraphicsView来显示
但是由于不论是QGraphicsView还是QGraphicsItem,所提供的功能都不足以满足本次项目的功能,因此需要从它们继承并完成新类如下:
MyGraphicsView:继承自QGraphicsView,添加对鼠标操作的判定和响应、动画队列等功能MyGraphicsVexItem:继承自QGraphicsItem,完成点的绘制和对应功能MyGraphicsLineItem:继承自QGraphicsItem,完成线的绘制和对应功能
下面,我将按照最初的设计思路带大家完成这三个类的实现:
首先,绘制一个点
目前为止,我们面对的任务有:
- 设计点对应的
MyGraphicsVexItem类 - 设计线对应的
MyGraphicsLineItem类 - 设计视图框架
MyGraphicsView将点和线联系起来
按照先挑软柿子捏的逻辑,MyGraphicsVexItem类应当是最先进行设计的
首先,先不考虑其他一切问题,如果仅仅只需要在视图场景中绘制一个点,由于点实际上的显示效果是一个圆,因此可以使用QGraphicsEllipseItem类:
#include <QGraphicsView>
#include <QGraphicsScene>
#include <QGraphicsEllipseItem>
/* Declaration of Graphics View and Graphics Scene */
QGraphicsView* myView = new QGraphicsView;
QGraphicsScene* myScene = new QGraphicsScene;
myScene->setSceneRect(-100, -100, 200, 200);
myView->setScene(myScene);
/* Addition of a point */
//Create and set the top left pos of the new point to (-25, -25) and the radius to 25
QGraphicsEllipseItem* newPoint = new QGraphicsEllipseItem(-25, -25, 50, 50);
myScene->addItem(newPoint);
这样我们就以(0,0)为圆心,25为半径绘制了一个圆。但很显然,这不可能是我们想要的效果,我们需要在绘制出一个简单圆形的基础上添加一些特性,比如点的颜色属性、点的选择状态或是点的Pop out动画
为了在画一个圆的基础上实现这些额外的功能,我们从QGraphicsEllipseItem处继承得到MyGraphicsVexItem类
class MyGraphicsVexItem : public QGraphicsEllipseItem {
//Constructors, Properties, Methods, etc.
}
先别着急:搭建一个测试框架
在开始着手进行类的各种设计之前,为了明确我们到底设计了什么东西出来,我们需要先搭建一个测试效果用的简单程序框架,它不需要任何UI设计,只需要能够测试我们的类就足够了
首先创建一个工程,在mainwindow.cpp中添加如下头文件
#include <QVBoxLayout>
#include <QGraphicsView>
#include <QGraphicsScene>
在MainWindow构造函数中添加如下代码:
//Set layout for central widget
QVBoxLayout* mainLayout = new QVBoxLayout();
ui->centralWidget->setLayout(mainLayout);
//Add GraphicsView widget
QGraphicsView* testView = new QGraphicsView(ui->centralWidget);
QGraphicsScene* testScene = new QGraphicsScene(ui->centralWidget);
testScene->setSceneRect(-150, -150, 300, 300);
testView->setScene(testScene);
//Add widget to layout
mainLayout->addWidget(testView);
编译程序,即可得到一个简单的GraphicsView显示窗口
后续的测试则通过在这个框架上添加或是修改特定的功能来测试效果
迈出第一步:使用参数绘制点
为了能够在新的类里面使用给定的参数画出点,我们需要如下参数:
QPointF center: 几何中心qreal radius: 半径QBrush regBrush: 默认画刷
由于考虑到其他方法也可能会访问或是修改这些参数,因此将他们作为类的成员变量。构造函数则在妥善设置和使用这些参数的基础上,结合父类构造函数完成
具体的修改代码如下:
private:
//Geometry properties
QPointF center;
qreal radius;
//Brush properties
//default color
QBrush regBrush = QBrush(QColor(58, 143, 192));
public:
MyGraphicsVexItem(QPointF _center, qreal _radius, QGraphicsItem* parent = nullptr);
MyGraphicsVexItem::MyGraphicsVexItem(QPointF _center, qreal _radius, QGraphicsItem* parent) :
QGraphicsEllipseItem(_center.x() - _radius, _center.y() - _radius, 2 * _radius, 2 * _radius, parent),
center(_center),
radius(_radius)
{
this->setPen(Qt::NoPen);
this->setBrush(regBrush);
}
此时我们在测试框架中用代码添加一个点测试效果:
//Add a new vex
MyGraphicsVexItem* newVex = new MyGraphicsVexItem(QPointF(0, 0), 10);
testScene->addItem(newVex);