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 相关文章推荐
jQuery的链式调用浅析
Dec 03 Javascript
jQuery 拖动层(在可视区域范围内)
May 24 Javascript
浅谈javascript的原型继承
Jul 25 Javascript
JS 如何获取radio选中后的值及不选择取radio的值
Oct 28 Javascript
JavaScript访问字符串中单个字符的两种方法
Jul 03 Javascript
使用jquery.qrcode.min.js实现中文转化二维码
Mar 11 Javascript
Linux下为Node.js程序配置MySQL或Oracle数据库的方法
Mar 19 Javascript
BootStrap Select清除选中的状态恢复默认状态
Jun 20 Javascript
angular2模块和共享模块详解
Apr 08 Javascript
详解swiper在vue中的应用(以3.0为例)
Sep 20 Javascript
js中addEventListener()与removeEventListener()用法案例分析
Mar 02 Javascript
vue组件的路由高亮问题解决方法
May 11 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之Smarty入门
2007/01/04 PHP
php正则表达匹配中文问题分析小结
2012/03/25 PHP
PHP超级全局变量数组小结
2012/10/04 PHP
PHP编译安装时常见错误解决办法
2015/05/28 PHP
让你的PHP7更快之Hugepage用法分析
2016/05/31 PHP
Docker 如何布置PHP开发环境
2016/06/21 PHP
PHP中使用OpenSSL生成证书及加密解密
2017/02/05 PHP
JS获取各种浏览器窗口大小的方法
2014/01/14 Javascript
AngularJs根据访问的页面动态加载Controller的解决方案
2015/02/04 Javascript
解决js函数闭包内存泄露问题的办法
2016/01/25 Javascript
Node.js 中exports 和 module.exports 的区别
2017/03/14 Javascript
前端常见跨域解决方案(全)
2017/09/19 Javascript
Angular4.0中引入laydate.js日期插件的方法教程
2017/12/25 Javascript
jQuery实现鼠标响应式透明度渐变动画效果示例
2018/02/13 jQuery
jQuery实现新闻播报滚动及淡入淡出效果示例
2018/03/23 jQuery
vue获取时间戳转换为日期格式代码实例
2019/04/17 Javascript
Python实现批量读取word中表格信息的方法
2015/07/30 Python
Python3使用requests发闪存的方法
2016/05/11 Python
Python while 循环使用的简单实例
2016/06/08 Python
Matplotlib 生成不同大小的subplots实例
2018/05/25 Python
Python第三方Window模块文件的几种安装方法
2018/11/22 Python
PyCharm+PySpark远程调试的环境配置的方法
2018/11/29 Python
在python 不同时区之间的差值与转换方法
2019/01/14 Python
Python实现删除排序数组中重复项的两种方法示例
2019/01/31 Python
python卸载后再次安装遇到的问题解决
2019/07/10 Python
Python命名空间及作用域原理实例解析
2020/08/12 Python
Python配置pip国内镜像源的实现
2020/08/20 Python
python类共享变量操作
2020/09/03 Python
HTML如何让IMG自动适应DIV容器大小的实现方法
2020/02/25 HTML / CSS
西班牙床垫网上商店:Colchones.es
2018/05/06 全球购物
新手上路标语
2014/06/20 职场文书
2014年医院后勤工作总结
2014/12/06 职场文书
家长会主持词开场白
2015/05/29 职场文书
疾病证明书
2015/06/19 职场文书
2019年汽车租赁合同范本!
2019/08/12 职场文书
python编程实现清理微信重复缓存文件
2021/11/01 Python