手写promise

05-promise-api.js



var fs = require('fs')

var p1 = new Promise(function (resolve, reject) {
  fs.readFile('./data/a.txt', 'utf8', function (err, data) {
    if (err) {
      reject(err)
    } else {
      resolve(data)
    }
  })
})

var p2 = new Promise(function (resolve, reject) {
  fs.readFile('./data/b.txt', 'utf8', function (err, data) {
    if (err) {
      reject(err)
    } else {
      resolve(data)
    }
  })
})

var p3 = new Promise(function (resolve, reject) {
  fs.readFile('./data/c.txt', 'utf8', function (err, data) {
    if (err) {
      reject(err)
    } else {
      resolve(data)
    }
  })
})

p1
  .then(function (data) {
    console.log(data)
    // 当 p1 读取成功的时候
    // 当前函数中 return 的结果就可以在后面的 then 中 function 接收到
    // 当你 return 123 后面就接收到 123
    //      return 'hello' 后面就接收到 'hello'
    //      没有 return 后面收到的就是 undefined
    // 上面那些 return 的数据没什么卵用
    // 真正有用的是:我们可以 return 一个 Promise 对象
    // 当 return 一个 Promise 对象的时候,后续的 then 中的 方法的第一个参数会作为 p2 的 resolve
    // 
    return p2
  }, function (err) {
    console.log('读取文件失败了', err)
  })
  .then(function (data) {
    console.log(data)
    return p3
  })
  .then(function (data) {
    console.log(data)
    console.log('end')
  })

06-封装promise-API.js



var fs = require('fs')

function pReadFile(filePath) {
  return new Promise(function (resolve, reject) {
    fs.readFile(filePath, 'utf8', function (err, data) {
      if (err) {
        reject(err)
      } else {
        resolve(data)
      }
    })
  })
}

pReadFile('./data/a.txt')
  .then(function (data) {
    console.log(data)
    return pReadFile('./data/b.txt')
  })
  .then(function (data) {
    console.log(data)
    return pReadFile('./data/c.txt')
  })
  .then(function (data) {
    console.log(data)
  })

07-promise使用场景.html

<!DOCTYPE html>
<html lang="en">

<head>
  <meta charset="UTF-8">
  <title>Document</title>
</head>

<body>
  <form action="00-js中的一等公民函数.js" id="user_form">
  </form>
  <script type="text/template" id="tpl">
    <div>
      <label for="">用户名</label>
      <input type="text" value="{{ user.username }}">
    </div>
    <div>
      <label for="">年龄</label>
      <input type="text" value="{{ user.age }}">
    </div>
    <div>
      <label for="">职业</label>
      <select name="" id="">
        {{ each jobs }} {{ if user.job === $value.id }}
        <option value="{{ $value.id }}" selected>{{ $value.name }}</option>
        {{ else }}
        <option value="{{ $value.id }}">{{ $value.name }}</option>
        {{ /if }} {{ /each }}
      </select>
    </div>
  </script>
  <script src="node_modules/art-template/lib/template-web.js"></script>
  <script src="node_modules/jquery/dist/jquery.js"></script>
  <script>
    // 用户表
    //  其中一个接口获取用户数据
    //  职业:2
    // 职业信息表
    //  其中一个接口获取所有的职业信息
    // get('http://127.0.0.1:3000/users/4', function (userData) {
    //   get('http://127.0.0.1:3000/jobs', function (jobsData) {
    //     var htmlStr = template('tpl', {
    //       user: JSON.parse(userData),
    //       jobs: JSON.parse(jobsData)
    //     })
    //     console.log(htmlStr)
    //     document.querySelector('#user_form').innerHTML = htmlStr
    //   })
    // })

    // var data = {}
    // $.get('http://127.0.0.1:3000/users/4')
    //   .then(function (user) {
    //     data.user = user
    //     return $.get('http://127.0.0.1:3000/jobs')
    //   })
    //   .then(function (jobs) {
    //     data.jobs = jobs
    //     var htmlStr = template('tpl', data)
    //     document.querySelector('#user_form').innerHTML = htmlStr
    //   })

    // var data = {}
    // pGet('http://127.0.0.1:3000/users/4')
    //   .then(function (user) {
    //     data.user = user
    //     return pGet('http://127.0.0.1:3000/jobs')
    //   })
    //   .then(function (jobs) {
    //     data.jobs = jobs
    //     var htmlStr = template('tpl', data)
    //     document.querySelector('#user_form').innerHTML = htmlStr
    //   })

    // pGet('http://127.0.0.1:3000/users/4', function (data) {
    //   console.log(data)
    // })

    pGet('http://127.0.0.1:3000/users/4')
      .then(function (data) {
        console.log(data)
      })

    function pGet(url, callback) {
      return new Promise(function (resolve, reject) {
        var oReq = new XMLHttpRequest()
        // 当请求加载成功之后要调用指定的函数
        oReq.onload = function () {
          // 我现在需要得到这里的 oReq.responseText
          callback && callback(JSON.parse(oReq.responseText))
          resolve(JSON.parse(oReq.responseText))
        }
        oReq.onerror = function (err) {
          reject(err)
        }
        oReq.open("get", url, true)
        oReq.send()
      })
    }


    // 这个 get 是 callback 方式的 API
    // 可以使用 Promise 来解决这个问题
    function get(url, callback) {
      var oReq = new XMLHttpRequest()
      // 当请求加载成功之后要调用指定的函数
      oReq.onload = function () {
        // 我现在需要得到这里的 oReq.responseText
        callback(oReq.responseText)
      }
      oReq.open("get", url, true)
      oReq.send()
    }
  </script>
</body>

</html>

08-mongoose所有的API都支持promise.js

var mongoose = require('mongoose')

var Schema = mongoose.Schema

// 1. 连接数据库
// 指定连接的数据库不需要存在,当你插入第一条数据之后就会自动被创建出来
mongoose.connect('mongodb://localhost/itcast')

// 2. 设计文档结构(表结构)
// 字段名称就是表结构中的属性名称
// 约束的目的是为了保证数据的完整性,不要有脏数据
var userSchema = new Schema({
  username: {
    type: String,
    required: true // 必须有
  },
  password: {
    type: String,
    required: true
  },
  email: {
    type: String
  }
})

// 3. 将文档结构发布为模型
//    mongoose.model 方法就是用来将一个架构发布为 model
//    第一个参数:传入一个大写名词单数字符串用来表示你的数据库名称
//                 mongoose 会自动将大写名词的字符串生成 小写复数 的集合名称
//                 例如这里的 User 最终会变为 users 集合名称
//    第二个参数:架构 Schema
//   
//    返回值:模型构造函数
var User = mongoose.model('User', userSchema)


// 4. 当我们有了模型构造函数之后,就可以使用这个构造函数对 users 集合中的数据为所欲为了(增删改查)
// **********************
// #region /新增数据
// **********************
// var admin = new User({
//   username: 'zs',
//   password: '123456',
//   email: 'admin@admin.com'
// })

// admin.save(function (err, ret) {
//   if (err) {
//     console.log('保存失败')
//   } else {
//     console.log('保存成功')
//     console.log(ret)
//   }
// })
// **********************
// #endregion /新增数据
// **********************




// **********************
// #region /查询数据
// **********************
// User.find(function (err, ret) {
//   if (err) {
//     console.log('查询失败')
//   } else {
//     console.log(ret)
//   }
// })

// 用户注册
// 1. 判断用户是否存在
//    如果已存在,结束注册
//    如果不存在,注册(保存一条用户信息)
User.find()
  .then(function (data) {
    console.log(data)
  })

// User.findOne({ username: 'aaa' }, function (user) {
//   if (user) {
//     console.log('已存在')
//   } else {
//     new User({
//       username: 'aaa',
//       password: '123',
//       email: 'dsadas'
//     }).save(function () {
      
//     })
//   }
// })

// User.findOne({
//     username: 'aaa'
//   })
//   .then(function (user) {
//     if (user) {
//       // 用户已存在,不能注册
//       console.log('用户已存在')
//     } else {
//       // 用户不存在,可以注册
//       return new User({
//         username: 'aaa',
//         password: '123',
//         email: 'dsadas'
//       }).save()
//     }
//   })
//   .then(function (ret) {
//   })

// User.find({
//   username: 'zs'
// }, function (err, ret) {
//   if (err) {
//     console.log('查询失败')
//   } else {
//     console.log(ret)
//   }
// })

// User.findOne({
//   username: 'zs'
// }, function (err, ret) {
//   if (err) {
//     console.log('查询失败')
//   } else {
//     console.log(ret)
//   }
// })
// **********************
// #endregion /查询数据
// **********************



// **********************
// #region /删除数据
// **********************
// User.remove({
//   username: 'zs'
// }, function (err, ret) {
//   if (err) {
//     console.log('删除失败')
//   } else {
//     console.log('删除成功')
//     console.log(ret)
//   }
// })
// **********************
// #endregion /删除数据
// **********************


// **********************
// #region /更新数据
// **********************
// User.findByIdAndUpdate('5a001b23d219eb00c8581184', {
//   password: '123'
// }, function (err, ret) {
//   if (err) {
//     console.log('更新失败')
//   } else {
//     console.log('更新成功')
//   }
// })
// **********************
// #endregion /更新数据
// **********************


手写一个 promise

一个最简单的 promise 构造函数,接收一个函数并执行,给函数传递一个成功的回调函数,成功之后执行回调函数

function Promise(fn){
    var callback = null
    this.then = function(onResolved){
        callback = onResolved
    }
    function resolve(value){
        callback(value)
    }
    fn(resolve)
}

上面的代码有问题,因为在 Promise 实例化的时候,callback 此时还是 null,可以用 setTimeout 来解决这个问题

function Promise(fn){
    var callback = null
    this.then = function(onResolved){
        callback = onResolved
    }
    function resolve(value){
        setTimeout(function(){
            callback(value)
        }, 0)
        
    }
    fn(resolve)
}

试试调用这个 Promise

// 没有 .then 绑定 callback,直接 resolve 都会出错
var p = new Promise(function(cb){
    cb(1) // callback 此时还是 null
})

var p = new Promise(function(cb){
    setTimeout(function(){
     cb(2) callback 此时还是 null
    }, 0)
})
// 先生成 promise 实例,然后 .then 绑定回调函数
function getUserId(){
    return new Promise(function(cb){
        setTimeout(function(){
            cb(3)
        }, 0)
    })
}
function showUserId(id){
    console.log(id)
}
getUserId().then(showUserId)
给 Promise 实例添加多个回调

function Promise(fn){
    var callbacks = []
    this.then = function(onResolved){
        callbacks.push(onResolved)
    }
    function resolve(value){
        setTimeout(function(){
            callbacks.forEach(function(callback){
                callback(value)
            })
        }, 0)
        
    }
    fn(resolve)
}

试试多个回调

function showUserIdAgain(id){
    console.log('Again' + id)
}
getUserId().then(showUserId)
getUserId().then(showUserIdAgain)

一般情况下我们会同步调用 .then 方法的时候,但是有的时候会异步调用 .then 方法。比如买泡面和烧开水,先去买泡面,买好回家水已经烧开了,这时再吃泡面。

function boilWater(){ // 烧开水
    return new Promise(function(resolve){
        setTimeout(function(){
            resolve('开水')
        }, 1000)
    })
}

function buyNoodle(){ // 买泡面
    return new Promise(function(resolve){
        setTimeout(function(){
            resolve('泡面')
        }, 2000)
    })
}
 var promiseOfWater = boilWater()

 buyNoodle.then(function(noodle){
     promiseOfWater.then(function(water){
         console.log(noodel + '买好了' + water + '也烧开了')
     })
 })

要让 promise 在调用 resolve 之后,还能执行通过 then 添加的回调方法,需要用一个标记区分 promise 状态,如果 promise 还是 pending 状态,那么把 callback 放进回调数组,如果 promise 已经是 fullfilled 状态,直接调用 callback 方法

function Promise(fn){
    var callbacks = []
    var state = 'pending'
    var value = null
    this.then = function(onResolved){
        if(state === 'pending'){
            callbacks.push(onResolved)
        }else{
            onResolved(value)
        }
        return this
        
    }
    function resolve(newValue){
        value = newValue
        state = 'fullfilled'
        setTimeout(function(){
            callbacks.forEach(function(callback){
                callback(value)
            })
        }, 0)
        
    }
    fn(resolve)
}

实现 .then 的链式调用,假设吃泡面是这样的

var promiseOfWater = boilWater()

boilWater().then(makeNoodle).then(eat)

.then 要返回 promise 实例,这个promise 实例要做的事情是:

  1. 在上一个 promise 执行 resolve 之后,先调用 onResolved 回调;
  2. 把得到的结果作为参数放到自身的 then 回调中执行
function Promise(fn){
    var handlers = []
    var state = 'pending'
    var value = null
    
    this.then = function(onResolved){
        return new Promise(function(resolve){
            handle({
                onResolved,
                resolve,
            })
        })
    }
    function handle(handler){
        if(state === 'pending'){
            handlers.push(handler)
        }else{
            if(handler.onResolved){
                var res = handler.onResolved(value)
                handler.resolve(res)
            }else{
                handler.resolve(value)
            }
        }
    }
    function resolve(newValue){
        value = newValue
        state = 'fullfilled'
        setTimeout(function(){
            handlers.forEach(function(handler){
                handle(handler)
            })
        }, 0)
    }
    fn(resolve)
}

在 promise 构造函数的 resolve 方法里要判断参数是不是 promise 实例,如果是的话,就先调用实例的 then 方法

function Promise(fn){
    var handlers = []
    var state = 'pending'
    var value = null
    
    this.then = function(onResolved){
        return new Promise(function(resolve){ // 返回 promise 实例,方便链式调用
            handle({
                onResolved,
                resolve,
            })
        })
    }
    function handle(handler){
        if(state === 'pending'){
            handlers.push(handler)
        }else{
            if(handler.onResolved){
                var res = handler.onResolved(value)
                handler.resolve(res)
            }else{
                handler.resolve(value)
            }
        }
    }
    // 如果 newValue 是 promise 实例,直接调用 .then 方法
    function resolve(newValue){
        if(typeof newValue === 'object' && typeof newValue.then === 'function'){
            // resolve 就是当前这个定义函数
            newValue.then(resolve)
        }else{
            value = newValue
            state = 'fullfilled'
            setTimeout(function(){
                handlers.forEach(function(handler){
                    handle(handler)
                })
            }, 0)
        }
        
    }
    fn(resolve)
}

处理 reject 的情况

function Promise(fn){
    var handlers = []
    var state = 'pending'
    var value = null
    
    this.then = function(onResolved, onRejected){
        return new Promise(function(resolve, reject){ // 返回 promise 实例,方便链式调用
            handle({
                onResolved,
                resolve,
                onRejected,
                reject,
            })
        })
    }
    function handle(handler){
        if(state === 'pending'){
            handlers.push(handler)
        }else if(state === 'fullfilled'){
            if(handler.onResolved){
                var res = handler.onResolved(value)
                handler.resolve(res)
            }else{
                handler.resolve(value)
            }
        }else{
            if(handler.onRejected){
                var res = handler.onRejected(value)
                // 因为前一个 promise 中有处理 reject 的情况
                // 所以直接调用下一个 promise 的 resolve
                handler.resolve(res)
            }else{
                handler.reject(value)
            }
        }
    }
    // 如果 newValue 是 promise 实例,直接调用 .then 方法
    function resolve(newValue){
        if(typeof newValue === 'object' && typeof newValue.then === 'function'){
            // resolve 就是当前这个定义函数
            newValue.then(resolve, reject)
        }else{
            value = newValue
            state = 'fullfilled'
            setTimeout(function(){
                handlers.forEach(function(handler){
                    handle(handler)
                })
            }, 0)
        }
        
    }
    function reject(reason){
        value = reason
        state = 'rejected'
        setTimeout(function(){
            handlers.forEach(function(handler){
                handle(handler)
            })
        }, 0)
    }
    fn(resolve, reject)
}

最后再处理异常的情况

function Promise(fn){
    var handlers = []
    var state = 'pending'
    var value = null
    
    this.then = function(onResolved, onRejected){
        return new Promise(function(resolve, reject){ // 返回 promise 实例,方便链式调用
            handle({
                onResolved,
                resolve,
                onRejected,
                reject,
            })
        })
    }
    function handle(handler){
        if(state === 'pending'){
            handlers.push(handler)
        }else if(state === 'fullfilled'){
            if(handler.onResolved){
                try {
                    var res = handler.onResolved(value)
                    handler.resolve(res)
                } catch(e) {
                    handler.reject(e)
                }
                
            }else{
                handler.resolve(value)
            }
        }else{
            if(handler.onRejected){
                try {
                    var res = handler.onRejected(value)
                    // 因为前一个 promise 中有处理 reject 的情况
                    // 所以直接调用下一个 promise 的 resolve
                    handler.resolve(res)
                } catch(e) {
                    handler.reject(e)
                }
            }else{
                handler.reject(value)
            }
        }
    }
    // 如果 newValue 是 promise 实例,直接调用 .then 方法
    function resolve(newValue){
        try {
            if (typeof newValue === 'object' && typeof newValue.then === 'function'){
                // resolve 就是当前这个定义函数
                newValue.then(resolve, reject)
            } else {
                value = newValue
                state = 'fullfilled'
                setTimeout(function(){
                    handlers.forEach(function(handler){
                        handle(handler)
                    })
                }, 0)
            }
        } catch(e) {
            reject(e)
        }
    }
    function reject(reason){
        value = reason
        state = 'rejected'
        setTimeout(function(){
            handlers.forEach(function(handler){
                handle(handler)
            })
        }, 0)
    }
    fn(resolve, reject)
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值