ES6-ES11新语法之ES6下篇

Set:

  ES6提供了新的数据结构Set(集合)。它类似数组,但成员的值都是唯一的,集合内部实现了 iterator(迭代器)接口,所以可以使用 扩展运算符 和 for...of 进行遍历

  集合的属性和方法:

    1、size:返回集合的元素个数

    2、add:增加一个新元素,返回当前集合

    3、delete:删除元素,返回Boolean值

    4、has:检测集合中是否包含某个元素,返回Boolean值

    5、clear:清空集合,返回undefined

利用Set进行数组去重,交集,并集,差集:

    let arr = [1, 2, 3, 4, 5, 4, 3, 2, 1, true]
    let newArr = [... new Set(arr)] // 数组去重
    console.log(newArr) // [1, 2, 3, 4, 5, true]

    let arr1 = [1, 2, 3, 1, 2, 3, false]

    let result = [...new Set(arr)].filter(item => new Set(arr1).has(item)) // 交集

    console.log(result) // [1, 2, 3]

    let union = [...new Set([...arr, ...arr1])] // 并集 

    console.log(union) // [1, 2, 3, 4, 5, true, false]

    let diff = [...new Set(arr)].filter(item => !(new Set(arr1).has(item))) // 差集,和交集相反,arr在前arr1在后意思是arr中有的arr1中没有

    console.log(diff) // [4, 5, true]

Map:

  ES6提供了Map数据结构。它类似于对象,也是键值对的集合。但是“键”的范围不限于字符串,各种类型的值(包括对象)都可以当做键。Map也实现了 iterator(迭代器)接口,所以可以使用 扩展运算符 和 fo...of 进行遍历

  Map的属性和方法:

    1、size:返回Map的元素个数

    2、set:增加一个新元素,返回当前Map

    3、get:返回键名对象的键值

    4、has:检测Map中是否包含某个元素,返回Boolean值

    5、clear:清空集合,返回undefined

class类:

  ES6提供了更接近传统语言的写法,引入了 Class(类)的概念,作为对象的模板。通过 class 关键字,可以定义类。基本上,ES6的 class 可以看做只是一个语法糖,它的绝大部分功能,ES5都能做到,新的 class 写法只是让对象的写法更加清晰、更面向对象编程

  知识点:

    1、class 声明类

    2、constructor 定义构造函数初始化

    3、extends 继承父类

    4、super 调用父级构造方法

    5、static 定义静态方法和属性

    6、父类方法可以重写

ES5和ES6实例化对象的区别:

    // ES5定义构造函数
    function Phone(brand, price) {
      this.brand = brand
      this.price = price
    }

    Phone.prototype.call = function () { console.log('打电话') }

    let xiaomi = new Phone('小米', 1999)

    console.log(xiaomi) // {brand: "小米", price: 1999}

    xiaomi.call() // 打电话

    // ES6定义构造函数
    class ShouJi {
      constructor(brand, price) {
        this.brand = brand
        this.price = price
      }
      call() {
        console.log('打电话呀')
      }
      static fn() { console.log('fn') }
    }

    let vivo = new ShouJi('vivo', 3000)

    console.log(vivo) // {brand: "vivo", price: 3000}

    vivo.call() // 打电话呀

    ShouJi.fn() // fn

类的静态成员只能由类(构造函数)去调用,不能由实例对象调用:

    // ES5
    function Phone() { }
    Phone.name = 'oppo手机'
    Phone.change = function () { console.log('改变oppo') }

    Phone.prototype.size = '5.5英寸'

    let oppo = new Phone
    console.log(oppo.name)
    console.log(Phone.name)
    // oppo.change() // 报错
    Phone.change() // 改变oppo
    console.log(oppo.size)

    // ES6
    class ShouJi {
      static name = 'vivo手机'
      static change() { console.log('改变vivo') }
    }

    let vivo = new ShouJi()
    console.log(vivo.name)
    console.log(ShouJi.name)
    // vivo.change() // 报错
    ShouJi.change() // 改变vivo

ES5的继承:

    function Phone(brand, price) {
      this.brand = brand
      this.price = price
      this.call = function(){ console.log('打电话') }
    }

    let p = new Phone('vivo', 1999)
    console.log(p)
    p.call()

    // 智能手机
    function SmartPhone(brand, price, color, size) {
      Phone.call(this, brand, price) // 通过调用构造函数实现继承
      // Phone.apply(this, [brand, price])
      this.color = color
      this.size = size
      this.photo = function () { console.log('拍照') }
      this.playGame = function () { console.log('游戏') }
    }

    SmartPhone.prototype = new Phone // 设置子级构造函数的原型
    SmartPhone.prototype.constructor = SmartPhone // 这句话不加没有影响

    let s = new SmartPhone('小米', '2000', 'red', '5.5')

    console.log(s)
    s.call()
    s.photo()
    s.playGame()

ES6的继承:通过 extends 关键字继承父类,在 constructor 中通过 super() 调用父类的构造方法

    class Phone {
      constructor(brand, price) {
        this.brand = brand
        this.price = price
      }
      call() { console.log('打电话') }
    }

    // 通过extends继承父类
    class SmartPhone extends Phone {
      constructor(brand, price, color, size) {
        super(brand, price) // 通过 super() 调用父类的构造方法,相当于ES5中 Phone.call(this, brand, price)
        this.color = color
        this.size = size
      }
      photo() { console.log('拍照') }
      playGame() { console.log('游戏') }
      call() { console.log('视频通话') } // 子类对父类方法的重写
    }

    let s = new SmartPhone('小米', 2999, 'white', '5.9英寸')

    console.log(s)
    s.call()
    s.photo()
    s.playGame()

class 的 get 和 set:

    class Phone {
      get price() {
        console.log('价格属性被读取了')
        return 1999
      }
      set price(newValue) {
        console.log('价格属性被修改了', newValue) // 价格属性被修改了 2999
      }
    }

    let p = new Phone()

    console.log(p.price) // 1999

    p.price = 2999

数值扩展:

    // Number.EPSILON 表示最小精度,两个数的大小小于这个精度的就认为是 ===
    function equal(a, b) {
      return Math.abs(a - b) < Number.EPSILON ? true : false
    }

    console.log(0.1 + 0.2 === 0.3) // false
    console.log(equal(0.1 + 0.2, 0.3)) // true

    // Number.isFinite 检测一个数据是否为有限数,返回Boolean值
    console.log(Number.isFinite(100)) // true
    console.log(Number.isFinite(100 / 0)) // false
    console.log(Number.isFinite(Infinity)) // false

    // Number.isNan 检测一个数是否为NaN,返回Boolean值
    console.log(Number.isNaN(100)) // false
    console.log(Number.isNaN(NaN)) // true
    console.log(Number.isNaN(undefined)) // false 因为 undefined !== NaN
    console.log(Number.isNaN(1 + undefined)) // true 因为 1 + undefined 的结果为 NaN
    // Number.isNaN()和isNaN()的区别:Number.isNaN不存在类型转换的行为,isNaN会通过Number方法,试图将所测参数转换成Number类型。Number.isNaN检测是不是全等于NaN,isNaN检测当前值转为Number类型后是不是一个数字
    console.log(isNaN(100)) // false
    console.log(isNaN(NaN)) // true
    console.log(isNaN(undefined)) // true 因为Number(undefined)的结果为NaN
    console.log(isNaN(1 + undefined)) // true 因为Number(1 + undefined)的结果为NaN

    // Number.isInteget 判断一个数是否为整数,返回Boolean值
    console.log(Number.isInteger(1))
    console.log(Number.isInteger(1.3))

    // Number.parseInt Number.parseFloat 将字符串转为数字取整或保留小数
    console.log(Number.parseInt('a100.33')) // NaN
    console.log(Number.parseInt('100.33aaa')) // 100
    console.log(Number.parseFloat('a100.33')) // NaN
    console.log(Number.parseFloat('100.33aaa你好')) // 100.33
    console.log(Number.parseInt(0.00000060)) // 6 因为6个及以上连续的0会自动转为科学计数法

    // Math.trunc 将数字的小数点部分抹掉
    console.log(Math.trunc('a100.33')) // NaN 因为无法解析字符串
    console.log(Math.trunc('100.33aaa')) // NaN 因为无法解析字符串
    console.log(Math.trunc(100.999)) // 100
    console.log(Math.trunc(0.00000060)) // 0

    // Math.sign 判断一个数为正数、负数还是零,分别返回1 0 -1
    console.log(Math.sign(100)) // 1
    console.log(Math.sign(0)) // 0
    console.log(Math.sign(-100)) // -1

对象方法扩展:

    // Object.is() 和===基本一致,但是在判断+0和-0、NaN和NaN时与===结果相反   js中==、===和Object.js()的区别
    console.log(Object.is(0, 0)) // true
    console.log(Object.is(+0, -0)) // false
    console.log(Object.is(NaN, NaN)) // true

    // Object.assign() 将两个对象中不同键名拿出来,相同键名的键值取后一个对象提供的键值,返回一个新的对象
    const obj = {
      host: 'localhost',
      port: '3000',
      name: 'root',
      pass: 'root',
      test: 'test'
    }
    const obj1 = {
      host: 'http://www.baidu.com',
      port: '4000',
      name: 'gem',
      pass: 'gen',
      test1: 'test1'
    }
    console.log(Object.assign(obj, obj1))

    // Object.keys() Object.values() Object.entries() Object.getOwnPropertyDescriptor() Object.fromEntries()
    console.log(Object.keys(obj)) // 获取键名,返回数组 ["host", "port", "name", "pass", "test"]
    console.log(Object.values(obj)) // 获取键值,返回数组 ["localhost", "3000", "root", "root", "test"]
    console.log(Object.entries(obj)) // 获取键值对,返回二维数组 [["host", "localhost"],["port", "3000"],["name", "root"],["pass", "root"],["test", "test"]]
    console.log(
      Object.getOwnPropertyDescriptor(obj, 'host')) // 返回某个属性的属性描述对象 {value: "localhost", writable: true, enumerable: true, configurable: true}

    let arr = [
      [1, 2],
      [3, 4]
    ]
    console.log(Object.fromEntries(arr)) // 将二维数组转为对象 {1: 2, 3: 4}

    // Object.setPrototypeOf() 设置原型对象    Object.getPrototypeOf() 获取原型对象
    const goddess = {
      name: '孙艺珍',
      age: 20
    }
    const country = {
      countryName: 'Korea',
      city: 'Seoul'
    }
    Object.setPrototypeOf(goddess, country)  // 不建议这么做,在创建对象时就要把原型上的属性设置好
    console.log(Object.getPrototypeOf(goddess)) // {countryName: "Korea", city: "Seoul"}
    console.log(goddess)

模块化:

  概念:将一个大的程序文件,拆分成一个个小的文件,然后将小文件组合起来

  优点:防止命名冲突,代码复用,高维护性

  ES6之前的模块化规范产品:

    CommonJS:nodeJS、Browserify

    AMD:requireJS

    CMD:seaJS

  ES6模块化语法:

    export:规定模块的对外接口

    import:输入其他模块提供的功能

    export和export default的区别

 

  export 的三种情况:

    1、分别暴露:(module1.js)

      export let name = '孙艺珍'

      export function add(a, b) {
        return a + b
      }

    

    2、统一暴露:(module2.js)

      const name = '小明'
      const age = 18
      function count(a, b) {
        return a - b
      }
      export { name, age, count }

    3、默认暴露,可以是任意数据类型,暴露对象居多:(module3.js)

      const name = '小明'
      const age = 18
      function count(a, b) {
        return a - b
      }
      export default {
        name,
        age,
        count
      }

  import 的三种情况:

    1、通过的导入方式

      import * as module1 from './js/module1.js' // module1对象中包括module1.js中的name属性和add方法
      import * as module2 from './js/module2.js' // module2对象中包括module1.js中的name、age属性和count方法
      import * as module3 from './js/module3.js' // module3.js中使用了export default导出,导出的对象在default对象中

    2、解构赋值形式

      import { name, add } from './js/module1.js' // 解构时参数名和module1.js中导出的变量名一致
      import { name as name1, age, count } from './js/module2.js' // as name1是给name属性起一个别名,场景:当module1.js和module2.js中都导出了name属性,name在同一个页面中引入时都需要用参数name接收,这里就需要定义别名以防冲突
      import { default as module3 } from './js/module3.js' // 将default中的对象重命名为module3
    如果不使用别名会报错:Uncaught SyntaxError: Identifier 'name' has already been declared

    3、简便形式,仅适用于export default方式导出模块的接收

      import module33 from './js/module3.js' // import { default as module3 } from './js/module3.js' 的简写

  利用 babel 将 ES6 转为 ES5:

      1、下载插件:npm i babel-cli babel-preset-env browserify -D
        babel-cli:babel命令行的工具
        babel-preset-env:预设包,将ES6转为ES5
        browserify:打包工具,实际项目中用的是webpack
      
      2、转换:根目录终端下执行 npx babel js -d dist/js --presets=babel-preset-env
        npx:babel局部安装用npx启动,babel如果是全局安装可以直接用babel启动
        js:第一个参数,需要转换为ES5的目录
        -d:输出选项,将转换结果存到哪个目录下
        dist/js:存放转换后结果的目录
        --presets=babel-preset-env:配置项传参,省了写 .babelrc 文件

      3、打包:npx browserify dist/js/app.js -o dist/main.js
        dist/js/app.js:第一个参数,入口文件
        -o:输出选项,将转换结果存到哪个目录下
        dist/main.js:输出到dist目录下新建 main.js 文件

      4、页面中使用 <script src="./dist/main.js"></script> 可以进行引入,此时浏览器不管支不支持ES6都可以识别,修改app.js中文件后需要重新转换和打包

  ES6模块化引入npm包:

    1、先下载这个包,npm i jquery

    2、入口文件app.js中引入:

      import $ from 'jquery' // 和commonJS中的 const $ = require('jquery') 一样

      $('body').css('backgroundColor', 'deeppink')

 

 

 

x

posted @ 2020-11-27 10:34  吴小明-  阅读(354)  评论(0编辑  收藏  举报