ES6-ES11新语法之ES6上篇

let:

  let声明变量的特点,和var有什么不同:

    1、变量不能重复声明

    2、块级作用域

    3、不存在变量提升

    4、不影响作用域链,同一个块级作用域下,声明在函数前的变量可以被函数访问到

  let小案例:

const:

  const的特点:

    1、一定要赋初始值

    2、一般常量使用大写(不是必须)

    3、常量的值不能修改

    4、块级作用域

    5、对于数组和对象的元素的修改,地址没有改变,不会报错。const定义的变量引用不能修改,值可以修改   const和Object.freeze()搭配使用

解构赋值:

    const arr = [100, 200, 300, 400]
    let [a, b, c, d] = arr
    console.log(a, b, c, d) // 100 200 300 400

    const obj = {
      name: '孙艺珍',
      age: 20,
      doo: function () {
        console.log('演电影')
      }
    }
    let { name, age, doo } = obj
    console.log(name)
    console.log(age)
    console.log(doo)
    doo()

  对象中的方法频繁调用想到可以使用结构赋值拿出该方法,调用就不用写对象名了

模板字符串:

  1、可以直接出现换行符,es5的字符串要用+拼接

  2、变量拼接

    let str = `模板字符串`
    console.log(str)

    let num = 100
    console.log(`我今年${num}岁了`)

    let mo = `
      <ul>
        <li>100</li>
        <li>100</li>
        <li>100</li>
        <li>100</li>
        <li>100</li>
      </ul>
    `
    console.log(mo)
    document.write(mo)

对象简化写法:

    let name = `"'syz'"`
    let doo = function () {
      console.log('doo')
    }
    let person = {
      name, // name: name
      doo, // doo: doo
      improve() { // 省略function关键字
        console.log('让你变漂亮')
      }
    }
    console.log(person)
    person.doo()
    person.improve()

箭头函数:

  1、this是静态的,始终指向函数声明时所在作用域下的this值

  2、不能作为构造函数实例化对象

  3、不能使用arguments变量

  4、简写    1)省略小括号:有且只有一个形参时    2)省略花括号:代码体有且只有一条语句时,此时语句的执行结果就是函数的返回值,return也必须要省略

    function fn() {
      console.log(this.name)
    }

    // 1、this是静态的,this始终指向函数声明时所在作用域下的this的值
    let fn1 = () => {
      console.log(this.name)
    }

    window.name = '孙艺珍'

    fn() // 孙艺珍
    fn1() // 孙艺珍

    let person = {
      name: '吴小明'
    }

    fn.call(person) // 吴小明
    fn1.call(person) // 孙艺珍 this始终指向函数声明时所在作用域下的this的值

    // 2、不能作为构造函数实例化对象
    let Person = (name, age) => {
      this.name = name
      this.age = age
    }

    // let p = new Person('xx', 20) // Uncaught TypeError: Person is not a constructor

    // 3、不能使用arguments变量
    function xx() {
      console.log(arguments)
    }
    xx(1, 2, 3) // Arguments(3) [1, 2, 3, callee: ƒ, Symbol(Symbol.iterator): ƒ]

    let yy = () => {
      console.log(arguments)
    }

    // yy(1, 2, 3) // Uncaught ReferenceError: arguments is not defined

    // 4、简写       省略小括号:有且只有一个形参时可以省略      省略花括号:当代码体有且只有一条语句时可以省略,此时语句执行的结果就是函数的返回值,return语句也必须要省略
    let add = n => {
      return n + n
    }
    console.log(add(4)) // 8
    let add1 = n => n + n
    console.log(add1(5)) // 10

  使用场景:

    1、this指向上一级的this,函数嵌套时,里层函数中使用外层函数的this,此时可以使用箭头函数。如果使用es5的函数,第一种方法是在里层函数的上方定义变量 let _this = this 保存外层函数的this;第二种方法是调用时利用call()或apply(),如:xxx.call(this)

    2、数组的方法回调中使用箭头函数

    // 返回数组中偶数项
    let arr = [1, 2, 3, 4, 5, 6]

    const even = arr.filter(function (item) {
      return item % 2 === 0
    })

    console.log(even)

    const even1 = arr.filter(item => item % 2 === 0)

    console.log(even1)

  总结:

    1、箭头函数适合与this无关的回调,定时器,数组的方法回调

    2、箭头函数不适合与this有关的回调,事件回调,对象的方法

参数默认值:

  1、形参初始值:具有默认值的参数----一般有初始值的参数位置放在后面,如果将参数c的位置放在前面或中间此时的值是NaN,因为b的值是undefined

    function add(a, b, c = 10) {
      return a + b + c
    }

    console.log(add(1, 2))

  2、和解构赋值结合

    function connect({ host = '127.0.0.1', username, password, port }) {
      console.log(host)
      console.log(username)
      console.log(password)
      console.log(port)
    }

    connect({
      host: 'baidu.com',
      username: 'root',
      password: 'root1',
      port: 3000
    })

rest参数:

    // ES5获取实参的方式:arguments
    function fn() {
      console.log(arguments)
    }

    fn(100, '200', true) // Arguments(3) [100, "200", true, callee: ƒ, Symbol(Symbol.iterator): ƒ]

    // ES6引入 rest 参数,用于获取实参,代替arguments
    function fn1(...rest) {
      console.log(rest)
    }

    fn1(100, '200', true) // [100, "200", true]

    // rest 参数必须放到参数最后
    function fn2(a, b, ...rest) {
      console.log(a, b, rest)
    }

    fn2(100, '200', true, {}) // 100 "200" (2) [true, {…}]

注意:arguments是一个类数组,rest是一个数组

扩展运算符:

    const arr=['孙艺珍','吴小明','郭德纲']

    function fn(...rest){
      console.log(rest)
    }
    // 扩展运算符能将数组转换为逗号分隔的参数序列
    // fn(...arr) // ["孙艺珍", "吴小明", "郭德纲"]
    fn('孙艺珍','吴小明','郭德纲') // ["孙艺珍", "吴小明", "郭德纲"]

...的使用:

    // 数组的合并
    const arr = ['唐僧', '八戒']
    const arr1 = ['沙悟净', '悟空']

    const newArr = arr.concat(arr1) // ["唐僧", "八戒", "沙悟净", "悟空"]
    const newArr1 = [...arr, ...arr1] // ["唐僧", "八戒", "沙悟净", "悟空"]

    console.log(newArr)
    console.log(newArr1)

    // 数组的拷贝(浅拷贝)
    const copyArr = [...arr]
    console.log(copyArr) // ["唐僧", "八戒"]

    // 将伪数组转为真正的数组
    const divs = document.querySelectorAll('div')
    const divArr = [...divs]
    console.log(divs) // NodeList(3) [div, div, div]
    console.log(divArr) // [div, div, div]

Symbol:

    /*
      Symbol:
        ES6新引入的简单数据类型,表示独一无二的值。它是JavaScript语言的第七种数据类型,是一种类似于字符串的数据类型。
      
      Symbol特点:
        1、Symbol的值是唯一的,用来解决命名冲突的问题
        2、Symbol值不能与其他数据进行运算
        3、Symbol定义的对象属性不能使用 for...in 循环病遍历,但是可以使用Reflect.ownKeys 来获取对象的所有键名
    
    */

    // 创建Symbol
    let s1 = Symbol()
    console.log(s1, typeof s1) // Symbol() "symbol"

    let s2 = Symbol('孙艺珍')
    let s3 = Symbol('孙艺珍')

    console.log(s2) // Symbol(孙艺珍)
    console.log(s3) // Symbol(孙艺珍)
    console.log(s2 == s3) // false

    // Symbol.for() 创建
    let s4 = Symbol.for('小明')
    let s5 = Symbol.for('小明')

    console.log(s4) // Symbol(小明)
    console.log(s5) // Symbol(小明)
    console.log(s4 === s5) // true

Symbol给对象中添加属性和方法的两种方式:

    // 1、利用Symbol给对象中添加方法
    let game = {
      name: '俄罗斯方块',
      up: function () { console.log('我是up') },
      down: function () { console.log('我是down') }
    }

    let methods = {
      up: Symbol(),
      down: Symbol()
    }
    // 如果直接写 game.up=function(){} 可能会替换掉 game 中原有的方法 
    game[methods.up] = function () { console.log('upupupupup') }
    game[methods.down] = function () { console.log('downdown') }

    console.log(game)

    game.up()

    // 2、Symbol() 不能直接作为键使用,可以写成 [Symbol('xxx')]
    let youxi = {
      name: '狼人杀',
      [Symbol('say')]: function () { console.log('我可以说话') },
      [Symbol('zibao')]: function () { console.log('我可以自爆') }
    }
    console.log(youxi)

Symbol内置值:

Symbol.hasInstance 当其他对象使用instanceof运算符,判断是否为该对象的实例时,会调用这个方法
Symbol.isConcatSpreadable 对象的Symbol.isConcatSpreadable属性等于的是一个布尔值,表示该对象用于Array.prototype.concat()时,是否可以展开
Symbol.species 创建衍生对象时,会使用该属性
Symbol.match 当执行str.match(myObject)时,如果该属性存在,会调用它,返回该方法的返回值
Symbol.replace 当该对象被str.replace(myObject)方法调用时,会返回该方法的返回值
Symbol.search 当该对象被str.search(myObject)方法调用时,会返回该方法的返回值
Symbol.split 当该对象被str.split(myObject)方法调用时,会返回该方法的返回值
Symbol.iterator 对象进行for...of循环时,会调用Symbol.iterator方法,返回该对象的默认遍历器
Symbol.toPrimitive 该对象被转为原始类型的值时,会调用这个方法,返回该对象对应的原始类型值
Symbol. toStringTag 在该对象上面调用toString方法时,返回该方法的返回值
Symbol. unscopables 该对象指定了使用with关键字时,哪些属性会被with环境排除
    // Symbol.hasInstance
    class Person {
      static [Symbol.hasInstance](param) {
        console.log(param) // {name: "111"}
        console.log('我被用来检测数据类型了')
        return true
      }
    }

    let obj = { name: '111' }
    let result = obj instanceof Person // true

    console.log(result)

    // Symbol.isConcatSpreadable
    let arr = [1, 2, 3]
    let arr1 = [4, 5, 6]
    arr1[Symbol.isConcatSpreadable] = false // 设置不可以展开
    let newArr = arr.concat(arr1)
    console.log(newArr) // [1, 2, 3, Array(3)]
    console.log(newArr[3]) // [4, 5, 6, Symbol(Symbol.isConcatSpreadable): false]

注意:遇到唯一性的场景时要想到Symbol

迭代器:

  遍历器(Iterator)就是一种机制。它是一种接口,为各种不同的数据结构提供统一的访问机制。任何数据只要部署 Iterator 接口(对象中的一个属性),就可以完成遍历操作。

  原生具备 Iterator 接口的数据(可用for...of遍历):Array、Arguments、Set、Map、String、TypedArray、NodeList

  工作原理:

    1、创建一个指针对象,指向当前数据结构的起始位置

    2、第一次调用对象的 next 方法,指针自动指向数据结构的第一个成员

    3、接下来不断调用 next 方法,指针一直往后移动,直到指向最后一个成员

    4、每调用 next 方法返回一个包含 value 和 done 属性的对象

    // for...of 遍历值,for...in 遍历键
    let str = '123456789'

    for (item of str) {
      console.log(item)
    }

    let arr = [1, 2, 3, 4, 5]
    for (item of arr) {
      console.log(item)
    }

    let iterator = arr[Symbol.iterator]()

    console.log(iterator)
    console.log(iterator.next()) // {value: 1, done: false}
    console.log(iterator.next()) // {value: 2, done: false}
    console.log(iterator.next()) // {value: 3, done: false}
    console.log(iterator.next()) // {value: 4, done: false}
    console.log(iterator.next()) // {value: 5, done: false}
    console.log(iterator.next()) // {value: undefined, done: true}

自定义遍历数据:

    const banji = {
      name: '重点班',
      students: ['小明', '小红', '小芳', '小华'],
      // 自定义iterator接口函数
      [Symbol.iterator]() {
        let index = 0
        let _this = this
        return {
          next: function () {
            if (index < _this.students.length) {
              const result = { value: _this.students[index], done: false }
              index++
              return result
            } else {
              return { value: undefined, done: true }
            }
          }
        }
      }
    }
    for (let item of banji) {
      console.log(item)
    }

注意:需要自定义遍历数据的时候,要想到迭代器

生成器:

  生成器函数是ES6提供的一种异步编程方案,语法行为与传统函数完全不同

  特点:

    1、* 的位置没有限制

    2、生成器函数返回的结果是迭代器对象,调用迭代器对象的 next 方法可以得到 yield 语句后的值

    3、yield 相当于函数的暂停标记,也可以认为是函数的分隔符,每调用一次 next 方法,执行一段代码

    4、next 方法可以传递实参,作为 yield 语句的返回值

    function * fn() {
      console.log(111)
      yield '你好呀'
      console.log(222)
      yield '我是安徽人'
      console.log(333)
      yield '你是哪里人'
      console.log(444)
    }

    let iterator = fn() // 生成迭代器对象
    console.log(iterator.next()) // {value: "你好呀", done: false}
    console.log(iterator.next()) // {value: "我是安徽人", done: false}
    console.log(iterator.next()) // {value: "你是哪里人", done: false}
    console.log(iterator.next()) // {value: undefined, done: true}

    // 遍历
    for(let item of fn()){
      console.log(item) // 返回结果为 yield 表达式后的内容
    }

生成器函数参数:

    function* gen(arg) {
      console.log(arg) // AAA
      let one = yield 111
      console.log(one) // BBB 调用 next() 时传入的参数作为 yield 语句的返回值----第二次调用 next() 传入的参数作为第一个 yield 语句的返回值
      let two = yield 222
      console.log(two) // CCC
      let three = yield 333
      console.log(three) // DDD
      let four = yield 444
      console.log(four) // undefined
    }
    let iterator = gen('AAA') // 获取迭代器对象
    console.log(iterator.next()) // {value: 111, done: false}
    console.log(iterator.next('BBB')) // {value: 222, done: false}
    console.log(iterator.next("CCC")) // {value: 333, done: false}
    console.log(iterator.next('DDD')) // {value: 444, done: false}
    console.log(iterator.next()) // {value: undefined, done: true}

小案例:

  1、异步编程:1秒后输出111,2秒后输出222,3秒后输出333

    // 第一种:回调地狱问题
    setTimeout(() => {
      console.log(111)
      setTimeout(() => {
        console.log(222)
        setTimeout(() => {
          console.log(333)
        }, 3000);
      }, 2000);
    }, 1000);

    // 第二种:利用生成器函数,获取迭代器对象,通过调用迭代器的 next() 方法一步步执行one two three
    function one() {
      setTimeout(() => {
        console.log(111)
        iterator.next()
      }, 1000);
    }

    function two() {
      setTimeout(() => {
        console.log(222)
        iterator.next()
      }, 2000);
    }

    function three() {
      setTimeout(() => {
        console.log(333)
        iterator.next()
      }, 3000);
    }

    function* gen() {
      yield one()
      yield two()
      yield three()
    }
    let iterator = gen() // 调用生成器函数,获取迭代器对象
    iterator.next()

  2、场景:请求接口当前接口需要依赖于上一次请求返回的数据,生成器函数是异步的,此处可以替代回调函数,解决回调地狱问题

    // 先拿到用户数据,再拿到订单数据,再拿到商品数据
    function getUsers() {
      setTimeout(() => {
        let data = '用户数据'
        iterator.next(data)
      }, 1000);
    }
    function getOrders() {
      setTimeout(() => {
        let data = '订单数据'
        iterator.next(data)
      }, 1000);
    }
    function getGoods() {
      setTimeout(() => {
        let data = '商品数据'
        iterator.next(data)
      }, 1000);
    }

    function* gen() {
      let users = yield getUsers()
      console.log(users)
      let orders = yield getOrders()
      console.log(orders)
      let goods = yield getGoods()
      console.log(goods)
    }

    let iterator = gen()
    iterator.next()

Promise:

  Promise是ES6引入的异步编程的新解决方案。语法上Promise是一个构造函数,用来封装异步操作并可以获取其成功或失败的结果

Promise基本语法:

    const p = new Promise(function (resolve, reject) {
      setTimeout(() => {
        let data = '数据'
        // resolve(data)
        reject('失败')
      }, 100);
    })

    p.then(function (value) {
      console.log(value)
    }).catch(function (err) {
      console.log(err)
    })

Promise封装读取文件:

const fs = require('fs')

// 1、直接读取
// fs.readFile('./resources/插秧诗.md', (err, data) => {
//   if (err) throw err
//   console.log(data.toString())
// })

// 2、使用Promise封装
const p = new Promise(function (resolve, reject) {
  fs.readFile('./resources/为学.md', (err, data) => {
    if (err) reject(err)
    resolve(data)
  })
})

p.then(function (value) {
  console.log(value.toString())
}).catch(function (err) {
  console.log(err)
})

Promise封装ajax:

    // 1、原生ajax
    const xhr = new XMLHttpRequest()
    xhr.addEventListener('load', loadHandler) // Firefox 支持 load 事件
    xhr.open('get', 'https://api.apiopen.top/getJoke')
    xhr.send()
    function loadHandler() {
      if (xhr.readyState === 4 && xhr.status === 200) {
        // console.log(xhr.response)
      } else {
        console.error(xhr.status)
      }
    }

    // 2、Promise封装
    const p = new Promise(function (resolve, reject) {
      const xhr1 = new XMLHttpRequest()
      xhr1.addEventListener('load', loadHandler1)
      xhr1.open('get', 'https://api.apiopen.top/getJoke')
      xhr1.send()
      function loadHandler1() {
        if (xhr1.readyState === 4 && xhr1.status === 200) {
          resolve(xhr1.response)
        } else {
          reject(xhr1.status)
        }
      }
    })

    p.then(function (value) {
      console.log(value)
    }).catch(function (err) {
      console.error(err)
    })

Promise读取多个文件,避免回调地狱:

const fs = require('fs')

// 1、使用回调,会有回调地狱问题
// fs.readFile('./resources/为学.md', (err, data) => {
//   fs.readFile('./resources/插秧诗.md', (err, data1) => {
//     fs.readFile('./resources/观书有感.md', (err, data2) => {
//       const result = data + '\r\n' + data1 + '\r\n' + data2
//       console.log(result)
//     })
//   })
// })

// 2、使用 Promise 实现
const p = new Promise((resolve, reject) => {
  fs.readFile('./resources/为学.md', (err, data) => {
    resolve(data)
  })
})

p.then((value) => {
  return new Promise((resolve, reject) => {
    fs.readFile('./resources/插秧诗.md', (err, data) => {
      resolve([value, data])
    })
  })
})
  .then((value) => {
    return new Promise((resolve, reject) => {
      fs.readFile('./resources/观书有感.md', (err, data) => {
        value.push(data)
        resolve(value)
      })
    })
  })
  .then((value) => {
    console.log(value.join('\r\n'))
  })

 

  

 

 

 

 

 

x

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