在javascript中,数据主要分基本类型和引用类型两种。

基本类型的赋值比较简单,但是引用类型的赋值,会存在一些问题,那我们用代码来分析一下。

一、浅拷贝

var one = "测试1";
var two = one;
two = "测试2";
console.log(one);

上面代码中,声明了一个变量one ,值为:“测试1” ,然后将变量one 赋值给了变量two ,

之后我们把变量two的值修改成:“测试2” 了,这时我们从控制台打印变量one ,输出的结果是:“测试1”。

也就是说修改变量two的值并不会对变量one造成影响。

那我们再来看下面这个例子:

var arr = ['白色','蓝色','紫色'];
var temp = arr;
temp[1]="天蓝色";
console.log(arr);

上面这段代码,控制台打印输出了  ‘白色’,'天蓝色','紫色'  ,这是怎么回事呢?我们明明修改的是temp,怎么arr也被修改了?

这其实很简单,因为在javascript中,数组属于引用类型数据。在上面的例子中,当把arr赋值给temp时,其实只是把访问地址赋值给了temp

我们来分析一下代码

var arr = ['白色','蓝色','紫色'];

在javascript中,数组的大小是不确定的,这时js会在堆内存中建立一块内存,用来存一个数组,而这个数组的值目前为:['白色','蓝色','紫色'] ,然后会生成一个地址,并且赋值给了arr ,

所以说arr存的是 ['白色','蓝色','紫色'] 在堆内存中的访问地址。

var temp = arr;

因为arr存的是一个地址,这时将arr赋值给了temp ,所以temp也拥有访问['白色','蓝色','紫色']的地址了。

temp[1]="天蓝色";

那么这时我们将temp[1]的值修改成了“天蓝色”,这一操作,首先temp通过地址访问堆内存对应的数据,并且修改了数据。

因为arr和temp存的只是访问同一个源的地址,源被temp改变了,arr再次访问时,访问的也就是改变过的源了。

但有时候,我们修改一个数据的时候,并不想影响另一个数据,使用深拷贝就可以解决这个问题。

二、深拷贝

数组中的深拷贝

数组实现深拷贝的方法:使用slice()或cancat()方法,或者是使用循环将当前数组的值,一个一个添加到新的数组中即可。

var arr = [1,3,4];
var temp = arr.slice(0);
temp[0]=6;
console.log(arr);//输出 1,3,4 不会被temp影响
var arr = [1,3,4];
var temp =arr.concat([]);
temp[0]=6;
console.log(arr);//输出1,3,4 不会被temp影响

其实上面两种都属于简单的深拷贝,如果是二维数组、三维数组呢?我们看一个例子

var ac = [1,[2,3],4];
var ad = ac.slice(0);
ad[1][0]=6;
ad[0]=5;
console.log(ac);

控制台输出结果如下:

使用了简单的深拷贝(slice方法)之后,可以发现ac的第一个值确实不受ad的影响,但是ac的第二个值却随着ad的改变而改变了。更深层的拷贝怎么办呢?其实可以利用递归函数来解决这个问题

        var duowei = [1,[2,3],4];
function dispose(data){
var out = [];
for(var i =0;i<data.length;i++){
if(data[i] instanceof Array){
out[i]=dispose(data[i]);
}else{
out[i]=data[i];
}
}
return out;
} var result = dispose(duowei);
result[1][0]=9;
console.log(duowei);

对象中的深拷贝

在对象中简单的深拷贝可以使用Object.assign来实现

let a = {
age: 1
}
let b = Object.assign({}, a)
a.age = 2
console.log(b.age) //

复杂的深拷贝可以使用JSON.parse(JSON.stringify(object))来解决,不过这个方法有一些局限性。

  • 会忽略 undefined
  • 会忽略 symbol
  • 不能序列化函数
  • 不能解决循环引用的对象
let a = {
age: 1,
jobs: {
first: 'FE'
}
}
let b = JSON.parse(JSON.stringify(a))
a.jobs.first = 'native'
console.log(b.jobs.first) // FE

其实方法有很多,需要自己去挖掘。

这里在分享一个数组和对象实现简单深拷贝通用的方法:通过展开运算符(…)来实现。

let a = {
age: 1
}
let b = {...a}
a.age = 2
console.log(b.age) //

《JavaScript总结》深拷贝和浅拷贝的更多相关文章

  1. javascript对象深拷贝,浅拷贝 ,支持数组

    javascript对象深拷贝,浅拷贝 ,支持数组 经常看到讨论c#深拷贝,浅拷贝的博客,最近js写的比较多, 所以也来玩玩js的对象拷贝. 下面是维基百科对深浅拷贝的解释: 浅拷贝 One meth ...

  2. 也来玩玩 javascript对象深拷贝,浅拷贝

    经常看到讨论c#深拷贝,浅拷贝的博客,最近js写的比较多, 所以也来玩玩js的对象拷贝. 下面是维基百科对深浅拷贝的解释: 浅拷贝 One method of copying an object is ...

  3. JavaScript的深拷贝和浅拷贝总结

    深拷贝和浅拷贝 深拷贝:拷贝实例:浅拷贝:拷贝引用(原对象). 说深拷贝和浅拷贝之前,我先去了解了下高程书上的JavaScript的变量类型: 基本类型:undefined.null.Boolean. ...

  4. JavaScript的深拷贝和浅拷贝

    一.数据类型 数据分为基本数据类型(String, Number, Boolean, Null, Undefined,Symbol)和对象数据类型.. 1.基本数据类型的特点:直接存储在栈(stack ...

  5. JavaScript的深拷贝与浅拷贝

    深拷贝和浅拷贝是在面试中经常遇到的问题.今天在这里总结一下. 深拷贝与浅拷贝的问题,涉及到JavaScript的变量类型,先来说说变量的类型,变量类型包括基本类型和引用类型. 基本类型:Undefin ...

  6. 详解javascript的深拷贝与浅拷贝

    1. 认识深拷贝和浅拷贝 javascript中一般有按值传递和按引用传递两种复制,按值传递的是基本数据类型(Number,String,Boolean,Null,Undefined),一般存放于内存 ...

  7. JavaScript 的 深拷贝和浅拷贝

    深拷贝和浅拷贝都是针对的引用类型, JS中的变量类型分为值类型(基本类型)和引用类型: 对值类型进行复制操作会对值进行一份拷贝,而对引用类型赋值,则会对地址进行拷贝,最终两个变量指向同一份数据 一.先 ...

  8. JavaScript之深拷贝和浅拷贝

    前言 工作中会经常遇到操作数组.对象的情况,你肯定会将原数组.对象进行‘备份’当真正对其操作时发现备份的也发生改变,此时你一脸懵逼,到时是为啥,不是已经备份了么,怎么备份的数组.对象也会发生变化.如果 ...

  9. javaScript深拷贝和浅拷贝简单梳理

    在了解深拷贝和浅拷贝之前,我们先梳理一下: JavaScript中,分为基本数据类型(原始值)和复杂类型(对象),同时它们各自的数据类型细分下又有好几种数据类型 基本数据类型 数字Number 字符串 ...

  10. 读懂javascript深拷贝与浅拷贝

    1. 认识深拷贝和浅拷贝 javascript中一般有按值传递和按引用传递两种复制,按值传递的是基本数据类型(Number,String,Boolean,Null,Undefined),一般存放于内存 ...

随机推荐

  1. freeMark模板引擎

    http://blog.csdn.net/shimiso/article/details/8778793

  2. python Flask框架mysql数据库配置

    我是一个没有笔记习惯的低级程序员,但是我还是喜欢编程,从小学就开始跟着玩电脑,对抓鸡,ddos,跳板刷钻开始了自己的IT 旅程,之后学习了各种语言,但是可惜都不没有达到精通,都是略懂一二,现在想把Py ...

  3. python编程之变量和简单的数据结构

    一.变量 前面我们用python输出了“hello world!” 这次我们在前面加入一行,定义一个变量,然后修改第二行. 添加变量导致Python解释器需要做更多工作.处理第1行代码时,它将文本“H ...

  4. go基本使用方法

    一,变量 var:声明变: var 变量名  数据类型 :同时还需要指定数据的类型 var 变量名 = 值  : 声明变量,根据变量值判断变量类型 :=   :省略var,直接可以(变量名:= 值), ...

  5. hbase单机版安装+phoneix SQL on hbase 单节点安装

    hbase 单机安装部署及phoneix 单机安装 Hbase 下载 (需先配置jdk) https://www.apache.org/dyn/closer.lua/hbase/2.0.1/hbase ...

  6. ps最最基础的文档

    因为学习PHP,但是公司没有前端工程师,修图的时候只好找被人帮忙,一个简答的问题,其实几分钟就搞定了,还要麻烦别人,就自己学了一下ps.一共花了3天时间.学习了一些简单的操作. 工具:Adobe Ph ...

  7. NSInvalidArgumentException这个到底是什么意思,我到网上查了下,每个的错误都不同。

    我几乎把所有的东西都注释了,但还是崩了 #import "ViewController.h" //#import "WeiBo.h" @interface Vi ...

  8. nohup和&后台运行,进程查看及终止

    1.nohup 用途:不挂断地运行命令. 语法:nohup Command [ Arg … ] [ & ] 无论是否将 nohup 命令的输出重定向到终端,输出都将附加到当前目录的 nohup ...

  9. 如何用kaldi做孤立词识别二

    基本模型没有变化,主要是调参,配置: %WER     65%  下降到了     15% 后面再继续优化... Graph compilation finish!steps/decode.sh -- ...

  10. C语言窗口例子

    #include <windows.h> LRESULT CALLBACK WndProc( HWND, UINT, WPARAM, LPARAM ) ; //声明用来处理消息的函数 in ...