前端面试题:深入理解基本类型、引用、赋值、浅拷贝、深拷贝
liebian365 2024-11-21 17:39 24 浏览 0 评论
谈到深拷贝浅拷贝,是面试中常常问到的一个面试题,特地整理方便小伙伴对此类知识点能够深入的理解。
如何区分深拷贝与浅拷贝,简单点来说,就是假设obj1复制了obj2,当修改obj1时,看obj2是否会发生变化,如果obj2也跟着变了,说明这是浅拷贝,反之如果obj2未变,那就是深拷贝。
要深刻理解浅拷贝和深拷贝的区别,那就不得不说一下基本类型数据和引用类型数据的区别。
一、基本类型数据
变量名字和值都会储存在栈内存中,js中的基本类型数据有:字符串、数值、布尔、undefined和null。例如:
var a = 'foo';
b = a; // 栈内存会开辟一个新的内存空间,此时b和a都是相互独立的
b = 'bar';
console.log(a); // 'foo'
这类操作算不上深拷贝,因为深拷贝本身只针对较为复杂的object类型数据。基本类型数据每次赋值都会开辟一块新的内存空间,因为基本类型数据占的内存毕竟很小。
二、引用类型数据
如果是引用数据类型,变量名存在栈内存中,值存在堆内存中,栈内存会提供一个引用的地址指向堆内存中的值。计算机为什么这样设计:就是因为引用数据占用内存较多。例如:
var obj1 = { a:'foo' }
var obj2 = obj1 //obj1赋值给obj2,
// 注意:虽然是赋值,但因为obj1是引用类型数据在堆内存中占用内存较多,
// 计算机为了节省开销,只是为obj2开辟一块栈存储地址并指向obj1的堆内存地址
obj2.a = 'bar' //所以当修改obj2中的值以后,查看obj1也跟着影响了
console.log( obj1.a )//bar
注意:以上操作修改obj2的同时obj1也修改了,但这不属于浅拷贝,这只是赋值操作。
延伸:
var obj1 = { a:'foo' }
var obj2 = obj1 //obj1赋值给obj2,
// 注意:虽然是赋值,但因为obj1是引用类型数据在堆内存中占用内存较多,
// 计算机为了节省开销,只是为obj2开辟一块栈存储地址并指向obj1的堆内存地址
obj2 = { a:'bar' } //注意:这是字面量创建对象,并且开辟了一个新的栈和堆内存空间,
//和obj1没有了关系
obj2.a = 'bar' //所以当修改obj2中的值以后,查看obj1未跟着影响
console.log( obj1.a )//foo
三、浅拷贝
简单的理解就是拷贝了对象的第一层属性,如果对象的某个属性还有第二层,第三层的数据等更多层的数据,浅拷贝是访问不到的。比如说某个属性的值是对象或数据,那浅拷贝无法复制该对象的数据。注意:诸如( var obj1 = {}; var obj2 = obj1 )的直接赋值操作不属于浅拷贝。
3.1 最简单的实现拷贝的方式为for ...in实现
function copy(data){
var newData
if (Object.prototype.toString.call(data) == '[object Array]') { //判断数组
newData = []
} else if (Object.prototype.toString.call(data) == '[object Object]') { //判断对象
newData = {}
}
for (var attr in data) { //for...in只拷贝了一层
newData[attr] = data[attr]
}
return newData
}
3.2Object.assign方法
function copy(data){
var newData
if (Object.prototype.toString.call(data) == '[object Array]') { //判断数组
newData = []
} else if (Object.prototype.toString.call(data) == '[object Object]') { //判断对象
newData = {}
}
for (var attr in data) { //for...in只拷贝了一层
newData[attr] = data[attr]
}
return newData
}
3.3 ...spread展开运算符实现
var obj = { a:'foo',b:'bar' }
var newObj = { ...obj } //展开运算符实现浅拷贝
obj.a = 1
console.log( newObj.a )//foo
四、深拷贝实现方式
4.1 浅拷贝+递归
var obj1 = {
a: 'foo'
}
var obj2 = {
b: 'bar'
}
var newObj= {}
Object.assign(newObj,obj1,obj2); //实现浅拷贝
obj1.a = 1;
console.log(newObj.a) // foo
4.2 通过json方法 JSON.parse(JSON.stringify())
var obj = { a:'foo',b:'bar' }
var newObj = { ...obj } //展开运算符实现浅拷贝
obj.a = 1
console.log( newObj.a )//foo
注意:json方法实现深拷贝有问题,如果源数据中有函数,则会丢失。
var obj1 = {
a: 'foo',
fn:function(){
console.log('fn handle')
}
}
let obj2 = JSON.parse(JSON.stringify(obj1 )); //方法会丢失
obj2.fn() //obj2.fn is not a function
4.3 函数库lodash
var _ = require('lodash');
var obj1 = {
a:'foo'
};
var obj2 = _.cloneDeep(obj1); //lodash函数库实现拷贝
obj1.a = 1
console.log(obj2.a)// foo
4.4 通过jQuery的extend实现深拷贝
var obj1 = {
a:'foo',
b:[ 'foo','bar' ]
};
var obj2= $.extend(true,obj1); // true为深拷贝,false为浅拷贝
相关推荐
- “版本末期”了?下周平衡补丁!国服最强5套牌!上分首选
-
明天,酒馆战棋就将迎来大更新,也聊了很多天战棋相关的内容了,趁此机会,给兄弟们穿插一篇构筑模式的卡组推荐!老规矩,我们先来看10职业胜率。目前10职业胜率排名与一周前基本类似,没有太多的变化。平衡补丁...
- VS2017 C++ 程序报错“error C2065:“M_PI”: 未声明的标识符"
-
首先,程序中头文件的选择,要选择头文件,在文件中是没有对M_PI的定义的。选择:项目——>”XXX属性"——>配置属性——>C/C++——>预处理器——>预处理器定义,...
- 东营交警实名曝光一批酒驾人员名单 88人受处罚
-
齐鲁网·闪电新闻5月24日讯酒后驾驶是对自己和他人生命安全极不负责的行为,为守护大家的平安出行路,东营交警一直将酒驾作为重点打击对象。5月23日,东营交警公布最新一批饮酒、醉酒名单。对以下驾驶人醉酒...
- Qt界面——搭配QCustomPlot(qt platform)
-
这是我第一个使用QCustomPlot控件的上位机,通过串口精确的5ms发送一次数据,再将读取的数据绘制到图表中。界面方面,尝试卡片式设计,外加QSS简单的配了个色。QCustomPlot官网:Qt...
- 大话西游2分享赢取种族坐骑手办!PK趣闻录由你书写
-
老友相聚,仗剑江湖!《大话西游2》2021全民PK季4月激燃打响,各PK玩法鏖战齐开,零门槛参与热情高涨。PK季期间,不仅各种玩法奖励丰厚,参与PK趣闻录活动,投稿自己在PK季遇到的趣事,还有机会带走...
- 测试谷歌VS Code AI 编程插件 Gemini Code Assist
-
用ClaudeSonnet3.7的天气测试编码,让谷歌VSCodeAI编程插件GeminiCodeAssist自动编程。生成的文件在浏览器中的效果如下:(附源代码)VSCode...
- 顾爷想知道第4.5期 国服便利性到底需优化啥?
-
前段时间DNF国服推出了名为“阿拉德B计划”的系列改版计划,截至目前我们已经看到了两项实装。不过关于便利性上,国服似乎还有很多路要走。自从顾爷回归DNF以来,几乎每天都在跟我抱怨关于DNF里面各种各样...
- 掌握Visual Studio项目配置【基础篇】
-
1.前言VisualStudio是Windows上最常用的C++集成开发环境之一,简称VS。VS功能十分强大,对应的,其配置系统较为复杂。不管是对于初学者还是有一定开发经验的开发者来说,捋清楚VS...
- 还嫌LED驱动设计套路深?那就来看看这篇文章吧
-
随着LED在各个领域的不同应用需求,LED驱动电路也在不断进步和发展。本文从LED的特性入手,推导出适合LED的电源驱动类型,再进一步介绍各类LED驱动设计。设计必读:LED四个关键特性特性一:非线...
- Visual Studio Community 2022(VS2022)安装图文方法
-
直接上步骤:1,首先可以下载安装一个VisualStudio安装器,叫做VisualStudioinstaller。这个安装文件很小,很快就安装完成了。2,打开VisualStudioins...
- Qt添加MSVC构建套件的方法(qt添加c++11)
-
前言有些时候,在Windows下因为某些需求需要使用MSVC编译器对程序进行编译,假设我们安装Qt的时候又只是安装了MingW构建套件,那么此时我们该如何给现有的Qt添加一个MSVC构建套件呢?本文以...
- Qt为什么站稳c++GUI的top1(qt c)
-
为什么现在QT越来越成为c++界面编程的第一选择,从事QT编程多年,在这之前做C++界面都是基于MFC。当时为什么会从MFC转到QT?主要原因是MFC开发界面想做得好看一些十分困难,引用第三方基于MF...
- qt开发IDE应该选择VS还是qt creator
-
如果一个公司选择了qt来开发自己的产品,在面临IDE的选择时会出现vs或者qtcreator,选择qt的IDE需要结合产品需求、部署平台、项目定位、程序猿本身和公司战略,因为大的软件产品需要明确IDE...
- Qt 5.14.2超详细安装教程,不会来打我
-
Qt简介Qt(官方发音[kju:t],音同cute)是一个跨平台的C++开库,主要用来开发图形用户界面(GraphicalUserInterface,GUI)程序。Qt是纯C++开...
- Cygwin配置与使用(四)——VI字体和颜色的配置
-
简介:VI的操作模式,基本上VI可以分为三种状态,分别是命令模式(commandmode)、插入模式(Insertmode)和底行模式(lastlinemode),各模式的功能区分如下:1)...
你 发表评论:
欢迎- 一周热门
- 最近发表
-
- “版本末期”了?下周平衡补丁!国服最强5套牌!上分首选
- VS2017 C++ 程序报错“error C2065:“M_PI”: 未声明的标识符"
- 东营交警实名曝光一批酒驾人员名单 88人受处罚
- Qt界面——搭配QCustomPlot(qt platform)
- 大话西游2分享赢取种族坐骑手办!PK趣闻录由你书写
- 测试谷歌VS Code AI 编程插件 Gemini Code Assist
- 顾爷想知道第4.5期 国服便利性到底需优化啥?
- 掌握Visual Studio项目配置【基础篇】
- 还嫌LED驱动设计套路深?那就来看看这篇文章吧
- Visual Studio Community 2022(VS2022)安装图文方法
- 标签列表
-
- wireshark怎么抓包 (75)
- qt sleep (64)
- cs1.6指令代码大全 (55)
- factory-method (60)
- sqlite3_bind_blob (52)
- hibernate update (63)
- c++ base64 (70)
- nc 命令 (52)
- wm_close (51)
- epollin (51)
- sqlca.sqlcode (57)
- lua ipairs (60)
- tv_usec (64)
- 命令行进入文件夹 (53)
- postgresql array (57)
- statfs函数 (57)
- .project文件 (54)
- lua require (56)
- for_each (67)
- c#工厂模式 (57)
- wxsqlite3 (66)
- dmesg -c (58)
- fopen参数 (53)
- tar -zxvf -c (55)
- 速递查询 (52)