IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> JavaScript知识库 -> es6总结 -> 正文阅读

[JavaScript知识库]es6总结

es6阮一峰

基本数据类型

  1. 数字、字符串、布尔、symbol、bigInt、undefined、null
  2. object

Set用法

类似于数组,但是成员的值都是唯一的,没有重复的值。

只有NaN=== 不一样  其他都是一样做比较的
var set = new Set(NaN)
set.add(NaN);
set // Set {NaN}
set.size //1
set.delete() //删除某个值 返回值表示是否删除成功
set.has() // 返回布尔值 表示是否有某个值
set.clear() // 清空所有项 没有返回值
[...new Set('ababbc')] // ['a','b','c']

也可以使用Array.from解开Set实例
let set = new Set(['red', 'green', 'blue']);
for (let item of set.keys()) {  console.log(item);}// red  green blue
for (let item of set.values()) {  console.log(item);}// red  green blue
// ["red", "red"]  ["green", "green"]  ["blue", "blue"]
for (let item of set.entries()) {  console.log(item);}



Set.prototype[Symbol.iterator] === Set.prototype.values // true  所以可以用for...of循环得到每一项值
for (let i of set) {
  console.log(i);//获取set内部的每一项
}

set.forEach((value, key) => console.log(key + ' : ' + value)) // key和value  值一样

可以把Set实例看做数组 进行map和filter操作 和数组用法类似



const p = [item,item,...] // item必须是对象也可以是数组 否则报错
const set = new WeakSet(b);
set.add(value)
set.delete(value)
set.has(value) 

set.size // undefined  没有size和forEach
set.forEach // undefined

Map用法

DOM 节点作为对象data的键,但是由于对象只接受字符串作为键名,所以element被自动转为字符串[object HTMLDivElement]。

const data = {};
const element = document.getElementById('myDiv');
data[element] = 'metadata';
data['[object HTMLDivElement]'] // "metadata"

为了解决这个问题,ES6 提供了 Map 数据结构。它类似于对象

var obj = {}
var map = new Map([
    [obj, '111'],
    ['key2', '222']
]);
map.get(obj) //111
//有这些方法  has get set delete    clear size forEach keys values entries
for...map.keys() =>  第一项是obj     也可以使用 [...map.keys()]
for...map.values() =>  第一项是"111"     也可以使用 [...map.values()]
for...map.entries() =>  第一项是[obj,'111']   也可以使用 [...map.entries()]
for...map.forEach(value, key) =>  第一项  value为'111'  key为obj    

WeakMap只接受对象作为键名
WeakMap 里面对element的引用就是弱引用,不会被计入垃圾回收机制。

只有这些方法 has get set delete
const wm1 = new WeakMap();

在这里插入图片描述

Promise

//请求某个table数据
function requestTableList(){
    var p = new Promise((resolve, reject) => {
           //去后台请求数据,这里可以是ajax,可以是axios,可以是fetch 
            resolve(res);
    });
    return p;
}
//延时函数,用于给请求计时 10s
function timeout(){
   var p = new Promise((resolve, reject) => {
       setTimeout(() => {
           reject('请求超时');
       }, 10000);
   });
   return p;
}
Promise.race([requestTableList(), timeout()]).then((data) =>{
  //进行成功回调处理
  console.log(data);
}).catch((err) => {
  // 失败回调处理
    console.log(err);
});
function promiseClick1(){
let p = new Promise(function(resolve, reject){
	setTimeout(function(){
		var num = Math.ceil(Math.random()*20); //生成1-10的随机数
		console.log('随机数生成的值:',num)
		if(num<=10){
			resolve(num);
		}
		else{
			reject('数字太于10了即将执行失败回调');
		}
	}, 2000);
   })
   return p
  }
function promiseClick2(){
	let p = new Promise(function(resolve, reject){
		setTimeout(function(){
			var num = Math.ceil(Math.random()*20); //生成1-10的随机数
			console.log('随机数生成的值:',num)
			if(num<=10){
				resolve(num);
			}
			else{
				reject('数字太于10了即将执行失败回调');
			}
		}, 2000);
   })
   return p
}
function promiseClick3(){
	let p = new Promise(function(resolve, reject){
		setTimeout(function(){
			var num = Math.ceil(Math.random()*20); //生成1-10的随机数
			console.log('随机数生成的值:',num)
			if(num<=10){
				resolve(num);
			}
			else{
				reject('数字太于10了即将执行失败回调');
			}
		}, 2000);
	})
	return p
 }

Promise.all([promiseClick3(), promiseClick2(), promiseClick1()]).then(function(results){
	console.log(results);
})

class继承

父辈为超类  子辈为子类   子类继承超类 调用super其实是调用超类的构造函数方法
class A {
    constructor(a) {
        this.a = a;
    }
}
class B extends A {
    constructor(a, b) {
        super(a);
        this.b = b;
    }
}
class C extends B {
    constructor(a, b, c) {
        super(a, b);
        this.c = c;
    }
}
console.log(new C(1, 2, 3))//C { a: 1, b: 2, c: 3 }


Promise并行和串行,以及await底层实现

前提代码

var query = function (value) {
    return new Promise(function (resolve, reject) {
        setTimeout(function () {
            resolve(value)
        }, value)
    })
}

并行 方式一 1秒后输出1000,再过1秒输出2000,再过1秒输出3000

query(1000).then(v => {
    console.log(v)
})
query(2000).then(v => {
    console.log(v)
})
query(3000).then(v => {
    console.log(v)
})

并行 方式二 三秒后输出 [1000, 2000, 3000]

Promise.all([query(1000),query(2000),query(3000)]).then(v=>{
    console.log(v)
})

串行 方式一 输出为 1秒后输出1000,再过2秒输出2000,再过3秒输出3000

query(1000).then(v => {
    console.log(v)
    return query(2000)
}).then(v => {
    console.log(v)
    return query(3000)
}).then(v => {
    console.log(v)
})

串行 方式二 输出为 1秒后输出1000,再过2秒输出2000,再过3秒输出3000

let itor = generator()
itor.next().value.then(value => {
    itor.next(value).value.then(value => {
        itor.next(value).value.then(value => {
            itor.next(value)
        })
    })
})

await底层实现

const isPromise = function (v) {
    if (v !== null && /^(object|function)$/i.test(typeof v)) {
        let then;
        try {
            then = x.then
        } catch (err) {
            return false
        }
        if (typeof then === 'function') return true
    }
    return false
}

function AsyncFunction(generator, ...params) {
    return new Promise((resolve, reject) => {
        let itor = generator(...params);
        const next = v => {
            let {
                value,
                done
            } = itor.next(v)
            if (done) {
                resolve(value)
                return
            }
            if (!isPromise(value)) value = Promise.resolve(value)
            value.then(v => {
                next(v)
            }).catch(reason => {
                reject(reason)
                itor.throw(reason)
            })
        }
        next()
    })
}

AsyncFunction(function* generator() {
    let value;
    y1 = yield query(1000)
    console.log(y1)
    y2 = yield query(2000)
    console.log(y2)
    y3 = yield query(3000)
    console.log(y3)
    return 'zanlan'
}).then(v => {
    console.log('所有请求都成功', v)
}).catch(reason => {
    console.log('某个请求失败', reason)
})

一般函数实现

    /* 真实项目 基于async/awai可以直接实现*/
    /* 我们上面基于promise + generator + AsyncFunction函数,实现的就是async/await处理机制 */
    /* async/awai是promise + generator的语法糖 */
    (async function () {
        let value;
        try {
            value = await Promise.reject('xxx')
            console.log('第一个请求成功', value)
        } catch (err) {
            console.log(err)
        }
        value = await query(2000)
        console.log('第二个请求成功', value)
        value = await query(3000)
        console.log('第三个请求成功', value)
    })()

generator原理

把一个函数当做普通函数执行,是否可以创造它这个类的一个实例?

	jquery.fn()  // 工厂设计模式  实现  

	function* fn() {  // generator 实现  ,itor 是fn的实例对象 fn内部代码没有执行
	    console.log(this)
	    return 10
	}
	let itor = fn()
	console.log(itor)

输出 window,在输出 { value:10 , done:true }

function* fn() {
    console.log(this)
    return 10
}
let itor = fn()
console.log(itor.next())

输出 如下

在这里插入图片描述

function* generator() {
    console.log(1)
    yield 2
    console.log(3)
    yield 4
    console.log(5)
    yield 6
    console.log(7)
    return 8
}
let itor = generator()
console.log(itor.next())
console.log(itor.next())
console.log(itor.next())
console.log(itor.next())

输出 如下

在这里插入图片描述

function* generator() {
    console.log(1)
    let y2 = yield 2
    console.log(y2)
    console.log(3)
    let y4 = yield 4
    console.log(y4)
    console.log(5)
    let y6 = yield 6
    console.log(y6)
    console.log(7)
    return 8
}

let itor = generator()
console.log(itor.next('aa'))
console.log(itor.next('bb'))
console.log(itor.next('cc'))
console.log(itor.next('dd'))

输出如下

在这里插入图片描述

function* generator1() {
    yield 2;
    yield 3;
    return 4
}

function* generator2() {
    yield 1;
    yield* generator1();
    yield 5;
    return 6;
}

let itor = generator2();
console.log(itor.next())
console.log(itor.next())
console.log(itor.next())
console.log(itor.next())
console.log(itor.next())

Iterator 源码

class Iterator {
    constructor(assemble) {
        let self = this;
        self.assemble = assemble;
        self.index = 0;
    }
    next() {
        let self = this;
        let assemble = self.assemble;
        if (self.index > assemble.length - 1) {
            return {
                done: true,
                value: undefined
            }
        }
        return {
            done: false,
            value: assemble[self.index++]
        }
    }
}

一般应用

let itor = new Iterator([10, 20, 30, 40])
console.log(itor.next())
console.log(itor.next())
console.log(itor.next())
console.log(itor.next())
console.log(itor.next())

下面代码可以使得一般对象使用for of 循环

Object.prototype[Symbol.iterator] = function () {
    let self = this
    let keys = Object.keys(self).concat(Object.getOwnPropertySymbols(self))
    let index = 0
    return {
        next() {
            if (keys.length - 1 < index) {
                return {
                    done: true,
                    value: undefined
                }
            }
            return {
                value: self[keys[index++]],
                done: false
            }
        }
    }
}

针对类数组,则直接借用真数组自带的Symbol.iterator函数

var obj = {
    0: 'a',
    1: 'b',
    length: 2
}
obj[Symbol.iterator] = Array.prototype[Symbol.iterator]

for (var value of obj) {
    console.log(value)
}

jquery是类数组,下面是jquery源码实现

jQuery.fn[Symbol.iterator] = [][Symbol.iterator];

async && await 面试题

已知两个函数,分别提问,得出结果

var f1 = function f1() {
    console.log(1)
    return new Promise(resolve => {
        setTimeout(() => {
            resolve(6)
            console.log(5)
        }, 2000);
    })
}
var f2 = function f2() {
    console.log(2)
    return new Promise(resolve => {
        setTimeout(() => {
            resolve(4)
            console.log(3)
        }, 1000);
    })
}

问题1 输出结果为 :1 5 6 2 3 4

var f = async function () {
    console.log(await f1())
    console.log(await f2())
}
f()

问题2 输出结果为 :1 2 3 5

Promise.all([
    f1(),
    f2(),
])

问题3 输出结果为 :1 2 3 4 5 6

Promise.all([
    (async () => { console.log(await f1()) })(),
    (async () => { console.log(await f2()) })(),
])

class继承

父辈为超类  子辈为子类   子类继承超类 调用super其实是调用超类的构造函数方法
class A {
    constructor(a) {
        this.a = a;
    }
}
class B extends A {
    constructor(a, b) {
        super(a);
        this.b = b;
    }
}
class C extends B {
    constructor(a, b, c) {
        super(a, b);
        this.c = c;
    }
}
console.log(new C(1, 2, 3))//C { a: 1, b: 2, c: 3 }

ES5和ES6的模块化导入导出

//不接收对象 
require:require('s.css'); //(es5)
improt  's.css' //(es6)

//接收对象
var o = require('s.js'); //es(5)
import o form s.js    //(es6)
导出一个模块对象(es5):
module.exports={
    add:add,
    sub:sub
}
导出一个模块对象(es6):
module.exports={
    add,
    sub
}
注意:这种写法属性名和属性值变量是同一个,否则要分开写
module.exprots={
    addFn:add,
    sub
}  
//es5
module.exports={
    addFun:function(x,y){
        return x+y;
    }
}
//es6
module.exports={
    addFun(x,y){
        return x+y;
    }
}
calc.js中有两个函数:
function add(){}
function sub(){}
//写法一
es5写法:
module.exports.add = add;
module.exports.sub = sub;
使用:
var calc = require('./calc.js');
es6写法:
exprot function add(){}
exprot function sub(){}

//写法二
es5:
module.exports = {add:add,sub:sub};
es6:
exprot default{
    add,sub
}
 //表示取得calc.js中所有暴露出来的对象(es6)
import calc from './calc.js'

//只获取到calc.js中的add方法(按需获取)
import {add} from './calc.js'
1、如果模块中是使用 export default {} 方式导出的对象
    只能通过  import 对象名称 from '模块路径'
    不能通过  import {对象名称} from '模块路径'

2、如果就想要import {对象名称} from '模块路径'通过这种方式来按需导入对象中的某个属性
    那么应该使用 export 跟着要导出的对象或者方法名称
    export function add(){}
    export function substrct(){}

    那么就可以使用:
    import {add,substrct} from '模块路径'
    只需要直接使用 add()方法即可
    注意这里不能直接使用:  import cacl from '模块路径' 这种方式导入,会报错

class继承

父辈为超类  子辈为子类   子类继承超类 调用super其实是调用超类的构造函数方法
class A {
    constructor(a) {
        this.a = a;
    }
}
class B extends A {
    constructor(a, b) {
        super(a);
        this.b = b;
    }
}
class C extends B {
    constructor(a, b, c) {
        super(a, b);
        this.c = c;
    }
}
console.log(new C(1, 2, 3))//C { a: 1, b: 2, c: 3 }

ES5 和 ES6 及Object用到过的方法

es6的prototype不可以枚举

class Person {
    constructor(x, y) {}
    toString() {}
}
console.log(Object.keys(Person.prototype)); //[]
console.log(Object.getOwnPropertyNames(Person.prototype)); //["constructor", "toString"]

function Person1() {}
Person1.prototype.toString = function () {};
console.log(Object.keys(Person1.prototype)); //["toString"]
console.log(Object.getOwnPropertyNames(Person1.prototype)); //["constructor", "toString"]

自动添加constructor函数


class Person {
    
}
等 价
class Person {
    constructor(x, y) {}
}

constructor默认return 实例化对象this,但可以自定义
class函数必须要new才能调用,es5不需要new也可以调用

class Foo{
	constructor(){
		return Object.create(null);
	}
}
new Foo() instanceof Foo //false

修改一个实例__proto__的方法,其他的实例也会牵连

class Person {
    constructor(x, y) {
        this.x = x;
        this.y = y;
    }
    toString() {
        return this.x + '--' + this.y;
    }
}
let p = new Person(1, 2);

console.log(p.hasOwnProperty('x')); //true
console.log(p.hasOwnProperty('y')); //true
console.log(p.hasOwnProperty('constructor')); //false
console.log(p.hasOwnProperty('toString')); //false
console.log(p.__proto__.hasOwnProperty('constructor')); //true
console.log(p.__proto__.hasOwnProperty('toString')); //true

let p1 = new Person(2, 3);
p1.__proto__.toString = function () {
    return 'what fuck?';
};
let p2 = new Person(3, 4);
console.log(p1.toString());//what fuck?
console.log(p2.toString());//what fuck?

class可以自定义变量,Person只对构造函数内部有效,而clafun则用于外部调用

let clafun = class Person {
    constructor(x, y) {
        this.x = x;
    }
    toString() {
        return this.x + '--' + Person.name;
    }
};
let p = new clafun(1);
console.log(p.toString());//1--Person
// var p1 = new Person(1);//Person is not defined

//如果没有用到clafun则可以简写为
//如果没有用到clafun则可以简写为
let p = new class Person {
    constructor(x, y) {
        this.x = x;
    }
    toString() {
        return this.x + '--' + Person.name;
    }
}(1);
console.log(p.toString());//1--Person
// var p1 = new Person(1);//Person is not defined

class不存在变量提升,let不会变量提升,假设class存在变量提升,则为报错Foo 是undefined ,而实际上没有报错,说明class不会变量提升

{
    let Foo=class {};
    class Bar extends Foo{
        
    }
}

this的指向问题,默认指向实例化对象,但是通过解构出来的方法,则函数内部的this是指向环境的因为es6默认是用严格模式的,所以是指向undefined,解决方法有三种

  1. bind改变this的指向
  2. 箭头函数
  3. proxy
console.log(this); //Window {postMessage: ?, blur: ?, focus: ?, close: ?, parent: Window, …}
class Person {
    constructor() {
        // this.fun1 = this.fun1.bind(this); //方法一
        // this.fun1 = () => {
        //     //方法二
        //     return this.fun2() + '-------------';
        // };
    }
    fun1() {
        console.log(this); //实例化调用则指向实例化  解构出来则指向环境,es6默认是严格模式,所以为undefined
        return this.fun2() + '--------';
    }
    fun2() {
        return 'what';
    }
}

let p = new Person();
console.log(p.fun1()); //what--------
const { fun1 } = p;
console.log(fun1()); //fun1内部的this指向运行的环境   Cannot read property 'fun2' of undefined

proxy添加this

function selfish(target) {
    const cache = new WeakMap();
    const handler = {
        get(target, key) {
            const value = Reflect.get(target, key);
            if (typeof value !== 'function') {
                return value;
            }
            if (!cache.has(value)) {
                cache.set(value, value.bind(target));
            }
            return cache.get(value);
        },
    };
    const proxy = new Proxy(target, handler);
    return proxy;
}
const p = selfish(new Person());

class继承static方法 es6只有静态方法没有静态属性

class Foo {
    static fun1() {
        return '111';
    }
}
class Bar extends Foo {
    static fun2() {
        return super.fun1() + '222';
    }
}
console.log(Bar.fun1());//111
console.log(Bar.fun2());//111222

class声明实例化属性和方法

class Foo {
   constructor() {
        this.state1 = {};
    }
    state = { name: 1 };
    fun1() {
        return 11111;
    }
}
console.log(new Foo().hasOwnProperty('state1')); //true
console.log(new Foo().hasOwnProperty('state')); //true
console.log(new Foo().__proto__.hasOwnProperty('fun1')); //true

ES5和ES6的get和set方法的比较

ES5

var obj = {};
Object.defineProperty(obj, 'name', {
    get: function () {
        console.log('正在访问name');
        return this._name;
    },
    set: function (val) {
        console.log('正在修改name');
        this._name = val;
    },
});

obj.name = 10;
console.log(obj);
console.log(obj.name);
// 正在修改name
// { _name: 10 }
// 正在访问name
// 10

ES6

class Person{
	constructor(){
		this._name = '';
	}
	get name(){
		console.log("正在访问name属性");
		return `我的名字是${this._name}`;
	}
	set name(val){
		console.log("正在修改name属性");
		this._name = val;
	}
}

const person = new Person();
person.name = "德玛-压缩";
console.log(person.name); //先访问set 再访问get

// 正在修改name
// 正在访问name
// 我的名字是德玛-压缩

es7

在这里插入图片描述

let a = 7 ** 12
let b = 2 ** 7
console.log(a === Math.pow(7,12)) // true    7的12次方
console.log(b === Math.pow(2,7)) 
  JavaScript知识库 最新文章
ES6的相关知识点
react 函数式组件 & react其他一些总结
Vue基础超详细
前端JS也可以连点成线(Vue中运用 AntVG6)
Vue事件处理的基本使用
Vue后台项目的记录 (一)
前后端分离vue跨域,devServer配置proxy代理
TypeScript
初识vuex
vue项目安装包指令收集
上一篇文章      下一篇文章      查看所有文章
加:2022-07-04 22:45:33  更:2022-07-04 22:45:56 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/11 10:55:24-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码