既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上大数据知识点,真正体系化!
由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新
#### 手撕深拷贝
普通写法:
const obj1 = {
age: 20,
name: '张三',
address: {
city: '北京'
},
arr: ['x', 'y', 'z']
}
const obj2 = deepClone(obj1)
obj2.address.city = '上海'
obj2.arr[0] = 'a'
console.log(obj1.address.city) // 北京
console.log(obj1.arr[0]) // x
// 深拷贝
function deepClone(obj = {}) {
if(typeof obj !== 'object' || obj == null) {
// obj 是 null,或者不是对象和数组,直接返回
return obj
}
// 初始化返回结果
let result
if(obj instanceof Array) {
result = []
} else {
result = {}
}
for(let key in obj) {
// 保证 key 不是原型的属性
if(obj.hasOwnProperty(key)) {
// 递归
result[key] = deepClone(obj[key])
}
}
// 返回结果
return result
}
简写形式:
// 深拷贝函数
function deepClone(obj) {
// 1 判断是否是非引用类型或者null
if (typeof obj !== 'object' || obj == null) return obj
// 2 创建一个容器
let result = new obj.constructor()
// 3 拿到对象的keys,给容器赋值
Object.keys(obj).forEach(v => result[key] = deepClone(obj[key]))
// 4 返回容器
return result
}
### 变量计算 - 类型转换
* 字符串拼接
* ==
* if 语句
* 逻辑运算
示例 1:字符串拼接
const a = 100 + 10 // 110
const b = 100 + '10' // '10010'
const c = true + '10' // 'true10'
示例 2:==
100 == '100' // true
0 == '' // true
0 == false // true
false == '' // true
null = undefined // true
示例 2 扩展:
除了 `==` null 之外,其他都用 `===`
const obj = {name: 'zhangsan'}
if (obj.age == null) {}
// 相当于:
if (obj.age === null || obj.age === undefined) {}
示例 3:truly 变量和 falsely 变量
* truly 变量:`!!a == true` 的变量
* falsely 变量:`!!a === false` 的变量
以下是 falsely 变量,除此之外都是 truly 变量
!!0 === false
!!NaN === false
!!'' === false
!!null === false
!!undefined == false
!!false === false
示例 4:逻辑判断
注:10 是 truly 变量,继续往后判断返回第二个值
console.log(10 && 0) // 0
console.log('' || 'abc') // 'abc'
console.log(!window.abc) // true
## 原型和原型链
### class 类
* constructor
* 属性
* 方法
示例:
// 学生类
class Student {
constructor(name, number) {
this.name = name
this.number = number
}
sayHi() {
console.log(`姓名 ${this.name}, 学号 ${this.number}`);
}
}
// 通过类 new 对象/实例
const xialu = new Student('夏洛', '2022')
console.log(xialu.name) // 夏洛
console.log(xialu.number) // 2022
xialu.sayHi() // 姓名 夏洛, 学号 2022
### 继承
* extends
* super
* 扩展或重写方法
示例:子类继承父类
// 父类
class People {
constructor(name) {
this.name = name
}
eat() {
console.log(`${this.name} eat food`);
}
}
// 子类
class Student extends People {
constructor(name, number) {
super(name)
this.number = number
}
sayHi() {
console.log(`姓名 ${this.name} 学号 ${this.number}`);
}
}
// 子类
class Teacher extends People {
constructor(name, major) {
super(name)
this.major = major
}
teach() {
console.log(`${this.name} 教授 ${this.major}`)
}
}
// 实例
const xialuo = new Student('夏洛', '2022')
console.log(xialuo.name) // 夏洛
console.log(xialuo.number) // 2022
xialuo.sayHi() // 姓名 夏洛, 学号 2022
xialuo.eat() // 夏洛 eat food
// 实例
const wanglaoshi = new Teacher('王老师', '语文')
console.log(wanglaoshi.name) // 王老师
console.log(wanglaoshi.major) // 语文
wanglaoshi.teach() // 王老师 教授 语文
wanglaoshi.eat() // 王老师 eat food
### JS原型(隐式原型和显式原型)
#### 类型判断 instanceof
console.log(xialuo instanceof Student) // true
console.log(xialuo instanceof People) // true
console.log(xialuo instanceof Object) // true
#### 原型
class 实际上是 **函数**
console.log(typeof Student) // 'function'
console.log(typeof Teacher) // 'function'
console.log(typeof People) // 'function'
#### 隐式原型和显式原型
* 隐式原型:`__prop__`
* 显式原型:`prototype`
实例对象的隐式原型等于对应构造函数的显示原型
console.log(xialuo.__proto__) // People {constructor: ƒ, sayHi: ƒ}
console.log(Student.prototype) // People {constructor: ƒ, sayHi: ƒ}
console.log(xialuo.__proto__ === Student.prototype) // true
**出处:https://coding.imooc.com/lesson/400.html#mid=30288**
![在这里插入图片描述](https://img-blog.csdnimg.cn/bc3ff06b39774b1ca02e70b1957e0bd0.png)
#### 原型关系
* 每个 class 都有一个显式原型 `prototype`
* 每个实例都有隐式原型 `__proto__`
* 实例的 `__proto__` 指向对应 class 的 `prototype`
#### 基于原型的执行规则
* 获取属性 xialuo.name 或执行方法 xialuo.sayHi 时
* 先在自身属性和方法寻找
* 如果找不到则自动去 `__proto__` 中查找
### 原型链
instanceof 顺着隐式原型往上找,找到返回 true,找不到返回 false
**出处:https://coding.imooc.com/lesson/400.html#mid=30289**
![在这里插入图片描述](https://img-blog.csdnimg.cn/964392dc1218457fa9aec556382b03f6.png)
#### hasOwnProperty
hasOwnProperty 会查找一个对象是否有某个属性,但是不会去查找它的原型链
console.log(xialuo.hasOwnProperty('name')) // true
console.log(xialuo.hasOwnProperty('eat')) // false
### 手写简易 jQuery
通过 class类 和 原型,手写 jQuery 部分功能
<p>第一段文字</p>
<p>第二段文字</p>
<p>第三段文字</p>
class jQuery {
constructor(selector) {
const result = document.querySelectorAll(selector)
const length = result.length
for (let i = 0; i < length; i++) {
this[i] = result[i]
}
this.length = length
this.selector = selector
}
get(index) {
return this[index]
}
each(fn) {
for (let i = 0; i < this.length; i++) {
const elem = this[i]
fn(elem)
}
}
on(type, fn) {
return this.each(elem => {
elem.addEventListener(type, fn, false)
})
}
// 扩展很多 DOM API
}
// 插件
jQuery.prototype.dialog = function (info) {
alert(info)
}
// “造轮子”
class myJQuery extends jQuery {
constructor(selector) {
super(selector)
}
// 扩展自己的方法
addClass(className) {
}
style(data) {
}
}
测试:
![在这里插入图片描述](https://img-blog.csdnimg.cn/25431f8897e749118c0d982ebf29df05.png)
## 作用域和闭包
### 作用域
* 全局作用域
* 函数作用域
* 块级作用域
全局作用域:如 window 对象、document 对象
函数作用域:只能在函数里面使用
块级作用域:在块内有效
示例:**全局作用域**
在任何地方都能获取到
window.a = 'zhangsan'
function fn() {
console.log(window.a)
}
fn() // zhangsan
示例:**函数作用域**
里面的函数能读取到外面函数的变量
function fn1() {
let a = 'zhangsan'
function fn2() {
let b = 'lisi'
console.log(a)
}
fn2()
}
fn1() // zhangsan
外面的函数不能读取里面函数的变量
function fn1() {
let a = 'zhangsan'
console.log(b)
function fn2() {
let b = 'lisi'
}
fn2()
}
fn1() // 报错:b is not defined
示例:**块级作用域**
在块之外读取不到变量
if (true) {
let x = 100
}
console.log(x) // 报错:x is not defined
示例:创建 10 个 `<a>` 标签,点击的时候弹出来对应的序号
let a
for(let i = 0; i < 10; i++) {
a = document.createElement('a')
a.innerHTML = i + '<br>'
a.addEventListener('click', function (e) {
e.preventDefault()
alert(i)
})
document.body.appendChild(a)
}
![在这里插入图片描述](https://img-blog.csdnimg.cn/713db7603aba4bcc9129369b26bc6a27.png)
### 自由变量
* 一个变量在当前作用域没有定义,但被使用了
* 向上级作用域,一层一层依次寻找,直至找到为止
* 如果到全局作用域都没找到,则报错 xx is not defined
示例:不在当前作用域的就一层层往上找,a、b 都要往上层找
let a = 1
function fn1() {
let b = 2
function fn2() {
let c = 3
console.log(a + b + c)
}
fn2()
}
fn1() // 6
### 闭包
作用域应用的特殊情况,有两种表现:
* 函数作为返回值被返回
* 函数作为参数被传递
总结:**所有自由变量的查找,是在函数定义的地方,向上级作用域查找,不是执行的地方**
示例:函数作为返回值
function create() {
const a = 100
return function() {
console.log(a)
}
}
const fn = create()
const a = 200
fn() // 100
示例:函数作为参数
function print(fn) {
const a = 200
fn()
}
const a = 100
function fn() {
console.log(a)
}
print(fn) // 100
### this
* 作为普通函数:指向 window
* 使用 call apply bind :传入什么绑定什么([call、apply、bind的区别](https://bbs.csdn.net/forums/4f45ff00ff254613a03fab5e56a57acb))
* 作为对象方法被调用:指向当前对象本身,异步指向 window,箭头函数的异步指向上级作用域
* 在 class 方法中调用:指向当前实例本身
* 箭头函数:取上级作用域 this 的值
注:**this 的取值是在函数执行的时候确定的,不是在函数定义的时候确定的**
示例 1:普通函数、使用 call、apply、bind
function fn1() {
console.log(this)
}
fn1() // window
示例 2:call、apply、bind 指定指向
function fn1() {
console.log(this)
}
fn1() // window
fn1.call({x: 100}) // {x: 100}
const fn2 = fn1.bind({x: 200})
fn2() // {x: 200}
示例 3:作为对象方法被调用、箭头函数
const zhangsan = {
name: 'zhangsan',
sayHi() {
console.log(this)
},
wait() {
setTimeout(function() {
console.log(this)
})
},
waitAgain() {
setTimeout(() => {
console.log(this)
})
}
}
zhangsan.sayHi() // this即当前对象
zhangsan.wait() // this === window
zhangsan.waitAgain() // this即当前对象
![在这里插入图片描述](https://i-blog.csdnimg.cn/blog_migrate/8b516476e485cdcc3f4cfbf7c7dc2361.png)
示例 4:在 class 方法中调用
![img](https://img-blog.csdnimg.cn/img_convert/43db1e02b2d33e6e011d3465af1d301a.png)
![img](https://img-blog.csdnimg.cn/img_convert/af4a4dd6566c9d46419f7b3a6c6d62c9.png)
![img](https://img-blog.csdnimg.cn/img_convert/d34dc7e154c28c4774201316c7606cdf.png)
**既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上大数据知识点,真正体系化!**
**由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新**
**[需要这份系统化资料的朋友,可以戳这里获取](https://bbs.csdn.net/forums/4f45ff00ff254613a03fab5e56a57acb)**
setTimeout(() => {
console.log(this)
})
}
}
zhangsan.sayHi() // this即当前对象
zhangsan.wait() // this === window
zhangsan.waitAgain() // this即当前对象
示例 4:在 class 方法中调用
[外链图片转存中…(img-mjOgFdCE-1715470760740)]
[外链图片转存中…(img-ugwcxPgI-1715470760740)]
[外链图片转存中…(img-R7Y5UEU9-1715470760741)]
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上大数据知识点,真正体系化!
由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新