js-由深拷贝浅拷贝,传值与传址-引发的关于堆(“heap”)栈(“stack”)的思考!

前端面试一定会都遇到输出值的问题,并且不止一道题而是几乎两页纸。这类问题大多都是看一个前端开发者的js基础,数据类型和变量作用域。今天我们就堆栈的概念来看看js中的变量是如何存储的。

Js的数据类型可分为 值类型(基本类型):字符串(String)、数字(Number)、布尔(Boolean)、对空(Null)、未定义(Undefined)。和 引用数据类型:对象(Object)、数组(Array)、函数(Function)。

由于栈和堆的缓存设计各异所以存储的数据也有所不同。

栈使用的是一级缓存,内置在CPU内部并与CPU同速运行,效率高,但存储空间小。适合存储简单的数据段,占据固定空间大小的基本数据类型(String,Number,Boolean,Null,Undefined)。

堆存放在二级缓存中,是位于CPU与内存之间的临时存储器,比一级缓存容量更大,但处理速度更慢。适合存储大小不定,构造复杂的引用类型值(Function,Array,Object)。

传值&传址

传值就是变量之间值的传递与引用,传址是相对于引用类型而言。如例:


会输出什么呢?

按照js自上而下的执行顺序我们期望得到<5,1,[2],[0],false>而事实却是这样的<5,1,[2],[2],true>。

例子1:a为Number基本类型,存在栈内存中。将a赋值给b也就是将a的值赋值给b,所以b=1;改变a的值对b不会有影响。

例子2:arr为Array引用类型,存储在堆内存中,栈内存储的是arr(Array)在堆内存空间的引用指针。将arr赋值给arr1也就是将arr在堆内存空间的引用指针赋值给arr1,所以他们都指向了同一个存储空间的相同数据,固 arr === arr1。


那么同样是存储为什么会有存值和存址之分?这就要从栈和堆的存储方式说起。

前面提到栈使用的是一级缓存,一级缓存被内置在CPU内部并与CPU同速运行,容量小,速度快,更适合占据固定空间的简单数据段,故而存值。而堆使用的是二级缓存,二级缓存比一级缓存速度慢,容量大,是一级缓存和内存之间临时交换数据的地方。适合存储大小不固定,构造复杂的引用类型值。所以当我们在程序中调用引用类型时,是通过栈内存中的引用指针在堆内存中查找到相应的存储数据。

现在回看arr的例子,先var申明arr,再申明arr1并赋值为arr,再改变arr的值,然后分别调用输出了arr和arr1。从执行顺序考虑只是改变了arr的值,arr1是不受影响的,但是在arr1 = arr的时候我们只是传递了引用类型再堆内存中的引用指针。所以都变了...

假如两个例子同时出现在面试题中,是不是一点疑惑都没有了呢。

深拷贝&浅拷贝

我们经常会遇到将对象赋值给另一对象的场景,有时候也会费力去写一些clone的方法,为什么呢?

如上面例子2中我们看到,对象的传递是传址是通过指针的引用,例子中的传递也就是表面上的传址这是浅拷贝。而深拷贝就是申请新的存储空间,将源对象的值循环遍历赋值给目标对象,存入到新的存储空间。

function clone(origin){

    var target = null;

    if(origin instanceof Array){

        target = origin.concat();

    }else{

        target = {};

        for(var item in origin){

            var val = origin[item];

            target[item] = typeof val == 'object'?clone(val):val;

        }

    }

    return target;

}

而ES6也给我们提供了更好用的方法 Object.assign(target, ...source)用于将所有可枚举属性的值从一个或多个源对象复制到目标对象。它返回目标对象。这里需要注意假如源对象的属性值是一个对象的引用,那么它也只指向那个引用。所以要慎用。


当然还有一种方法也不得不提就是jQuery的$.extend()与Object.assign有着异曲同工之妙,但$.extend多了首个参数deep从而实现了深拷贝如列:


 第一次试着写这种长篇大论,可能漏洞百出,还请多多指正。文章没有深层意义,仅供参考,以免将您带入歧途。一些堆栈,一级缓存,二级缓存的知识也都是参照百度百科,如果感兴趣的可以自己去研究研究。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,179评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,229评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,032评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,533评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,531评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,539评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,916评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,574评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,813评论 1 296
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,568评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,654评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,354评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,937评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,918评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,152评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,852评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,378评论 2 342

推荐阅读更多精彩内容