首页 > 其他分享 >JS值和类型(必学知识点总结)

JS值和类型(必学知识点总结)

时间:2022-12-20 19:45:14浏览次数:69  
标签:obj1 知识点 操作数 obj 对象 必学 JS let 拷贝

目录

值和类型


八种数据类型

undefinednullbooleannumberstringsymbolbigintobject


原始值和引用值

原始值:按值访问。值存储在栈中。变量赋值传递时会创建该值的副本,两个变量(复制与被复制)完全独立。

常见原始值类型:undefinednullbooleannumberstringsymbolbigint

let num1 = 5
let num1 = num2
num2 = 4 // num1此时不发生改变

引用值:按引用地址访问。对象存储在堆中,其引用地址会存储在栈中指向对象。变量赋值传递时会复制其引用地址,两个变量(复制与被复制)会指向同一个对象,对象数据的修改会同时反映到两个变量上。

常见引用类型(object):ObjectFunctionArrayDateRegExp等。

let obj1 = { a: 1 }
let obj2 = obj1
obj2.a = 4 // 此时发生改变 obj1.a = 1 => obj1.a = 4 

访问对象的方式

  1. .点访问:不支持变量名称和特殊字符。对象不存在的属性,通过点访问赋值后会创建该属性。
  2. []中括号访问:支持变量名称和特殊字符。对象不存在的属性,通过中括号访问赋值后也会创建该属性。
let a = {}
let key = 'sex'
a.name = '小明' // 对象.属性名
a['age'] = 22 // 对象['属性名']
a[key] = '男' // 对象[变量名]
console.log(a) // { name: '小明', age: 22 }

相等与全等运算符

==相等运算符:检查其两个操作数是否相等,它会尝试强制类型转换并且比较不同类型的操作数。

===全等运算符:检查它的两个操作数是否相等, 全等运算符总是认为不同类型的操作数是不同的。

注意:比较对象时,需要考虑其引用地址。也可用toString()方法转换后比较。

({name: '小红'}).toString() === ({name: '小米'}).toString() => true

相等和不相等操作符转换规则:

  1. 如果有一个操作数是布尔值,则在比较相等性之前先将其转换为数值——false转换为0,而true转换为1
  2. 如果一个操作数是字符串,另一个操作数是数值,在比较相等性之前先将字符串转换为数值;
  3. 如果一个操作数是对象,另一个操作数不是,则调用对象的valueOf()方法(返回对象原始值),用得到的基本类型值按照前面的规则进行比较;

相等和不相等操作符比较规则:

  1. nullundefined是相等的。
  2. 要比较相等性之前,不能将nullundefined转换成其他任何值。
  3. 如果有一个操作数是NaN,则相等操作符返回false,而不相等操作符返回true。重要提示:即使两个操作数都是NaN,相等操作符也返回false;因为按照规则,NaN不等于NaN
  4. 如果两个操作数都是对象,则比较它们是不是同一个对象。如果两个操作数都指向同一个对象,则相等操作符返回true;否则,返回false

typeof 和 instanceof

typeof 操作符

可推断类型(返回值:对应类型字符串)如下:

undefinedbooleanstirngnumberobjectfunctionbigint

注意:null值表示一个空对象指针,使用typeof操作符检测null值时会返回"object"

instanceof 操作符

可推断类型(返回值:布尔值)如下:

ObjectArrayDateFunctionRegExp、基本包装类型(BooleanNumberString)、单体内置对象(GlobalMath

手写 typeof 和 instanceof 方法

前置知识:原型链、toString方法

对象toString方法应用:

  1. 检测对象类型
let obj = new Number('5')
Object.prototype.toString.call(obj) // '[object Number]'
  1. 实现进制转换
let num = 5
num.toString(2) // '101'

手写typeof方法

function myTypeof(value) {
    return Object.prototype.toString.call(value).slice(8, -1).toLowerCase()
}

手写instanceof方法:

function myInstanceof(left, right) {
    while (1) {
        if (left === null) {
            return false
        }
        if (left === right.prototype) {
            return true
        }
        left = left.__proto__
    }
}

说明:变量left为实例对象,变量right为构造函数。


深拷贝与浅拷贝

浅拷贝

基本概念:对象的浅拷贝是其属性与拷贝源对象的属性共享相同引用(指向相同的底层值)的副本。

实现方式:

  1. Object.assign()
  2. Object.create()
  3. 扩展运算符:{ ...obj }
  4. Array.concat()
  5. Array.slice()
  6. Array.from()
  7. lodash API : _.clone()
let obj1 = { a: 2, b: { s: 4 } }
let obj2 = { ...obj1 }
obj2.a = 3 // obj1.a 不发生改变
obj2.b.s = 5 // obj1.b.s 此时发生改变,值更改为5

注意:以上方式仅对对象第一层作深拷贝,无法对多层级对象实现深拷贝

深拷贝

基本概念:对象的深拷贝是指其属性与其拷贝的源对象的属性不共享相同的引用(指向相同的底层值)的副本。

实现方式:

  1. 递归实现(简版)
function cloneDeep(obj) {
    let res = Array.isArray(obj[key]) ? [] : {};
    for (let key in obj) {
        if (obj.hasOwnProperty(key)) {
            if (obj[key] && obj[key] === 'object') {
                res[key] = cloneDeep(obj[key])
            } else {
                res[key] = obj[key]
            }
        }
    }
    return res
}
  1. JSON.parse(JSON.stringify(obj))

  2. lodash API : _.cloneDeep()

let obj1 = { a: 2, b: { s: 4 } }
let obj2 = JSON.parse(JSON.stringify(obj1))
obj2.a = 3 // obj1.a 不发生改变
obj2.b.s = 5 // obj1.b.s 不发生改变

注意:深拷贝会使源和副本互相独立,即对多层级对象能够实现深拷贝


参考

MDN-深拷贝

JavaScript高级程序设计(第4版)

标签:obj1,知识点,操作数,obj,对象,必学,JS,let,拷贝
From: https://www.cnblogs.com/chscript/p/16994944.html

相关文章

  • JS闭包和作用域(必学知识点总结)
    目录闭包和作用域变量声明变量和函数的声明提升作用域和作用域链执行上下文闭包垃圾回收机制闭包和作用域变量声明var声明特点在使用var声明变量时,变量会被自动添......
  • 根据分析查看相关知识点分析iOS 三种录制视频方式
    这篇文章讨论了关于如何配置视频捕获管线(pipeline)和最大限度地利用硬件性能的一些不同选择。这里有个使用了不同管线的样例app,可以在​​ GitHub​​查看。 第一......
  • JS混淆加密的作用
    在软件开发过程中,有时会使用代码混淆技术来使代码难以被阅读或破解。这种技术通常被用于防止恶意使用或盗用代码。在JavaScript中,有许多工具可以用来混淆代码,例如Google......
  • Docker daemon.json 的配置项目合集
    vim/etc/docker/daemon.json{"authorization-plugins":[],"data-root":"",#设置docker运行时的根目录"dns":[],#设置容器的DNS地址......
  • JS 的 9 种作用域
    作用域想必大家都知道,就是变量生效的范围,比如函数就会生成一个作用域,声明的变量只在函数内生效。而这样的作用域一共有9种,其中几种绝大多数前端都说不出来。下面我们就......
  • 记录--可视化大屏-用threejs撸一个3d中国地图
    这里给大家分享我在网上总结出来的一些知识,希望对大家有所帮助不想看繁琐步骤的,可以直接去github下载项目,如果可以顺便来个star哈哈本项目使用vue-cli创建,但不影响使......
  • .Net7 自动拷贝appsettings.json到debug文件下
    IDERider在配置json时遇到路径的问题Theconfigurationfile'appsettings.json'wasnotfoundandisnotoptional.TheexpectedphysicalpathwasIConfiguration......
  • puppeteer( Nodejs 版 selenium )快速入门
    puppeteer官网:​​https://pptr.dev/​​Puppeteer中文文档(与官Puppeteer中文文档 :​​https://learnku.com/docs/puppeteer/3.1.0​​Puppeteerv1.5.0中文翻peteer......
  • Node.js 教程
     菜鸟教程:​​https://www.runoob.com/nodejs/nodejs-tutorial.html​​nodejs官网:​​https://nodejs.org/en/download/​​廖雪峰nodejs:​​https://www.liaoxuefen......
  • mvc中,js 如何直接使用后端参数
    问题比如后端传了一个boolean类型的参数,js如果直接使用这个参数,比如这么写就是错的if(${redevice}){document.getElementById('redevice').checked=true;}......