图的实现是一件很麻烦的事情,很多同学可能在学数据结构时只是理解了图的基本操作和遍历原理,但并没有动手实践过。在此,我说说我的实现过程。

首先,在草稿纸上画一个图表,这里是有向图,无向图也一样,如下:

我用的是vector+vector容器作为存储结构,如下:vector+vector)
                                                 |0|——|1|——|3|
                                                 |1|——|2|——|3|
                                                 |2|——|3|
                                                 |3|

第二步,用自然语言建立这个图,注意数据结构的书一般没有说明图是怎么建立的,在此我简述一下:

1)、加入4个顶点——G.InsertVextex(V);

2)、在两个顶点之间增加有向弧——G.AddEdge(x, y)。

这样图就建立了。

我写了两个Graph的构造函数,也可以直接用一个数组实现1)的操作如下:

vector<VNode> v(4);

Graph G(v);

第三步,广度优先遍历和深度优先遍历。遍历的目的是为了对每个顶点进行一些处理,能够遍历这个图是其他操作的基础。甚至可以用遍历的方法建立这个图,就像我之前用先序遍历建立二叉树一个道理。

这三步完成,图的实现就基本解决了。至于最小生成树、关键路径等问题,以后再说。

/*************************
Date : 2013-9-14
Author : DVD0423
Function: 有向图 存储结构如下:(vector+vector)
|0|——*——*——*
|1|——*
|2|——*——*
|3|——*——*——*——*——*
|4|——*——*——*
|5|
|6|——*——*
|7|——*
***************************/
#include <iostream>
#include <vector>
#include <queue>
using namespace std;
typedef int DataType;
const DataType INIT_DATA = -1;
const int NO_NODE = -1;
const int NO_EDGE = -2; class VNode{
public:
//data member
bool visited; //访问标志
DataType data; //顶点内的数据
vector<int> e; //邻接的边
//member function
VNode(DataType val = INIT_DATA, bool flag = false):data(val), visited(flag){}
void Visit()
{
cout<<data<<"\t";
}
bool operator== (VNode &y){
if(data == y.data && e == y.e)
return true;
else
return false;
}
};
class Graph{
public:
//data member
int vexnum; //图中顶点数
int edgenum; //图中边数
vector<VNode> V; //图的存储结构
//member function
Graph()
{
vexnum = 0;
edgenum = 0;
}
Graph(vector<VNode> &v);
int GetVexNum();
int GetEdgeNum();
//节点操作
void InsertVertex(VNode &x);
bool DeleteVertex(int x);
//边的操作
//bool IfEdge(int x, int y); //判断是否有(x, y)的边
bool AddEdge(int x, int y);
bool RemoveEdge(int x, int y);
//邻边操作
void PrintNeighbors(int x); //
VNode &Neighbor(VNode &x, int n); //x的第n个邻接节点,默认n为第一个
//广度遍历和深度遍历
void BreadthFirstSearch();
void BFS(VNode &x); //队列实现
void DepthFirstSearch();
void DFS(VNode &x); //递归函数
~Graph(){};
};
Graph::Graph(vector<VNode> &v)
{
V.assign(v.begin(), v.end());
vexnum = V.size();
edgenum = GetEdgeNum();
} int Graph::GetVexNum()
{
vexnum = V.size();
return vexnum;
} int Graph::GetEdgeNum()
{
edgenum = 0;
for(vector<VNode>::iterator iter = V.begin(); iter != V.end(); ++iter)
{
edgenum += iter->e.size();
}
return edgenum;
} void Graph::InsertVertex(VNode &x)
{
V.push_back(x);
vexnum++;
}
//删除x节点
bool Graph::DeleteVertex(int x)
{
if(x >= V.size() || x < 0)
{
cout<<"没有相应节点"<<endl;
return false;
}
vector<VNode>::iterator iter = V.begin()+x;
V.erase(iter);
vexnum--;
return true;
} bool Graph::AddEdge(int x, int y)
{
if((x >= V.size() || x < 0) || (y >= V.size() || y < 0))
{
cout<<"没有相应节点"<<endl;
return false;
}
int size = V[x].e.size();
for(vector<int>::iterator iter = V[x].e.begin(); iter != V[x].e.end(); ++iter)
{
if(V[*iter] == V[y]) //已经有(x, y)
{
cout<<"已经有弧(x, y)"<<endl;
return false;
}
}
V[x].e.push_back(y); //没有,则把y加入x的邻边队列中
edgenum++;
return true;
}
bool Graph::RemoveEdge(int x, int y)
{
if((x >= V.size() || x < 0) || (y >= V.size() || y < 0))
{
cout<<"没有相应节点"<<endl;
return false;
}
for(vector<int>::iterator iter = V[x].e.begin(); iter != V[x].e.end(); ++iter)
{
if(V[*iter] == V[y])
{
V[x].e.erase(iter);
edgenum--;
return true;
}
}
cout<<"没有弧(x, y)"<<endl;
return false;
} //邻边操作
void Graph::PrintNeighbors(int x)
{
cout<<"列出节点"<<x<<"的所有邻接节点"<<endl;
for(vector<int>::iterator iter = V[x].e.begin(); iter != V[x].e.end(); ++iter)
{
cout<<*iter<<"\t";
}
cout<<endl;
} VNode &Graph::Neighbor(VNode &x, int n = 1)
{ if(n > x.e.size() || n <= 0)
{
cout<<"没有相应边"<<endl;
exit(0);
}
return V[(x.e[n-1])];
}
/******
遍历
**********/
void Graph::BreadthFirstSearch()
{
for(int i = 0; i != vexnum; ++i)
{
if(!V[i].visited)
BFS(V[i]);
}
for(vector<VNode>::iterator k = V.begin(); k != V.end(); ++k)
k->visited = false; }
void Graph::BFS(VNode &x)
{
queue<VNode> q;
q.push(x);
while(!q.empty())
{
VNode &_x = q.front();
if(!_x.visited)
{
_x.Visit();
_x.visited = true;
int size = _x.e.size(); for(int i = 0; i < size; ++i)
{
VNode &w = V[_x.e[i]];
if(!w.visited)
{
w.Visit();
w.visited = true;
q.push(w);
}
}
}
q.pop();
}
} void Graph::DepthFirstSearch()
{
for(int i = 0; i != vexnum; ++i)
{
if(!V[i].visited)
DFS(V[i]);
}
for(vector<VNode>::iterator k = V.begin(); k != V.end(); ++k)//改变访问标志,以便下次访问
k->visited = false;
}
void Graph::DFS(VNode &x)
{ x.Visit();
x.visited = true;
int size = x.e.size();
for(int i = 0; i < size; ++i)
{
VNode &w = V[x.e[i]];
if(!w.visited)
DFS(w);
}
}

下面是main函数

int main()
{
Graph G;
vector<VNode> v(4);
for(int i = 0; i != 4; ++i)
{
v[i].data = i; //节点信息
G.InsertVertex(v[i]);
}
//Graph G(v);
G.AddEdge(0, 1);
G.AddEdge(0, 3);
G.AddEdge(1, 2);
G.AddEdge(2, 3);
G.AddEdge(1, 3); G.BreadthFirstSearch();
return 0;
}

C++数据结构之图的更多相关文章

  1. python数据结构之图的实现

    python数据结构之图的实现,官方有一篇文章介绍,http://www.python.org/doc/essays/graphs.html 下面简要的介绍下: 比如有这么一张图: A -> B ...

  2. hdu 1233:还是畅通工程(数据结构,图,最小生成树,普里姆(Prim)算法)

    还是畅通工程 Time Limit : 4000/2000ms (Java/Other)   Memory Limit : 65536/32768K (Java/Other) Total Submis ...

  3. 利用python+graphviz绘制数据结构关系图和指定目录下头文件包含关系图

    作为一名linux系统下的C语言开发,日常工作中经常遇到两个问题: 一是分析代码过程中,各种数据结构互相关联,只通过代码很难理清系统中所有结构体的整体架构,影响代码消化的效率; 二是多层头文件嵌套包含 ...

  4. python数据结构之图的实现方法

    python数据结构之图的实现方法 本文实例讲述了python数据结构之图的实现方法.分享给大家供大家参考.具体如下: 下面简要的介绍下: 比如有这么一张图:     A -> B     A ...

  5. python数据结构之图深度优先和广度优先实例详解

    本文实例讲述了python数据结构之图深度优先和广度优先用法.分享给大家供大家参考.具体如下: 首先有一个概念:回溯 回溯法(探索与回溯法)是一种选优搜索法,按选优条件向前搜索,以达到目标.但当探索到 ...

  6. 数据结构之图 Part2 - 1

    邻接矩阵 网上很少有C# 写图的数据结构的例子,实际的项目中也从来没用过Array 这坨东西,随手写个,勿喷. namespace LH.GraphConsole { public struct Gr ...

  7. 数据结构之图 Part1

    Part 1 预计使用7天的时间来过掉图相关的数据结构.第一天主要是一天图的基本概念,熟练掌握定义是一切交流和沟通的基础. 1定义 1.1图 有穷非空顶点,外加边. G(V,E) Graph Vert ...

  8. 数据结构之--图(Graphics)

    1.1:图的定义和术语   图是一种比线性表和树更为复杂的数据结构.在线性表中,数据元素之间仅有线性关系,每个元素仅有一个直接前驱和一个直接后继:在树形结构中,数据元素之间有着明显的层次关系,并且每一 ...

  9. 【js数据结构】图的深度优先搜索与广度优先搜索

    图类的构建 function Graph(v) {this.vertices = v;this.edges = 0;this.adj = []; for (var i = 0; i < this ...

随机推荐

  1. table边框不显示

    今日在做报表的时候发现,最后一行隐藏后整个报表的下边框会不显示,猜测是td的边框隐藏后但table并未设置边框,导致下边框没有出现.因此设置了table边框后问题解决.table和td的边框关系如下实 ...

  2. Android应用开发基础篇(6)-----Service

    链接地址:http://www.cnblogs.com/lknlfy/archive/2012/02/20/2360336.html 一.概述 我们知道,Service是Android的四大组件之一. ...

  3. BZOJ 1083: [SCOI2005]繁忙的都市(MST)

    裸的最小生成树..直接跑就行了 ---------------------------------------------------------------------- #include<c ...

  4. 设置windows密码只存在NTLM-Hash下

    修改注册表 HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\Control\Lsa 下添加名为"NoLMHash"的DWORD值,并设置为1 ...

  5. IOS 学习笔记(5) 控件 文本视图(UITextView)的使用方法

    相对于UILabell所支持的较短文本内容,UITextView对于长文本的支持更好.UITextView能够以滚动的方式全部浏览到长文本,并且就像UILabel那样,从ISO6,他也提供了对NSAt ...

  6. [HDU 4666]Hyperspace[最远曼哈顿距离][STL]

    题意: 许多 k 维点, 求这些点之间的最远曼哈顿距离. 并且有 q 次操作, 插入一个点或者删除一个点. 每次操作之后均输出结果. 思路: 用"疑似绝对值"的思想, 维护每种状态 ...

  7. C++_template_栈的链式存储及实现

    由于在C++数据结构中的代码不完整,特补全.等日后当工程库调用. 若有疑问,请留言. #include<iostream> using namespace std; template< ...

  8. golang实现tcp接入服务器

    接入服务器和后端业务服务其维持tcp连接,多个前端请求通过接入服务器访问后端业务服务器,接入服务器可以方便增加路由功能,维护多个业务服务器,根据消息ID路由到具体的业务服务器. 项目目录如下 simp ...

  9. [每日一题] 11gOCP 1z0-052 :2013-09-15 Enterprise Manager Support Workbench..................B9

    转载请注明出处:http://blog.csdn.net/guoyjoe/article/details/11715219 正确答案:ABD EnterpriseManger Support Work ...

  10. 开始QT+OpenCV学问

    最近一个月.由于超声造影软件工具做.因此,开始接触OpenCV.使用OpenCV的话.除了图像处理,其它功能都非常弱.所以又開始学习MFC. 从原先的.net C#编程环境一下变成MFC还真有点不习惯 ...