深度优先遍历 and 广度优先遍历

遍历在前端的应用场景不多,多数是处理DOM节点数或者 深拷贝。下面笔者以深拷贝为例,简单说明一些这两种遍历。

深度优先遍历

想象有一颗节点树,从某个顶点开始,一直往下遍历,直到遍历到的节点都访问过后,往回走,遍历没有访问的节点,感觉很像递归。下面笔者就用递归实现 深度优先遍历。


function getRegExp(target){
var flat = '';
if(target.global) flat+='g';
if(target.ignoreCase) flat+='i';
if(target.multiline) flat+='m';
return target;
} function DFS(target,visiteds){
var Type = Object.prototype.toString.call(target).slice(8,-1);
console.log(Type)
var copy = Type == 'Array'?[]:{};
visiteds = visiteds|| []; //处理环形数据,防止无限循环
switch(Type){
case 'Date':
copy = new Date(target.getTime());
break;
case 'RegExp':
copy = new RegExp(target.source,getRegExp(target));
break;
case 'Array':
case 'Object':
var index = visiteds.indexOf(target);
if(index>-1){
copy = visiteds[index];
}else{
visiteds.push(target);
for(var key in target){
copy[key] = DFS(target[key],visiteds);
}
}
break;
default:
copy = target;
break; }
return copy;
};

来个es6版本的:

const deepClone = (obj, hash=new WeakMap()) =>{
if(obj instanceof RegExp) return new RegExp(obj);
if(obj instanceof Date) return new Date(obj);
if(typeof obj === 'function') return new Function('return '+obj.toString())();
if(obj === null || typeof obj !== 'object') return obj;
if(hash.has(obj)){
return hash.get(obj);
} let t = obj.constructor();
hash.set(obj,t);
for(let k in obj){
if(obj.hasOwnProperty(k)){
t[k] = deepClone(obj[k],hash);
}
}
return t;
}

广度优先遍历

如果把深度优先遍历看成纵向遍历,那么广度优先遍历就是横向遍历,一层一层的往下遍历。下面用队列(FIFO)来实现。


function getEmpty(o){
var Type = Object.prototype.toString.call(o).slice(8,-1);
if(Type === 'Object'){
return {};
}
if(Type === 'Array'){
return [];
}
if(Type==='Date'){
return new Date(o.getTime());
}
if(Type==='RegExp'){
return new RegExp(o.source,getRegExp(o))
} return o;
} function getRegExp(o){
var flat = '';
if(o.global) flat+='g';
if(o.ignoreCase) flat+='i';
if(o.multiline) flat+='m';
return o;
} function BFS(target){
var queue = [];
var targetMap = []; //处理环形数据,防止无限循环
var copy = getEmpty(target);
if(copy!==target){
queue.push([target,copy]);
}
while(queue.length>0){
var [_target,_copy] = queue.shift(); //*
for(var key in _target){
var index = targetMap.indexOf(_target[key]);
if(index>-1){
_copy[key] = targetMap[index]
continue;
}
_copy[key] = getEmpty(_target[key])
if(_copy[key]!==_target[key]){
queue.push([_target[key],_copy[key]]); //*
targetMap.push(_target[key]);
}
}
}
return copy
}

总结

上面用两种不同的方法实现了深拷贝,但是只针对Object,Array的情况,其他的复杂对象没有考虑到,当然你也可以添加更多的处理,但笔者认为目前这样已经足够用了。。。
深度优先遍历,关键在于理解递归,而广度优先遍历,关键在于理解,queue.shift出去的数据保存着原来数据的引用,所以才能够在不断的进栈出栈中修改值(间接修改值)

深度优先遍历 and 广度优先遍历的更多相关文章

  1. js实现深度优先遍历和广度优先遍历

    深度优先遍历和广度优先遍历 什么是深度优先和广度优先 其实简单来说 深度优先就是自上而下的遍历搜索 广度优先则是逐层遍历, 如下图所示 1.深度优先 2.广度优先 两者的区别 对于算法来说 无非就是时 ...

  2. C++ 二叉树深度优先遍历和广度优先遍历

    二叉树的创建代码==>C++ 创建和遍历二叉树 深度优先遍历:是沿着树的深度遍历树的节点,尽可能深的搜索树的分支. //深度优先遍历二叉树void depthFirstSearch(Tree r ...

  3. 邻接矩阵c源码(构造邻接矩阵,深度优先遍历,广度优先遍历,最小生成树prim,kruskal算法)

    matrix.c #include <stdio.h> #include <stdlib.h> #include <stdbool.h> #include < ...

  4. C++编程练习(9)----“图的存储结构以及图的遍历“(邻接矩阵、深度优先遍历、广度优先遍历)

    图的存储结构 1)邻接矩阵 用两个数组来表示图,一个一维数组存储图中顶点信息,一个二维数组(邻接矩阵)存储图中边或弧的信息. 2)邻接表 3)十字链表 4)邻接多重表 5)边集数组 本文只用代码实现用 ...

  5. js实现对树深度优先遍历与广度优先遍历

    深度优先与广度优先的定义 首先我们先要知道什么是深度优先什么是广度优先. 深度优先遍历是指从某个顶点出发,首先访问这个顶点,然后找出刚访问这个结点的第一个未被访问的邻结点,然后再以此邻结点为顶点,继续 ...

  6. python、java实现二叉树,细说二叉树添加节点、深度优先(先序、中序、后续)遍历 、广度优先 遍历算法

    数据结构可以说是编程的内功心法,掌握好数据结构真的非常重要.目前基本上流行的数据结构都是c和c++版本的,我最近在学习python,尝试着用python实现了二叉树的基本操作.写下一篇博文,总结一下, ...

  7. 【图的遍历】广度优先遍历(DFS)、深度优先遍历(BFS)及其应用

    无向图满足约束条件的路径 •[目的]:掌握深度优先遍历算法在求解图路径搜索问题的应用 [内容]:编写一个程序,设计相关算法,从无向图G中找出满足如下条件的所有路径:  (1)给定起点u和终点v.  ( ...

  8. 二叉树的深度优先遍历与广度优先遍历 [ C++ 实现 ]

    深度优先搜索算法(Depth First Search),是搜索算法的一种.是沿着树的深度遍历树的节点,尽可能深的搜索树的分支. 当节点v的所有边都己被探寻过,搜索将回溯到发现节点v的那条边的起始节点 ...

  9. Dom的深度优先遍历和广度优先遍历

    //深度优先遍历的递归写法 function DFTraversal(node) { var nodes = []; if (node != null) { nodes.push(node); var ...

随机推荐

  1. JavaScript--clientX,clientY、pageX,pageY、offsetLeft,offsetTop/offsetWidth,offsetHeight、scrollLeft,scrollTop/scrollWidth,scrollHeight、clientHeight,clientWidth区别

    /*在事件的内部console.dir(event)*/ /** * 事件对象event * clientX/clientY 获取鼠标基于浏览器窗口(可视区域的坐标位置)全兼容 * * pageX/p ...

  2. $.extend用法详解(一)

    jQuery.extend( target [, object1 ] [, objectN ] ) 在这里target是Object,它有两个作用: 1. 如果后面没有对应的object1及objec ...

  3. Codeforces 13C

    这题纠结了近半年,一直没有好的思路. 刚开始看这题的时候就是暴力,明显的TLE 后来才知道这题的“一种解”肯定是"原数列中某些数的集合" (很明显这题的最优策略并不唯一) 有原数列 ...

  4. python 并发之进程

    一.什么是进程 进程(Process)是计算机中的程序关于某数据集合上的一次运行活动,是系统进行资源分配和调度的基本单位,是操作系统结构的基础.在早期面向进程设计的计算机结构中,进程是程序的基本执行实 ...

  5. 快速启动Oracle服务

    快速启动Oracle服务的批处理命令步骤 新建记事本 粘贴如下内容: @echo off echo 确定要启动Oracle 11g服务吗? pause net start OracleOraDb11g ...

  6. [已转移]IE事件流和DOM标准事件流的区别

    该文章已转移到博客:https://cynthia0329.github.io/ 1.执行的顺序不一样 冒泡型事件模型: button->div->body (IE事件流) 捕获型事件模型 ...

  7. phpexcel使用说明2

      转自:http://serisboy.iteye.com/blog/1928139 首先到phpexcel官网上下载最新的phpexcel类,下周解压缩一个classes文件夹,里面包含了PHPE ...

  8. poj 1655 Balancing Act 求树的重心【树形dp】

    poj 1655 Balancing Act 题意:求树的重心且编号数最小 一棵树的重心是指一个结点u,去掉它后剩下的子树结点数最少. (图片来源: PatrickZhou 感谢博主) 看上面的图就好 ...

  9. iOS 设计

    APP引导页设计经验分享 http://www.cocoachina.com/design/20150615/12126.html 获取app安装的进度,6种不同的加载指示 http://www.co ...

  10. 8.5打包libgdx为一个桌面程序(jar包)

    简陋的地图编辑终于做好了,于是要开始制作地图了,想导出为一个windows下可用的程序,让熟人代做地图,然后找人问了下打包流程,其实跟普通java打包为jar没什么区别,记录如下: 导出类型选第三个 ...