JavaScript数据结构和算法之图和图算法


Posted in Javascript onFebruary 11, 2015

图的定义

图(Graph)是由顶点的有穷非空集合和顶点之间边的集合组成,通常表示为:G(V,E),其中,G表示一个图,V是图G中顶点的集合,E是图G中边的集合。

有向图

JavaScript数据结构和算法之图和图算法

有向边:若从顶点Vi到Vj的边有方向,则称这条边为有向边,也成为弧(Arc),用有序偶<Vi,Vj>来表示,Vi称为弧尾,Vj称为弧头。

无序图

JavaScript数据结构和算法之图和图算法

无向边:若顶点Vi到Vj之间的边没有方向,则称这条边为无向边(Edge),用无序偶(Vi,Vj)来表示。

简单图

简单图:在图结构中,若不存在顶点到其自身的边,且同一条边不重复出现,则称这样的图为简单图。

图类

表示顶点

创建图类的第一步就是要创建一个Vertex类来保存顶点和边。这个类的作用和链表、二叉搜索树的Node类一样。Vertex类有两个数据成员:一个用于标识顶点,另一个表明是否被访问过的布尔值。分别被命名为label和wasVisited。

function Vertex(label){

    this.label = label;

}

我们将所有顶点保存在数组中,在图类里,可以通过他们在数组中的位置引用他们

表示边

图的实际信息都保存在“边”上面,因为他们描述了图的结构。二叉树的一个父节点只能有两个子节点,而图的结构却要灵活得多,一个顶点既可以有一条边,也可以有多条边和它相连。

我们将表示图的边的方法成为邻接表或者邻接表数组。它将存储由顶点的相邻顶点列表构成的数组

构建图

定义如下一个Graph类:

function Graph(v){

    this.vertices = v;//vertices至高点

    this.edges = 0;

    this.adj = [];

    for(var i =0;I<this.vertices;++i){

        this.adj[i] = [];

        this.adj[i].push('');

    }

    this.addEdge = addEdge;

    this.toString = toString;

}

这个类会记录一个图表示了多少条边,并使用一个长度与图的顶点数来记录顶点的数量。
function addEdge(){

    this.adj[v].push(w);

    this.adj[w].push(v);

    this.edges++;

}

这里我们使用for循环为数组中的每个元素添加一个子数组来存储所有的相邻顶点,并将所有元素初始化为空字符串。

图的遍历

深度优先遍历

深度优先遍历(DepthFirstSearch),也有称为深度优先搜索,简称为DFS。

比如在一个房间内寻找一把钥匙,无论从哪一间房间开始都可以,将房间内的墙角、床头柜、床上、床下、衣柜、电视柜等挨个寻找,做到不放过任何一个死角,当所有的抽屉、储藏柜中全部都找遍后,接着再寻找下一个房间。

深度优先搜索:

JavaScript数据结构和算法之图和图算法

深度优先搜索就是访问一个没有访问过的顶点,将他标记为已访问,再递归地去访问在初始顶点的邻接表中其他没有访问过的顶点

为Graph类添加一个数组:

this.marked = [];//保存已访问过的顶点

for(var i=0;i<this.vertices;++i){

    this.marked[i] = false;//初始化为false

}

深度优先搜索函数:

function dfs(v){

    this.marked[v] = true;

    //if语句在这里不是必须的

    if(this.adj[v] != undefined){

        print("Visited vertex: " + v );

        for each(var w in this.adj[v]){

            if(!this.marked[w]){

                this.dfs(w);

            }

        }

    }

}

广度优先搜索

广度优先搜索(BFS)属于一种盲目搜寻法,目的是系统地展开并检查图中的所有节点,以找寻结果。换句话说,它并不考虑结果的可能位置,彻底地搜索整张图,直到找到结果为止。

广度优先搜索从第一个顶点开始,尝试访问尽可能靠近它的顶点,如下图所示:

JavaScript数据结构和算法之图和图算法

其工作原理为:

 1. 首先查找与当前顶点相邻的未访问的顶点,将其添加到已访问顶点列表及队列中;
 2. 然后从图中取出下一个顶点v,添加到已访问的顶点列表
 3. 最后将所有与v相邻的未访问顶点添加到队列中
下面是广度优先搜索函数的定义:

function bfs(s){

    var queue = [];

    this.marked = true;

    queue.push(s);//添加到队尾

    while(queue.length>0){

        var v = queue.shift();//从队首移除

        if(v == undefined){

            print("Visited vertex: " + v);

        }

        for each(var w in this.adj[v]){

            if(!this.marked[w]){

                this.edgeTo[w] = v;

                this.marked[w] = true;

                queue.push(w);

            }

        }

    }

}

JavaScript数据结构和算法之图和图算法

最短路径

在执行广度优先搜索时,会自动查找从一个顶点到另一个相连顶点的最短路径

确定路径

要查找最短路径,需要修改广度优先搜索算法来记录从一个顶点到另一个顶点的路径,我们需要一个数组来保存从一个顶点操下一个顶点的所有边,我们将这个数组命名为edgeTo

this.edgeTo = [];//将这行添加到Graph类中
//bfs函数

function bfs(s){

    var queue = [];

    this.marked = true;

    queue.push(s);//添加到队尾

    while(queue.length>0){

        var v = queue.shift();//从队首移除

        if(v == undefined){

            print("Visited vertex: " + v);

        }

        for each(var w in this.adj[v]){

            if(!this.marked[w]){

                this.edgeTo[w] = v;

                this.marked[w] = true;

                queue.push(w);

            }

        }

    }

}

拓扑排序算法

拓扑排序会对有向图的所有顶点进行排序,使有向边从前面的顶点指向后面的顶点。
拓扑排序算法与BFS类似,不同的是,拓扑排序算法不会立即输出已访问的顶点,而是访问当前顶点邻接表中的所有相邻顶点,直到这个列表穷尽时,才会将当前顶点压入栈中。

拓扑排序算法被拆分为两个函数,第一个函数是topSort(),用来设置排序进程并调用一个辅助函数topSortHelper(),然后显示排序好的顶点列表

拓扑排序算法主要工作是在递归函数topSortHelper()中完成的,这个函数会将当前顶点标记为已访问,然后递归访问当前顶点邻接表中的每个顶点,标记这些顶点为已访问。最后,将当前顶点压入栈中。

//topSort()函数

function topSort(){

    var stack = [];

    var visited = [];

    for(var i =0;i<this.vertices;i++){

        visited[i] = false;

    }

    for(var i = 0;i<this.vertices;i++){

        if(visited[i] == false){

            this.topSortHelper(i,visited,stack);

        }

    }

    for(var i = 0;i<stack.length;i++){

        if(stack[i] !=undefined && stack[i] != false){

            print(this.vertexList[stack[i]]);

        }

    }

}
//topSortHelper()函数

function topSortHelper(v,visited,stack){

    visited[v] = true;

    for each(var w in this.adj[v]){

        if(!visited[w]){

            this.topSortHelper(visited[w],visited,stack);

        }

    }

    stack.push(v);

}
Javascript 相关文章推荐
将函数的实际参数转换成数组的方法
Jan 25 Javascript
学习从实践开始之jQuery插件开发 菜单插件开发
May 03 Javascript
详解AngularJS如何实现跨域请求
Aug 22 Javascript
javascript的几种写法总结
Sep 30 Javascript
jQuery Dialog 打开时自动聚焦的解决方法(两种方法)
Nov 24 Javascript
React Native之TextInput组件解析示例
Aug 22 Javascript
js 开发之autocomplete=&quot;off&quot;在chrom中失效的解决办法
Sep 28 Javascript
vue实现word,pdf文件的导出功能
Jul 31 Javascript
jquery多级树形下拉菜单的实例代码
Jul 09 jQuery
详解解决小程序中webview页面多层history返回问题
Aug 20 Javascript
基于form-data请求格式详解
Oct 29 Javascript
VUE项目实现主题切换的多种方法
Nov 26 Vue.js
Javascript核心读书有感之类型、值和变量
Feb 11 #Javascript
JavaScript中的继承方式详解
Feb 11 #Javascript
JavaScript中原型和原型链详解
Feb 11 #Javascript
Node.js中的缓冲与流模块详细介绍
Feb 11 #Javascript
javascript中var的重要性分析
Feb 11 #Javascript
JavaScript设计模式之工厂模式和构造器模式
Feb 11 #Javascript
js实现可兼容IE、FF、Chrome、Opera及Safari的音乐播放器
Feb 11 #Javascript
You might like
php 日期时间处理函数小结
2009/12/18 PHP
PHP实现AES256加密算法实例
2014/09/22 PHP
php使用Jpgraph创建折线图效果示例
2017/02/15 PHP
在第一个input框内输入内容.textarea自动得到第一个文件框的值的javascript代码
2007/04/20 Javascript
js身份证验证超强脚本
2008/10/26 Javascript
JS判断不同分辨率调用不同的CSS样式文件实现思路及测试代码
2013/01/23 Javascript
利用js实现选项卡的特别效果的实例
2013/03/03 Javascript
javascript修改表格背景色实例代码分享
2013/12/10 Javascript
bootstrap日期控件问题(双日期、清空等问题解决)
2017/04/19 Javascript
Vue.js教程之axios与网络传输的学习实践
2017/04/29 Javascript
bootstrap栅格系统示例代码分享
2017/05/22 Javascript
javascript将非数值转换为数值
2018/09/13 Javascript
layui table 复选框跳页后再回来保持原来选中的状态示例
2019/10/26 Javascript
JavaScript switch语句使用方法简介
2019/12/30 Javascript
Vue通过getAction的finally来最大程度避免影响主数据呈现问题
2020/04/24 Javascript
创建与框架无关的JavaScript插件
2020/12/01 Javascript
python实现随机漫步算法
2018/08/27 Python
python实现任意位置文件分割的实例
2018/12/14 Python
使用python实现对元素的长截图功能
2019/11/14 Python
Python内置函数locals和globals对比
2020/04/28 Python
Python如何获取文件指定行的内容
2020/05/27 Python
CSS3制作3D立方体loading特效
2020/11/09 HTML / CSS
Hobbs官方网站:英国奢华女性时尚服装
2020/02/22 全球购物
介绍下Java中==和equals的区别
2013/09/01 面试题
毕业生护理专业个人求职信范文
2014/01/04 职场文书
大学秋游活动方案
2014/02/11 职场文书
优秀少先队工作者事迹材料
2014/05/13 职场文书
化妆品活动策划方案
2014/05/23 职场文书
农村党支部书记四风问题个人对照检查材料
2014/09/21 职场文书
房屋买卖委托书格式范本格式
2014/10/13 职场文书
刑事和解协议书范本
2014/11/19 职场文书
2014员工聘用协议书(最新版)
2014/11/24 职场文书
2015年幼儿园教研活动总结
2015/03/25 职场文书
仓库统计员岗位职责
2015/04/14 职场文书
志愿者服务活动总结报告
2015/05/06 职场文书
python运算符之与用户交互
2022/04/13 Python