JavaScript数组和对象的复制


Posted in Javascript onMarch 21, 2017

一、数据类型

从狭义上来说,JS把所有的数据分成两大类型:基本类型和引用类型,其中基本类型包括Undefined、Null、Boolean、Number和String,引用类型为Object,常用的Array、Date、RegExp、Function等都属于Object类型。

基本型数据和引用型数据的区别之一就是,在复制变量时,基本型数据复制独立的一份新的拷贝,而引用型数据复制的是原变量的引用。下面是一个例子:

// 基本类型数据的复制
var a = 10;
var b = a; // b = 10
a = 20; // a = 20, b = 10
// 引用类型数据的复制
var m = [1, 2];
var n = m;
m[0] = 10;
console.log(n[0]); // 10

如果我想复制引用类型本身的值而非引用,显然不能采用上面的方式。

二、数组的浅复制

浅复制是指对象(数组)被复制时,其引用字段的值不会被复制,而是复制了对应字段的引用。如:

var src = [
 'alpha',
 ['bravo', 'chalie']
];
var dest = [];
for (var i = 0; i < src.length; i++) {
 dest[i] = src[i];
}
//此时,如果改变src中的引用字段,dest中相应的字段也会被改变
src[1].push('delta');
console.log(dest[1]); // ['bravo', 'chalie', 'delta']

浅复制一般用于一维数组,即数组中不存在引用类型的情况。常用的浅复制方法有:

concat方法

var src = ['alpha', 'bravo'],
  dest = [];
 dest = dest.concat(src);

concat方法更多地被用在数组合并中,比如:

var a = ['alpha', 'bravo'],
  b = ['chalie', 'delta'],
  combine;
 combine = a.concat(b);

特别要指出,concat用于数组合并时,是将两个(或多个)数组中的所有元素复制到新的对象,对于大型数组来说,开销比较大。更好的办法是把后一个数组的元素复制到前一个数组中:

var src = ['alpha', 'bravo'],
  dest = ['chalie', 'delta'];
 Array.prototype.push.apply(src, dest);

slice方法

slice方法可以从已有数组中返回选定的元素,返回的是一个新数组。

var src = ['alpha', 'bravo'],
 var dest = src.slice(0);

三、对象的浅复制

对象的浅复制可以用for-in遍历来实现,在es6中提供了更为方便的Object.assign()方法。

var src = {name: 'fox', age: 20},
  dest = null;
 dest = Object.assign({}, src);

也可以使用jQuery中的$.extend,underscore中的_.extend等方法来实现对象的复制。

四、深度复制

浅复制的应用场景有限,更多情况下,我们希望能够将对象复制出一个完整的副本,这就需要用到typeof或instanof操作符来对各个字段的类型进行判断。如果某字段是基本类型的,可以直接复制。如果某字段是引用类型的,还需要对该字段的所有字段进行上述判断,这就很容易让我们考虑使用递归来实现这个功能。

function deep_copy(src, dest) {
 for (var p in src) {
  if (Array.isArray(src[p]) || src[p] instanceof Object) {
   dest[p] = Array.isArray(src[p]) ? [] : {};
   arguments.callee(dest[p], src[p]);
  }else {
   dest[p] = src[p];
  }
 }
}

在上述代码中,由于数组是特殊的对象,因此可以用for-in来遍历。

另外,还可以使用json大法:

function deep_copy_in_json(src) {
  return JSON.parse(JSON.stringify(src));
 }

这样做虽然比较简便,但原对象的很多属性在操作后会丢失,比如construtor属性以及对象原型中的一些方法。

以上就是本文的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,同时也希望多多支持三水点靠木!

Javascript 相关文章推荐
中文输入法不触发onkeyup事件的解决办法
Jul 09 Javascript
js调试工具console.log()方法查看js代码的执行情况
Aug 08 Javascript
js限制文本框只能输入整数或者带小数点的数字
Apr 27 Javascript
JS实现的简洁二级导航菜单雏形效果
Oct 13 Javascript
jquery+json实现动态商品内容展示的方法
Jan 14 Javascript
拥有一个属于自己的javascript表单验证插件
Mar 24 Javascript
使用jQuery Rotare实现微信大转盘抽奖功能
Jun 20 Javascript
JavaScript 字符串常用操作小结(非常实用)
Nov 30 Javascript
javascript中this关键字详解
Dec 12 Javascript
vue中component组件的props使用详解
Sep 04 Javascript
详解react-router4 异步加载路由两种方法
Sep 12 Javascript
基于vue 实现token验证的实例代码
Dec 14 Javascript
Vue响应式添加、修改数组和对象的值
Mar 20 #Javascript
zTree实现节点修改的实时刷新功能
Mar 20 #Javascript
Vue指令的钩子函数使用方法
Mar 20 #Javascript
非常实用的vue导航钩子
Mar 20 #Javascript
Vue2.0实现1.0的搜索过滤器功能实例代码
Mar 20 #Javascript
如何解决vue与传统jquery插件冲突
Mar 20 #Javascript
Vue.js路由vue-router使用方法详解
Mar 20 #Javascript
You might like
PHP 简单数组排序实现代码
2009/08/05 PHP
input file获得文件根目录简单实现
2013/04/26 PHP
PHP精确计算功能示例
2016/11/29 PHP
jQuery LigerUI 插件介绍及使用之ligerDrag和ligerResizable示例代码打包
2011/04/06 Javascript
基于jquery实现点击左右按钮图片横向滚动
2013/04/11 Javascript
网站内容禁止复制和粘贴、另存为的js代码
2014/02/26 Javascript
jQuery循环动画与获取组件尺寸的方法
2015/02/02 Javascript
jQuery实现加入购物车飞入动画效果
2015/03/14 Javascript
jquery获取复选框checkbox的值实现方法
2016/05/30 Javascript
javascript鼠标滑过显示二级菜单特效
2020/11/18 Javascript
原生JS实现-星级评分系统的简单实例
2016/08/21 Javascript
javascript使用 concat 方法对数组进行合并的方法
2016/09/08 Javascript
jQuery选择器_动力节点Java学院整理
2017/07/05 jQuery
Node.JS使用Sequelize操作MySQL的示例代码
2017/10/09 Javascript
基于js原生和ajax的get和post方法以及jsonp的原生写法实例
2017/10/16 Javascript
微信小程序使用audio组件播放音乐功能示例【附源码下载】
2017/12/08 Javascript
NodeJS实现不可逆加密与密码密文保存的方法
2018/03/16 NodeJs
js字符串类型String常用操作实例总结
2019/07/05 Javascript
js利用iframe实现选项卡效果
2020/08/09 Javascript
Python查找相似单词的方法
2015/03/05 Python
Python性能提升之延迟初始化
2016/12/04 Python
python将ansible配置转为json格式实例代码
2017/05/15 Python
让Django支持Sql Server作后端数据库的方法
2018/05/29 Python
使用python生成杨辉三角形的示例代码
2018/08/29 Python
selenium+python自动化测试之环境搭建
2019/01/23 Python
Django实现跨域请求过程详解
2019/07/25 Python
Python将字典转换为XML的方法
2020/08/01 Python
Django返回HTML文件的实现方法
2020/09/17 Python
CSS3打造磨砂玻璃背景效果
2016/09/28 HTML / CSS
HTML5 video 视频标签使用介绍
2014/02/03 HTML / CSS
详解HTML5 Canvas标签及基本使用
2020/01/10 HTML / CSS
Linux开机引导的步骤是什么
2015/10/19 面试题
工程造价自荐信
2013/10/09 职场文书
学生宿舍管理制度
2014/01/30 职场文书
小学数学教学经验交流材料
2014/05/22 职场文书
幼儿园运动会口号
2014/06/07 职场文书