Appearance
基础汇总2
7.3 Async和Await
- async 函数 用于声明一个函数是异步的。它可以让我们使用同步的写法来处理异步操作,使代码更加简洁易读。
- await 用于等待一个异步方法执行完成。它会暂停当前函数的执行,直到异步方法返回结果。
- async/await 是一个用同步思维解决异步问题的方案。它可以让我们像编写同步代码一样编写异步代码,避免了嵌套回调的“回调地狱”。
async 函数的返回值:
- 如果 async 函数的返回值不是 Promise,那么它会隐式地包装在一个 Promise 中。
- 如果没有在 async 函数中写 return 语句,那么 Promise 对象的 resolve 值就是 undefined。
await 表达式:
- await 后面接 expression,返回值为 expression 的值。
- await 表达式会暂停整个 async 函数的执行进程并出让其控制权。
- 当 await 后面的函数为 Promise 时,会等待 resolve 或 reject 返回之后才会恢复进程。
- Promise 的返回值会作为 await 表达式的返回值。
- 当 await 后面的函数无返回值时,表达式结果为 undefined。
- 如果函数体内有一个 await 表达式,async 函数就一定会异步执行。
- 在 await 表达式之后的代码可以被认为是存在在链式调用的 then 回调中。
- 多个 await 表达式都将加入链式调用的 then 回调中,返回值将作为最后一个 then 回调的返回值。
7.5 Promise
Promise 是一个对象,代表一个异步操作的最终完成(成功或失败)及其结果。它是异步编程的一种解决方案,通过链式调用解决回调地狱问题。
Promise 的状态:
- Pending 等待态: 初始状态,表示异步操作尚未完成。
- Fulfilled 成功态: 表示异步操作成功完成,并带有成功结果。
- Rejected 失败态: 表示异步操作失败,并带有失败原因。
Promise 的特点:
- 一个 Promise 对象只能改变一次状态。
- 无论成功还是失败,都会有一个结果数据。
- 成功的结果数据一般称为 value,而失败的结果一般称为 reason。
手写promise
js
function myPromise(exector){
const self = this;
this.state = 'pending'; // promise状态
this.value; //promise的值
this.onFullfilledCb = []; //存储then方法中注册的回调函数(第一个参数)
this.onRejectedCb = []; //存储then方法中注册的回调函数(第二个参数)
this.reason = ''; //失败原因
try{
exector(resolve, reject); // exector(resolve.bind(this), reject.bind(this))
}cacth((e)=>{
reject(e);
});
//成功执行
function resolve(value){
if(self.state === 'pending'){ //用bind之后,可使用this
self.state = 'fullfilled';
self.value = value;
self.onFullfilledCb.forEach((fn)=>{ fn() })
}
}
//失败执行
function reject(e){
if(self.state === 'pending'){
self.state = 'rejected';
self.reason = e;
self.onRejectedCb.forEach((fn)=>{ fn(e) })
}
}
}
手写 Promise.then
js
myPromise.prototype.then = function(onFullfiled, onRejected){
const self = this;
if(this.state === 'fullfiled'){
onFullfiled(self.value);
}
if(this.state === 'rejected'){
onRejected(self.reason);
}
if(this.state === 'pending'){
this.onFullfilledCb.push(()=>{ onFullfiled(self.value )});
this.onRejected.push(()=>{ onRejectedCb(self.reason )});
}
}
手写Promise.all
js
myPromise.prototype.all = function(promiseArray){
return new myPromise((resolve, reject)=>{
let resultCount = 0;
let resultArray = new Array(promiseArray.length);
for(var i=0; i<promise.length; i++){
Promise.resolve(promise[i].then(res=>{ //Promise.resolve 用于处理数组中参数为 非promise
resultCount++;
resultArray[i] = res;
if(resultCount === promise.length){
resolve(resultArray);
}
}, error=>{reject(error)}))
}
})
}
Promise.all 入参可以不是数组,可以是任何可迭代对象。当其中一个任务报错时,其他还会正常执行
7.6 原型链与继承
7.6.1 构造函数constructor
构造函数 是一个类的特殊方法,用于创建该类的新实例。它会在使用 new 关键字创建对象时自动执行。
构造函数的作用:
- 初始化成员变量和方法
- 创建新对象
super 关键字
- super 关键字 用于访问父类中的成员和方法。
在构造函数中使用 super 关键字:
- 在构造函数中使用 super 关键字时,必须在使用 this 关键字之前使用。
- super 关键字可以用来调用父类中的构造函数。
- 默认情况下,JavaScript 会自动在无参构造函数中添加 super() 语句。
- 如果自定义了构造函数,则需要手动调用 super() 语句,否则会报错。
构造函数分为 实例成员 和 静态成员
- 实例成员:实例成员就是在构造函数内部,通过this添加的成员。实例成员只能通过实例化的对象来访问。
- 静态成员:在构造函数本身上添加的成员,只能通过构造函数来访问。
构造函数.prototype ==> 实例原型 实例原型.constructor ==> 构造函数
js
function Star(name,age) {
this.name = name; //实例成员
this.age = age; //实例成员
}
//静态成员
Star.sex = '女';
new一个新对象的过程,发生了什么?
(1) 创建空对象 son {}
(2) 将对象的原型链连接到父类对象的原型上:son.__proto__ = Father.prototype
(3) 将 this 指针重新绑定到新对象上:使构造函数的this指向新对象Father.call(this)
(4) 执行构造函数的代码,为新对象添加属性和方法:son.name
(5) 返回新对象
7.6.2 原型
每一个javascript对象(除null外)创建的时候,都会与之关联另一个对象(父类对象),这个父类对象就是我们所说的原型,每一个对象都会从原型中“继承”属性。
MDN:
+ 每个对象拥有一个原型对象,对象以其原型为模板、从原型继承方法和属性。
+ 原型对象也可能拥有原型,并从中继承方法和属性,一层一层、以此类推。
+ 这种关系常被称为原型链 (prototype chain),它解释了为何一个对象会拥有定义在其他对象中的属性和方法。
- 对象的原型prototype 【可以通过Object.getPrototypeOf(obj)或obj.__proto__获得】是每个实例都有的属性
- 构造函数的prototype属性 是构造函数的属性。JS实现类,是在 “...在对象实例和它的构造器之间建立一个链接(它是__proto__属性,是从构造函数的prototype属性派生的),之后通过上溯原型链,在构造器中找到这些属性和方法。”(MDN)
JS中,每个函数都有一个prototype属性,这个属性指向函数的原型对象。
__proto__
是每个对象都有的属性, 大多数情况下,__proto__
可以理解为“构造器的原型”,但是通过 Object.create()创建的对象有可能不是, Object.create()方法创建一个新对象,使用现有的对象来提供新创建的对象的__proto__
。
js
__proto__ === constructor.prototype
let a1 = { a: 1 };
let a2 = Object.create(a1);
console.log("a2.__proto__:", a2.__proto__); //Object {a: 1}
console.log(a2.__proto__ === a1.constructor.prototype); //false
let a3 = Object.create(a1.__proto__);
console.log("a3.__proto__",a3.__proto__);
console.log(a3.__proto__ === a1.constructor.prototype); //true
7.6.3 原型链
原型与原型之间通过__proto__层层串起来的直到Object.prototype.__proto__为null的链叫做原型链
7.6.4 继承
继承方式
原型链继承、构造函数继承、原型链+借用构造函数的组合继承、ES6中class 的继承、组合继承优化1、组合继承优化2
- 原型链继承
Student.prototype = new Person() // 子类型的原型为父类型的一个实例对象
特点:父类新增原型方法/原型属性,子类都能访问到; 简单,易于实现;
缺点:
- 无法实现多继承;
- 来自原型对象的所有属性被所有实例共享
- 创建子类实例时,无法向父类构造函数传参
- 要想为子类新增属性和方法,必须要在Student.prototype = new Person() 之后执行,不能放到构造器中
- 构造函数继承
jsfunction Student(name, age, price) { Person.call(this, name, age) //在子类型构造函数中通用call()调用父类型构造函数 ... }
特点:
- 解决了原型链继承中子类实例共享父类引用属性的问题
- 创建子类实例时,可以向父类传递参数
- 可以实现多继承(call多个父类对象)
缺点:
- 实例并不是父类的实例,只是子类的实例
- 只能继承父类的实例属性和方法,不能继承原型属性和方法
- 无法实现函数复用,每个子类都有父类实例函数的副本,影响性能
- 原型链+借用构造函数的组合继承
jsfunction Person (name, age) { this.name = name, this.age = age, this.setAge = function () { } } Person.prototype.setAge = function () { console.log("111") } function Student (name, age, price) { Person.call(this, name, age) this.price = price this.setScore = function () { } } Student.prototype = new Person() Student.prototype.constructor = Student//组合继承也是需要修复构造函数指向的 Student.prototype.sayHello = function () { } var s1 = new Student('Tom', 20, 15000) var s2 = new Student('Jack', 22, 14000) console.log(s1) console.log(s1.constructor) //Student console.log(p1.constructor) //Person
优点:
- 可以继承实例属性/方法,也可以继承原型属性/方法
- 不存在引用属性共享问题
- 可传参
- 函数可复用
缺点:
- 调用了两次父类构造函数,生成了两份实例
7.7 bind、apply、call
call、apply(第二个参数是一个参数数组)、bind都是改变this指向的方法。call、apply立即调用,bind返回绑定函数。
注意异常处理、fn = this找到要执行的函数、返回函数执行结果
手写 apply
js
Function.prototype._Apply = function (content = window, args = []) {//给content和args添加默认值
if (!(args instanceof Object)) {// 如果第二个参数不是对象的实例,就返回一个错误
throw new TypeError('Create List From Array Like called on non-object');
}
const fn = Symbol(); //指定唯一属性,防止delete删除错误
// 显示绑定函数this
content.fn = this;
// 执行fn方法,并接受返回值
const res = arguments[1] ? content.fn(...arguments[1]) : content.fn()
delete content.fn;// 删除fn方法
return res
}
手写 call
js
Function.prototype.Call = function(content=window){// 给执行上下文 添加默认值
const fn = Symbol(); //指定唯一属性,防止delete删除错误
content.fn = this//给 content 添加一个方法指向this;显示绑定this
// 获取第一个参数以后的所有参数
const args = Array.from(arguments).slice(1);
// 调用上下文的fn属性, 此时函数的this指向上下文, 也就是我们传入的content
const res = arguments.length > 1 ? content.fn(...args) : content.fn();
// 删除上下文的content的fn方法, 避免污染上下文
delete content.fn;
return res
}
注意: 参数拼接、返回函数<br>
new调用时this为fn的实例,否则fn执行上下文为入参context<br>
关联返回函数的原型, 避免在this上直接操作
手写 bind
js
Function.prototype.bind = function() {
var args = Array.prototype.slice.call(arguments);
var context = args.splice(0,1)[0];
var fn = this;
var temp = function() {};
var res = function() {
let rest = Array.prototype.slice.call(arguments);
// 如果new被调用, new绑定this优先级高, this应该指向fn的实例
// new运算符时 因为维护过原型,所以this既是temp的实例,也是fn的实例
return fn.apply(this instanceof temp ? this : context, args.concat(rest));
}
//维护res的原型, 避免直接在this上操作
if(this.prototype) {
temp.prototype = this.prototype;
}
res.prototype = new temp();
return res;
}
手写 new
new 操作运算符
- 创建一个新的对象
- 继承父类原型上的方法.
- 添加父类的属性到新的对象上并初始化. 保存方法的执行结果.
- 如果执行结果有返回值并且是一个对象, 返回执行的结果, 否则, 返回新创建的对象。
js
function _new(fn,...rest){
//基于fn的prototype构建对象的原型
const thisObj = Object.create(fn.prototype);
//将thisObj作为fn的this,继承其属性,并获取返回结果为result
const result = fn.apply(thisObj,rest);
//根据result对象的类型决定返回结果, result 通过Object.create创建是Object的实例
// 正常规定,如何fn返回的是null或undefined(也就是不返回内容),我们返回的是obj,否则返回 result
return result instanceof Object ? result : thisObj;
// 或者用下面方式判断
// const isObject = typeof res === "object" && res !== null;
// const isFunction = typeof res === "function";
// return isObject || isFunction ? res : obj;
}
手写 instanceof
js
const myInstanceof = (left, right) => {
if (typeof left !== "object" || left === null) {
return false;
}
let proto = Object.getPrototypeOf(left);
while (true) {
if (proto === null) return false;
if (proto === right.prototype) return true;
proto = Object.getPrototypeOf(proto);
}
};
8. 函数节流和防抖
为了解决函数高频率触发,浏览器无法正常响应的情况。 可视化
8.1 防抖 (debounce)
函数防抖,就是指触发时间后在规定时间内函数只能执行一次,如果在规定时间内又触发了事件,则会重新计算函数执行时间。
实现
js
function debounce(fn, delay){
let timer;
return function(){
clearTimeout(timer); //触发函数时, 重置
timer = setTimeout(()=>{
fn.apply(this, arguments);
}, delay)
}
}
应用场景 (连续事件触发一次)
- 搜索框搜索输入。只需用户最后一次输入完,再发送请求
- 手机号、邮箱验证输入检测
- 窗口大小Resize。只需窗口调整完成后,计算窗口大小。防止重复渲染。
8.2 节流(throttle)
限制一个函数在规定时间内只能执行一次。即连续触发事件时,在 n 秒中只执行一次函数。
实现
js
function throttle(fn, delay){
let isAvaliabed = true;
return function(){
const self = this;
if(self.isAvaliabed){
self.isAvaliabed = false;
setTimeout(()=>{
fn.bind(self, arguments);
self.isAvaliabled = true;
}, delay)
}
}
}
// 时间戳写法:第一次会执行,定时器写法:最后一次点击,还是会重新计时,综合如下
function throttle(fn, delay){
let timer = null;
let startTime = 0;
return function(){
let context = this;
let args = arguments;
let currTime = new Date();
let remain = delay - ( curTime - startTime );
clearTimeout(timer);
if(remain <= 0){
fn.apply(context, arguments)
startTime = new Date()
}else{
timer = setTimeout(fn, remain)
}
}
}
应用场景
- 滚动加载,加载更多或滚动底部监听
- 谷歌搜索框,搜索联想功能
- 高频点击提交,表单重复提交。
sort排序需要指定入参 nums.sort((a,b)=>{return a-b}), 直接在nums上进行排序
JS 手写
// 类数组转化为数组
js
const toArr = (likeArr) => {
const a1 = Array.from(likeArr);
const a2 = Array.prototype.slice.call(likeArr);
const a3 = [...likeArr];
const a4 = Array.prototype.concat.apply([], likeArr);
};
// 函数珂里化
js
function add(...args) {
const totalArgs = [...args];
function f(...args2) {
totalArgs.push(...args2);
return f;
}
f.toString = () => {
return totalArgs.reduce((prev, curr) => prev + curr, 0);
};
return f;
}
// console.log(add(1, 2, 3)(4)(5).toString());
// 图片懒加载
js
function lazyload() {
const imgs = document.querySelectorAll("img");
// 视口高度
const viewHeight = document.documentElement.clientHeight;
// 滚动条高度
const scrollHeight =
document.documentElement.scrollTop || document.body.scrollTop;
// 整个页面高度
// document.documentElement.scrollHeight
for (let i = 0; i < imgs.length; i++) {
const offsetTop = imgs[i].offsetTop;
if (offsetTop < viewHeight + scrollHeight) {
imgs[i].src = imgs[i].dataset.src;
}
}
}
手写 Promise
js
// 三种状态 不可逆
const ENUM = {
PENDING: 'pending',
FULLFILLED: 'fullfilled',
REJECTED: 'rejected',
}
class myPromise{
constructor(exector){
this.status = ENUM.PENDING; //状态标识
this.value = undefined; //成功返回值
this.reason = undefined; //失败说明
this.onRejectedCb = []; //成功回调数组
this.onFullFilledCb = []; //失败回调数组
try{
exector(resolve, reject);
}catch(e){
this.reason = e;
this.status = ENUM.REJECTED;
reject(e);
}
const resolve = (value)=>{
if(this.status === ENUM.PENDING){
this.value = value;
this.status = ENUM.FULLFILLED;
//依次执行成功状态回调函数
this.onFullFilledCb.forEach((cb)=>{
cb(this.value);
})
}
}
const reject = (reason)=>{
if(this.status === ENUM.PENDING){
this.reason = reason;
this.status = ENUM.REJECTED;
//依次执行失败状态回调函数
this.onRejectedCb.forEach((cb)=>{
cb(this.reason);
})
}
}
}
}
// Promise.then实现
// then方法传入两个参数 onFulfilled, onRejected
myPromise.prototype.then = function(onFulfilled, onRejected){
//入参异常处理
onFulfilled = typeof onFulfilled === "function" ? onFulfilled : (v) => v;
onRejected =
typeof onRejected === "function"
? onRejected
: (reason) => {
throw new Error(reason instanceof Error ? reason.message : reason);
};
// 保存this
const self = this;
return new myPromise((resolve, reject) => {
if (self.status === ENUM.PENDING) {
self.onFullFilledCb.push(() => {
try {
setTimeout(() => {
const result = onFulfilled(self.value);
result instanceof myPromise
? result.then(resolve, reject)
: resolve(result);
});
} catch (e) {
reject(e);
}
});
self.onRejectedCb.push(() => {
try {
setTimeout(() => {
const result = onRejected(self.reason);
result instanceof myPromise
? result.then(resolve, reject)
: resolve(result);
});
} catch (e) {
reject(e);
}
});
} else if (self.status === ENUM.FULFILLED) {
try {
setTimeout(() => {
const result = onFulfilled(self.value);
result instanceof myPromise
? result.then(resolve, reject)
: resolve(result);
});
} catch (e) {
reject(e);
}
} else if (self.status === ENUM.REJECTED) {
try {
setTimeout(() => {
const result = onRejected(self.reason);
result instanceof myPromise
? result.then(resolve, reject)
: resolve(result);
});
} catch (e) {
reject(e);
}
}
});
}
myPromise.prototype.all = function(arrList){
//入参异常处理
if (!Array.isArray(arrList)) {
const type = typeof arrList;
return new TypeError(`TypeError: ${type} ${arrList} is not iterable`)
}
return new myPromise((resolve, reject) => {
const backArr = []
const count = 0
const processResultByKey = (value, index) => {
backArr[index] = value
if (++count === arrList.length) {
resolve(backArr)
}
}
for (let i = 0; i < arrList.length; i++) {
const item = arrList[i];
if (item && item.then === 'function') {
item.then((value) => {
processResultByKey(value, i)
}, reject)
} else {
processResultByKey(item, i)
}
}
})
}
##数组常用方法
pop、push、reverse、shift、sort、concat、join、toString、indexOf、map(value, key)
indexOf() 方法可返回某个指定的字符串值在字符串中首次出现的位置(大小写敏感)。
- splice
方法向/从数组中添加/删除项目,然后返回被删除的项目
js
/*
index: 必需。整数,规定添加/删除项目的位置,使用负数可从数组结尾处规定位置。
howMany: 必需。要删除的项目数量。如果设置为 0,则不会删除项目。
item1: ..., itemX 可选。向数组添加的新项目。
*/
splice(index, howMany, item1, ..., itemX)
- slice
从已有的数组中返回选定的元素
js
/*
start: 规定从何处开始选取。如果是负数,那么它规定从数组尾部开始算起的位置, -1指最后一个元素
end: 可选。规定从何处结束选取。该参数是数组片断结束处的数组下标。如果没有指定该参数,那么切分的数组包含从 start 到数组结束的所有元素。
*/
slice(start, end)
- unshift
向数组的开头添加一个或更多元素(改变原数组),并返回新的数组
js
/*
newelement1~X:向数组添加的第一到N个元素
*/
unshift(newelement1,newelement2,....,newelementX)
- reduce
js
/*
reduce() 方法接收一个函数作为累加器,数组中的每个值(从左到右)开始缩减,最终计算为一个值
total 必需。初始值, 或者计算结束后的返回值。
currentValue 必需。当前元素
currentIndex 可选。当前元素的索引
arr 可选。当前元素所属的数组对象。
initialValue 可选。传递给函数的初始值
*/
reduce(function(total, currentValue, currentIndex, arr), initialValue)
https://codesandbox.io/s/javascript-shouxieti-rfo2c?file=/src/index.js
QA
class P{}; console.log(P._proto_)
每个对象都有__proto__的属性,大多情况下指向构造函数的原型.
每个函数都有一个prototype属性,这个属性指向函数的原型对象.
class实际是function的语法糖, P.proto 实际是Function.prototype.
说 class 是 Function 的语法糖,是因为 class 本质上可以被 Function 实现,两者在功能上没有本质区别。 class 语法最终会被编译成 Function 代码。例如:
js
class Person {
constructor(name) {
this.name = name;
}
sayHello() {
console.log(`Hello, my name is ${this.name}`);
}
}
// 编译后代码
function Person(name) {
this.name = name;
}
Person.prototype.sayHello = function() {
console.log(`Hello, my name is ${this.name}`);
};