js中的数据类型

在介绍javascript中的对象的拷贝之前,我先介绍一个基础的东西,javascript中的数据类型。

我们做前端的应该都知到在es6 之前,javascript中的数据类型BooleanNumberStringUndefinedObjectNull,后来在es6 中又引入了一种新的数据类型为:Symbol。而这些数据类型又被分为基本数据类型和引用数据类型,基本数据类型存储在栈中;引用数据类型存储在堆中。其中基本数据类型包括有:BooleanNumberStringUndefinedNull以及刚刚提到的新的数据类型Symbol,引用类型包括:ObjectFunctionArray。我为甚么会提到FunctionArray主要是因为我么在对象的深拷贝过程中需要对这两种数据类型进行特殊处理。

什么是对象的拷贝

介绍完了js中的数据类型之后,来看一下拷贝,什么是对象的拷贝,说白了就是复制,将原来的东西给复制一份。就比如说,将磁盘上的一个文件给拷贝一份,就是将磁盘中的文件给复制了一个一模一样的新的文件。就比如说下面的例子;

var a = 123;
var b = a;
var c = {name: 'zhangsan', age: 18};
var d = c; var e = {};
for (var key in c) {
if (c.hasOwnProperty(key)) {
e[key] = c[key];
}
}

对象的拷贝又被分为深拷贝和浅拷贝。

浅拷贝

就上面的代码来解释,我们看最后一个拷贝,就是我们将变量c中所有的属性然后赋值给变量e,这种情况不出问题的前提就是我们定义的对象a中的所有的成员都为基本类型,而非引用类型,一旦存在有引用类型的成员,这个时候的拷贝是将成员变量的地址赋给拷贝过去了,而,成员变量地址指向的真实的引用依旧是同一引用,因此,当指向中的引用内容发生变化时,同样会两个对象中的成员也会发生同样的改变;

比如说下面的这个例子:

var a = {
name: 'zhangsan',
age: 28,
children: [1,2,3,4,5],
son: {
name: 'zhangsi',
age: 1
}
} var shallowCopy = function (obj) {
var newObj = {};
for (var key in obj) {
if (obj.hasOwnProperty(key)) {
newObj[key] = obj[key];
}
}
return newObj;
} var b = shallowCopy(a);
console.log(a.children[0]);
console.log(a.son.name);
console.log(b.children[0]);
console.log(b.son.name);
a.children[0] = 22;
a.son.name = 'name'; console.log(b.children[0]);
console.log(b.son.name);

在上面的这个例子中,我在后面改变了achildren以及a.som.name,我们会发现这个情况下面,b相对应的内容也发生了变化。并未发生实际上的拷贝,这就是浅拷贝。

深拷贝

在了解了浅拷贝的基础上,我们再来深入的了解一下深拷贝,有些时候我们是需要进行深拷贝的,这个时候复制的就不仅仅是一个引用地址,而是一个真实的引用内容。基于这个理论,就可以在复制的过程中加入一个判断,判断所要复制的量是引用类型还是基本类型,如果是基本类型的话,就直接赋值过去,如果是引用类型的话,就需要继续对引用类型进行拷贝。

因此我们将对上面浅拷贝的代码进修改如下:

var a = {
name: 'zhangsan',
age: 28,
children: [1,2,3,4,5],
son: {
name: 'zhangsi',
age: 1
}
} var deepCopy = function(obj) {
if (typeof obj !== 'object') return;
var newObj = obj instanceof Array ? [] : {};
for (var key in obj) {
if (obj.hasOwnProperty(key)) {
newObj[key] = typeof obj[key] === 'object' ? deepCopy(obj[key]) : obj[key];
}
}
return newObj;
} var b = deepCopy(a);
console.log(a.children[0]);
console.log(a.son.name);
console.log(b.children[0]);
console.log(b.son.name);
a.children[0] = 22;
a.son.name = 'name'; console.log(b.children[0]);
console.log(b.son.name);

我们能够看到,最后的输出和浅拷贝的输出是不一样的,输出的依旧是之前的之前的值,而不是发生变化的值。当然上面的这个深拷贝的例子还仅仅制止一个基础的,还不够完善,仅仅能够完成基本的对象拷贝。具体的实现我们可以参考的还有很多。

实现深拷贝的常用方法

  • jQuery 中的实现

    在jQuery中,深浅拷贝是使用的同一个方法就是extend,这个函数的第一个参数是表示这次的拷贝是深拷贝还是浅拷贝,如果要进行深拷贝的话,则传入true,否则不传或者是传false。具体的实现内容如下:

    jQuery.extend = jQuery.fn.extend = function() {
    var options, name, src, copy, copyIsArray, clone,
    target = arguments[ 0 ] || {},
    i = 1,
    length = arguments.length,
    deep = false; // Handle a deep copy situation
    if ( typeof target === "boolean" ) {
    deep = target; // Skip the boolean and the target
    target = arguments[ i ] || {};
    i++;
    } // Handle case when target is a string or something (possible in deep copy)
    if ( typeof target !== "object" && typeof target !== "function" ) {
    target = {};
    } // Extend jQuery itself if only one argument is passed
    if ( i === length ) {
    target = this;
    i--;
    } for ( ; i < length; i++ ) { // Only deal with non-null/undefined values
    if ( ( options = arguments[ i ] ) != null ) { // Extend the base object
    for ( name in options ) {
    copy = options[ name ]; // Prevent Object.prototype pollution
    // Prevent never-ending loop
    if ( name === "__proto__" || target === copy ) {
    continue;
    } // Recurse if we're merging plain objects or arrays
    if ( deep && copy && ( jQuery.isPlainObject( copy ) ||
    ( copyIsArray = Array.isArray( copy ) ) ) ) {
    src = target[ name ]; // Ensure proper type for the source value
    if ( copyIsArray && !Array.isArray( src ) ) {
    clone = [];
    } else if ( !copyIsArray && !jQuery.isPlainObject( src ) ) {
    clone = {};
    } else {
    clone = src;
    }
    copyIsArray = false; // Never move original objects, clone them
    target[ name ] = jQuery.extend( deep, clone, copy ); // Don't bring in undefined values
    } else if ( copy !== undefined ) {
    target[ name ] = copy;
    }
    }
    }
    } // Return the modified object
    return target;
    };

    具体的解析变就不做过于深入的介绍,这里的思路其实就是我们上面所介绍的思路的一个优化,将各种情况都帮我们给考虑清楚了。

javascript中的对象拷贝的更多相关文章

  1. javascript中的对象字面量为啥这么酷

    原文链接 : Why object literals in JavaScript are cool 原文作者 : Dmitri Pavlutin 译者 : neal1991 个人主页:http://n ...

  2. javascript中的对象,原型,原型链和面向对象

    一.javascript中的属性.方法 1.首先,关于javascript中的函数/“方法”,说明两点: 1)如果访问的对象属性是一个函数,有些开发者容易认为该函数属于这个对象,因此把“属性访问”叫做 ...

  3. Javascript 中判断对象为空

    发现了一个巧妙的实现: 需要检查一个对象(Object)是否为空,即不包含任何元素.Javascript 中的对象就是一个字典,其中包含了一系列的键值对(Key Value Pair).检查一个对象是 ...

  4. JavaScript 中的对象

    JavaScript 中的对象 在 JavaScript 中,对象是数据(变量),拥有属性和方法. JavaScript 中的所有事物都是对象:字符串.数字.数组.日期,等等.   访问对象的属性 访 ...

  5. javascript中Date对象的应用——简易日历的实现

    × 目录 [1]效果 [2]HTML [3]CSS[4]JS 前面的话 简易日历作为javascript中Date对象的常见应用,用途较广泛.本文将详细说明简易日历的实现思路 效果演示 HTML说明 ...

  6. JavaScript中判断对象类型方法大全1

    我们知道,JavaScript中检测对象类型的运算符有:typeof.instanceof,还有对象的constructor属性: 1) typeof 运算符 typeof 是一元运算符,返回结果是一 ...

  7. (转)javascript中的对象查找

    本文转自:http://otakustay.com/object-lookup-in-javascript/  ---很棒的一篇文章,作者的其他文章还暂时没读,但相信作者是一个谦虚 谨慎的好工程师 近 ...

  8. Javascript中的对象和原型(3)

    在Javascript中的对象和原型(二)中我们提到,用构造函数创建的对象里面,每个对象之间都是独立的,这样就会降低系统资源的利用率,解决这样问题,我们就要用到下面提到的原型对象. 一 原型对象 原型 ...

  9. (转)javascript中event对象详解

    原文:http://jiajiale.iteye.com/blog/195906 javascript中event对象详解          博客分类: javaScript JavaScriptCS ...

随机推荐

  1. java 小程序分析:参数传递

    public class Test{ public static void main (String [] ABC){ StringBuffer a = new StringBuffer(" ...

  2. 在可插拔settings的基础上加入类似中间件的设计

    在可插拔settings的基础上加入类似中间件的设计 settings可插拔设计可以看之前的文章 https://www.cnblogs.com/zx125/p/11735505.html 设计思路 ...

  3. DNS服务反向解析实验

    DNS域名解析服务是用于解析域名与ip地址对应关系的服务,功能上可以实现正向解析和反向解析 正向解析:根据主机名(域名)查找对应的IP地址. 反向解析:根据IP地址查找对应的主机名(域名). 下面我来 ...

  4. 图库网站Unsplash高清原图爬虫【华为云技术分享】

    [摘要] 写博客的好工具,快速获得高清图片 在百度图片爬虫小助手里,我开发了一个爬虫,来节约我写博客时搜集图片的时间. 但是,也出现了一些问题,主要有以下几点: 百度图片上的质量参差不齐,大部分图片质 ...

  5. Docker数据挂载

    Docker数据管理 在容器中管理数据主要有两种方式: 数据卷(Volumes) 挂载主机目录(Bind mounts) 数据卷 数据卷是一个可供一个或多个容器使用的特殊目录,它绕过UFS,可以提供很 ...

  6. Cypher基本指令学习1

    1.查询节点 查询所有节点match (n) return n 查询带有标签的节点 match(movie:Flyer) return movie.name 查询关联节点(查询A导演的所有电影) ma ...

  7. Android 通知的基本用法

    一.概念 通知(Notification)是 Android 系统中比较有特色的一个功能,当某个应用程序希望像用户发送一些提示消息的时候,然而此时应用程序并不在前台运行,此时就可以借助通知来实现.发出 ...

  8. Xcode 10 Error: Multiple commands produce

    目录 Xcode 9.4.1运行react-native 可以,但是在Xcode 10运行报错,报错信息如下: 解决方法 1. 选择 File > Project Settings (或者 Fi ...

  9. 牛客剑指offer(持续更新~)

    第一题:二维数组的查找 题目描述 在一个二维数组中(每个一维数组的长度相同),每一行都按照从左到右递增的顺序排序,每一列都按照从上到下递增的顺序排序.请完成一个函数,输入这样的一个二维数组和一个整数, ...

  10. 记录一些实用的小技巧-JS篇

    1.16进制随机颜色 let color = '#'+Math.random().toString(16).slice(-6) 2.类型判断工具函数 function isType(target, t ...