手写一个promise

[toc]

本文遵循的Promise/A+规范实现一个简略版本Promise, 用代码理解规范中的每一句话.

Promise 的状态

规范描述

一个 Promise 的当前状态必须为以下三种状态中的一种:等待态(Pending)完成态(Fulfilled)拒绝态(Rejected)

  • 等待态(Pending)
    处于等待态时,promise 需满足以下条件:
    • 可以迁移至执行态或拒绝态
  • 完成态(Fulfilled)
    处于执行态时,promise 需满足以下条件:
    • 不能迁移至其他任何状态
    • 必须拥有一个不可变的终值
  • 拒绝态(Rejected)
    处于拒绝态时,promise 需满足以下条件:
    • 不能迁移至其他任何状态
    • 必须拥有一个不可变的据因

代码实现

采用类实现:

class MyPromise {
}

实现构造函数:

const _ = require('lodash');

const PENDING = 0;
const FULFILLED = 1;
const REJECTED = 2;

class MyPromise {
  constructor(executor) {
    if (!(this instanceof MyPromise)) {
      throw new TypeError("Failed to construct 'Promise': Please use the 'new' operator, this object constructor cannot be called as a function.");
    }
    if (!_.isFunction(executor)) {
      throw new TypeError('You must pass a resolver function as the first argument to the promise constructor');
    }
    this._status = PENDING;
    this._value = undefined;
    this._thenCallbacks = [];
    this._reason = undefined;
    try {
      executor((...args) => this._resolve(...args), (...args) => this._reject(...args));
    } catch(e) {
      this._reject(e);
    }
  }
}

在构造函数中, 首先处理函数边界, 然后初始化状态,终值(_value), 拒因(_reason)和then的回调函数队列; 最后调用传入的executor.executor中用于控制Promise状态的变化.传递两个回调函数作为参数,第一个参数叫做resove,第二个参数叫做 reject, 在对象中的处理处理逻辑为:

_resolve(value) {
    if (this._status !== PENDING) {
      return;
    }
    this._status = FULFILLED;
    this._value = value;
  }

  _reject(reason) {
    if (this._status !== PENDING) {
      return;
    }
    this._status = REJECTED;
    this._reason = reason;
  }

_resolve, _reject有状态边界判断, 如果被调用多次, 采用首次调用并忽略剩下的调用._resolve中只允许等待态转为完成态,然后接收终值._reject中只允许等待态转为拒绝态, 然后接收拒因.

状态的判断保证_resolve, _reject中主要逻辑在当前promise中最多被执行一次: 状态最多改变一次;then的回调函数最多调用一次.

Then函数

规范描述

一个promise必须提供一个then方法以访问其当前值、终值和据因。
promisethen方法接受两个参数:

promise.then(onFulfilled, onRejected)

参数可选

onFulfilledonRejected都是可选参数。

  • 如果onFulfilled不是函数,其必须被忽略
  • 如果onRejected不是函数,其必须被忽略

onFulfilled特性

如果onFulfilled是函数:

  • promise执行结束后其必须被调用,其第一个参数为promise的终值
  • promise执行结束前其不可被调用
  • 其调用次数不可超过一次

调用时机

onFulfilledonRejected只有在执行环境堆栈仅包含平台代码时才可被调用

调用要求

onFulfilledonRejected必须被作为函数调用(即没有 this 值)注2

多次调用

then方法可以被同一个promise调用多次

promise成功执行时,所有onFulfilled需按照其注册顺序依次回调
promise被拒绝执行时,所有的onRejected需按照其注册顺序依次回调

返回

then方法必须返回一个promise对象

promise2 = promise1.then(onFulfilled, onRejected);   
  • 如果onFulfilled或者onRejected返回一个值 x ,则运行下面的Promise解决过程:[[Resolve]](promise2, x)
  • 如果onFulfilled或者onRejected抛出一个异常e,则promise2必须拒绝执行,并返回拒因e
  • 如果onFulfilled不是函数且promise1成功执行,promise2必须成功执行并返回相同的值
  • 如果 onRejected 不是函数且promise1拒绝执行,promise2必须拒绝执行并返回相同的据因

代码实现

需要在构造函数添加回调函数队列:

constructor(executor) {
    // ...
    this._value = undefined;
    this._thenCallbacks = [];
    this._reason = undefined;
    // ...
  }

首先, 实现一个then函数.由于本章节代码牵扯到很多部分, 所以尽量用代码注释来说明实现的规范:

then(onFulfilled, onRejected) {
    // 如果 onFulfilled 不是函数,其必须被忽略
    const _onFulfilled = _.isFunction(onFulfilled) ? onFulfilled : void 0;
    // 如果 onRejected 不是函数,其必须被忽略
    const _onRejected = _.isFunction(onRejected) ? onRejected : void 0;

    // then 方法可以被同一个 promise 调用多次
    this._thenCallbacks.push([_onFulfilled, _onRejected]);

    return new MyPromise((resolve, reject) => {
      // 等待实现
    });
  }

onFulfilledonRejected, 需要在_resove或者reject时被调用, 将resolve, reject改造为:

_resolve(value) {
    if (this._status !== PENDING) {
      return;
    }
    this._status = FULFILLED;
    this._value = value;
    // 如果then的回调函数onFulfilled, onRejected为函数的话, 需要
    // 在 promise 执行结束前其不可被调用,当 promise 执行结束后其必须被调用
    // 其调用次数不可超过一次
    // 只有在执行环境堆栈仅包含平台代码时才可被调用 
    process.nextTick(this._callThenCallbacks);
  }

  _reject(reason) {
    if (this._status !== PENDING) {
      return;
    }
    this._status = REJECTED;
    this._reason = reason;
    // 如果then的回调函数onFulfilled, onRejected为函数的话, 需要
    // 在 promise 执行结束前其不可被调用,当 promise 执行结束后其必须被调用
    // 其调用次数不可超过一次
    // 只有在执行环境堆栈仅包含平台代码时才可被调用 
    process.nextTick(this._callThenCallbacks);
  }

虽然规范中没有说明onFulfilledonRejected必需为微任务(micro-task)还是宏任务(macro-task),但是其他实现基本都是基于微任务.

这里由于本文只是在node环境实现,所以采用process.nextTick来启用微任务, 为了跨平台, 一般的Promise实现框架, 都会使用多种方式来实现在执行环境堆栈仅包含平台代码时才可被调用, 如MutationObserver, MessageChannel, vertx等, 最后可能使用setTimeout实现.

按照Promise/A+规范来说,onFulfilledonRejected只有在执行环境堆栈仅包含平台代码时才可被调用, 但是在nodejs或者es6环境中的Promise对象, 需要像下面的实现:

_resolve(value) {
    // 只有在执行环境堆栈仅包含平台代码时才可被调用 
    process.nextTick(() => {
      if (this._status !== PENDING) {
        return;
      }
      this._status = FULFILLED;
      this._value = value;
      // 如果then的回调函数onFulfilled, onRejected为函数的话, 需要
      // 在 promise 执行结束前其不可被调用,当 promise 执行结束后其必须被调用
      // 其调用次数不可超过一次
      // 只有在执行环境堆栈仅包含平台代码时才可被调用 
      // process.nextTick(() => this._callThenCallbacks());
      this._callThenCallbacks();
    });
  }

  _reject(reason) {
    // 只有在执行环境堆栈仅包含平台代码时才可被调用 
    process.nextTick(() => {
      if (this._status !== PENDING) {
        return;
      }
      this._status = REJECTED;
      this._reason = reason;
      // 如果then的回调函数onFulfilled, onRejected为函数的话, 需要
      // 在 promise 执行结束前其不可被调用,当 promise 执行结束后其必须被调用
      // 其调用次数不可超过一次
      // process.nextTick(() => this._callThenCallbacks());
      this._callThenCallbacks();
    });
  }

对于比较流行的prmise polyfill库es-promise的实现,采用的是上面一种启用微任务的时机, 对于babel中的垫片core-js中实现的promise, 采用的是下一种启用时机.

Then函数.返回规范有复杂的要求,为了实现这些要求, 需要改变上面的then函数的实现:

then(onFulfilled, onRejected) {
    // 如果 onFulfilled 不是函数,其必须被忽略
    const _onFulfilled = _.isFunction(onFulfilled) ? onFulfilled : void 0;
    // 如果 onRejected 不是函数,其必须被忽略
    const _onRejected = _.isFunction(onRejected) ? onRejected : void 0;
    
    let childResolve;
    let childReject;
    const childPromise = new MyPromise((resolve, reject) => {
      childResolve = resolve;
      childReject = reject;
    });

    // then 方法可以被同一个 promise 调用多次
    this._thenCallbacks.push([_onFulfilled, _onRejected, childResolve, childReject]);

    return childPromise;
  }

_callThenCallbacks用于处理在promise状态改变后处理then回调函数队列. 在处理每一个then回调函数后, 还需要对于then回调函数返回的结果, 结合当前的promise状态,调整当前then函数返回的promise2的状态:

  // 调用then回调函数队列
  _callThenCallbacks() {
    if (_.isEmpty(this._thenCallbacks)) {
      return;
    }
    this._thenCallbacks.forEach(([onFulfilled, onRejected, childResolve, childReject]) => {
      try {
        if (this._status === FULFILLED && !onFulfilled) {
          // 如果 onFulfilled 不是函数且 promise1 成功执行, promise2 必须成功执行并返回相同的值
          childResolve(this._value);
          return;
        }
        if (this._status === REJECTED && !onRejected) {
          // 如果 onRejected 不是函数且 promise1 拒绝执行, promise2 必须拒绝执行并返回相同的据因
          childReject(this._reason);
        }
        let x;
        if (this._status === REJECTED && onRejected) {
          // 当 promise 被拒绝执行时,所有的 onRejected 需按照其注册顺序依次回调
          // 其第一个参数为 promise 的拒因
          // 必须被作为函数调用(即没有 this 值)
          x = onRejected(this._reason);
        } else if (this._status === FULFILLED && onFulfilled) {
          // 当 promise 成功执行时,所有 onFulfilled 需按照其注册顺序依次回调
          // 其第一个参数为 promise 的终值
          // 必须被作为函数调用(即没有 this 值)
          x = onFulfilled(this._value);
        }
        // 如果 onFulfilled 或者 onRejected 返回一个值 x ,则运行下面的 Promise 解决过程
        this._resolvePromise(x, childResolve, childReject);
      } catch (error) {
        childReject(error);
      }
    });
  }

其中_resolvePromise代表Promise解决过程, 将在下文说明.

Promise解决过程

规范描述

Promise解决过程是一个抽象的操作,其需输入一个promise和一个值,我们表示为[[Resolve]](promise, x),如果xthen 方法且看上去像一个Promise,解决程序即尝试使promise接受x的状态;否则其用x的值来执行 promise 。

这种thenable的特性使得Promise的实现更具有通用性:只要其暴露出一个遵循 Promise/A+ 协议的then方法即可;这同时也使遵循 Promise/A+ 规范的实现可以与那些不太规范但可用的实现能良好共存。

运行[[Resolve]](promise, x)需遵循以下步骤:

  • xpromise相等
    如果x为Promise,则使promise 接受x`的状态:

    • 如果x处于等待态,promise需保持为等待态直至x被执行或拒绝
    • 如果x处于执行态,用相同的值执行promise
    • 如果x处于拒绝态,用相同的据因拒绝promise
  • x为对象或函数
    如果x为对象或者函数:

    • x.then赋值给then
    • 如果取x.then的值时抛出错误e,则以e为据因拒绝promise
    • 如果then是函数,将x作为函数的作用域this调用之。传递两个回调函数作为参数,第一个参数叫做resolvePromise,第二个参数叫做rejectPromise:
      • 如果resolvePromise以值y为参数被调用,则运行[[Resolve]](promise, y)
      • 如果rejectPromise以据因r为参数被调用,则以据因r拒绝promise
      • 如果resolvePromiserejectPromise均被调用,或者被同一参数调用了多次,则优先采用首次调用并忽略剩下的调用
      • 如果调用then方法抛出了异常e
        • 如果resolvePromiserejectPromise已经被调用,则忽略之
        • 否则以e为据因拒绝promise
      • 如果then不是函数,以x为参数执行promise
  • 如果x不为对象或者函数,以x为参数执行promise`

如果一个promise被一个循环的thenable链中的对象解决,而[[Resolve]](promise, thenable)的递归性质又使得其被再次调用,根据上述的算法将会陷入无限递归之中。算法虽不强制要求,但也鼓励施者检测这样的递归是否存在,若检测到存在则以一个可识别的TypeError为据因来拒绝promise.

代码实现

函数_resolvePromise实现, 采用代码注释说明:

// Promise 解决过程
  _resolvePromise(x, childResolve, childReject) {
    // x 与 promise 相等
    if (x === this) {
      // 如果 promise 和 x 指向同一对象,以 TypeError 为据因拒绝执行 promise
      throw new TypeError("You cannot resolve a promise with itself");
    }
    // 如果 x 为 Promise ,则使 promise 接受 x 的状态
    if (x instanceof MyPromise) {
      // 如果 x 处于等待态
      console.log('======PENDING===', x._status);
      if (x._status === PENDING) {
        // promise 需保持为等待态直至 x 被执行或拒绝
        x.then(childResolve, childReject);
        return;
      }
      // 如果 x 处于执行态
      if (x._status === FULFILLED) {
        // 用相同的值执行 promise
        childResolve(x._value);
        return;
      }
      // 如果 x 处于执行态
      if (x._status === REJECTED) {
        // 用相同的值执行 promise
        childReject(x._reason);
        return;
      }
    }
    // x 为对象或函数
    if (_.isObject(x) || _.isFunction(x)) {
      // 把 x.then 赋值给 then
      let then;
      try {
        then = x.then;
      } catch (error) {
        // 如果取 x.then 的值时抛出错误 e ,则以 e 为据因拒绝 promise
        // 其实这里不需要捕获, 因为最外层有捕获, 这里为了保持跟规范一致
        childReject(error);
        return;
      }
      // 如果 then 是函数
      if (_.isFunction(then)) {
        // 将 x 作为函数的作用域 this 调用之
        let called = false;
        try {
          then.call(x, (y) => {
            // 如果 resolvePromise 和 rejectPromise 均被调用,或者被同一参数调用了多次,
            // 则优先采用首次调用并忽略剩下的调用
            if (called) {
              return;
            }
            called = true;
  
            // 如果 resolvePromise 以值 y 为参数被调用
            this._resolvePromise(y, childResolve, childReject);
          }, (r) => {
            // 如果 resolvePromise 或 rejectPromise 已经被调用,则忽略之
            if (called) {
              return;
            }
            called = true;
  
            // 如果 rejectPromise 以据因 r 为参数被调用,则以据因 r 拒绝 promise
            childReject(r);
          }); 
        } catch (error) {
          // 如果调用 then 方法抛出了异常 e

          // 如果 resolvePromise 或 rejectPromise 已经被调用,则忽略之
          if (called) {
            return;
          }

          // 否则以 e 为据因拒绝 promise
          childReject(error);
        }
        return;
      }
      // 如果 then 不是函数, 以 x 为参数执行 promise
      childResolve(x);
      return;
    }
    // 如果 x 不为对象或者函数, 以 x 为参数执行 promise
    childResolve(x);
  }

全部代码

const _ = require('lodash');

const PENDING = 0;
const FULFILLED = 1;
const REJECTED = 2;

class MyPromise {
  constructor(executor) {
    if (!(this instanceof MyPromise)) {
      throw new TypeError("Failed to construct 'Promise': Please use the 'new' operator, this object constructor cannot be called as a function.");
    }
    if (!_.isFunction(executor)) {
      throw new TypeError('You must pass a resolver function as the first argument to the promise constructor');
    }
    this._status = PENDING;
    this._value = undefined;
    this._thenCallbacks = [];
    this._reason = undefined;
    try {
      executor((...args) => this._resolve(...args), (...args) => this._reject(...args));
    } catch(e) {
      this._reject(e);
    }
  }

  _resolve(value) {
    // 只有在执行环境堆栈仅包含平台代码时才可被调用 
    process.nextTick(() => {
      if (this._status !== PENDING) {
        return;
      }
      this._status = FULFILLED;
      this._value = value;
      // 如果then的回调函数onFulfilled, onRejected为函数的话, 需要
      // 在 promise 执行结束前其不可被调用,当 promise 执行结束后其必须被调用
      // 其调用次数不可超过一次
      // 只有在执行环境堆栈仅包含平台代码时才可被调用 
      // process.nextTick(() => this._callThenCallbacks());
      this._callThenCallbacks();
    });
  }

  _reject(reason) {
    // 只有在执行环境堆栈仅包含平台代码时才可被调用 
    process.nextTick(() => {
      if (this._status !== PENDING) {
        return;
      }
      this._status = REJECTED;
      this._reason = reason;
      // 如果then的回调函数onFulfilled, onRejected为函数的话, 需要
      // 在 promise 执行结束前其不可被调用,当 promise 执行结束后其必须被调用
      // 其调用次数不可超过一次
      // 只有在执行环境堆栈仅包含平台代码时才可被调用 
      // process.nextTick(() => this._callThenCallbacks());
      this._callThenCallbacks();
    });
  }

  // 调用then回调函数队列
  _callThenCallbacks() {
    if (_.isEmpty(this._thenCallbacks)) {
      return;
    }
    this._thenCallbacks.forEach(([onFulfilled, onRejected, childResolve, childReject]) => {
      try {
        if (this._status === FULFILLED && !onFulfilled) {
          // 如果 onFulfilled 不是函数且 promise1 成功执行, promise2 必须成功执行并返回相同的值
          childResolve(this._value);
          return;
        }
        if (this._status === REJECTED && !onRejected) {
          // 如果 onRejected 不是函数且 promise1 拒绝执行, promise2 必须拒绝执行并返回相同的据因
          childReject(this._reason);
        }
        let x;
        if (this._status === REJECTED && onRejected) {
          // 当 promise 被拒绝执行时,所有的 onRejected 需按照其注册顺序依次回调
          // 其第一个参数为 promise 的拒因
          // 必须被作为函数调用(即没有 this 值)
          x = onRejected(this._reason);
        } else if (this._status === FULFILLED && onFulfilled) {
          // 当 promise 成功执行时,所有 onFulfilled 需按照其注册顺序依次回调
          // 其第一个参数为 promise 的终值
          // 必须被作为函数调用(即没有 this 值)
          x = onFulfilled(this._value);
        }
        // 如果 onFulfilled 或者 onRejected 返回一个值 x ,则运行下面的 Promise 解决过程
        this._resolvePromise(x, childResolve, childReject);
      } catch (error) {
        childReject(error);
      }
    });
  }

  // Promise 解决过程
  _resolvePromise(x, childResolve, childReject) {
    // x 与 promise 相等
    if (x === this) {
      // 如果 promise 和 x 指向同一对象,以 TypeError 为据因拒绝执行 promise
      throw new TypeError("You cannot resolve a promise with itself");
    }
    // 如果 x 为 Promise ,则使 promise 接受 x 的状态
    if (x instanceof MyPromise) {
      // 如果 x 处于等待态
      console.log('======PENDING===', x._status);
      if (x._status === PENDING) {
        // promise 需保持为等待态直至 x 被执行或拒绝
        x.then(childResolve, childReject);
        return;
      }
      // 如果 x 处于执行态
      if (x._status === FULFILLED) {
        // 用相同的值执行 promise
        childResolve(x._value);
        return;
      }
      // 如果 x 处于执行态
      if (x._status === REJECTED) {
        // 用相同的值执行 promise
        childReject(x._reason);
        return;
      }
    }
    // x 为对象或函数
    if (_.isObject(x) || _.isFunction(x)) {
      // 把 x.then 赋值给 then
      let then;
      try {
        then = x.then;
      } catch (error) {
        // 如果取 x.then 的值时抛出错误 e ,则以 e 为据因拒绝 promise
        // 其实这里不需要捕获, 因为最外层有捕获, 这里为了保持跟规范一致
        childReject(error);
        return;
      }
      // 如果 then 是函数
      if (_.isFunction(then)) {
        // 将 x 作为函数的作用域 this 调用之
        let called = false;
        try {
          then.call(x, (y) => {
            // 如果 resolvePromise 和 rejectPromise 均被调用,或者被同一参数调用了多次,
            // 则优先采用首次调用并忽略剩下的调用
            if (called) {
              return;
            }
            called = true;
  
            // 如果 resolvePromise 以值 y 为参数被调用
            this._resolvePromise(y, childResolve, childReject);
          }, (r) => {
            // 如果 resolvePromise 或 rejectPromise 已经被调用,则忽略之
            if (called) {
              return;
            }
            called = true;
  
            // 如果 rejectPromise 以据因 r 为参数被调用,则以据因 r 拒绝 promise
            childReject(r);
          }); 
        } catch (error) {
          // 如果调用 then 方法抛出了异常 e

          // 如果 resolvePromise 或 rejectPromise 已经被调用,则忽略之
          if (called) {
            return;
          }

          // 否则以 e 为据因拒绝 promise
          childReject(error);
        }
        return;
      }
      // 如果 then 不是函数, 以 x 为参数执行 promise
      childResolve(x);
      return;
    }
    // 如果 x 不为对象或者函数, 以 x 为参数执行 promise
    childResolve(x);
  }

  then(onFulfilled, onRejected) {
    // 如果 onFulfilled 不是函数,其必须被忽略
    const _onFulfilled = _.isFunction(onFulfilled) ? onFulfilled : void 0;
    // 如果 onRejected 不是函数,其必须被忽略
    const _onRejected = _.isFunction(onRejected) ? onRejected : void 0;
    
    let childResolve;
    let childReject;
    const childPromise = new MyPromise((resolve, reject) => {
      childResolve = resolve;
      childReject = reject;
    });

    // then 方法可以被同一个 promise 调用多次
    this._thenCallbacks.push([_onFulfilled, _onRejected, childResolve, childReject]);

    return childPromise;
  }
}

module.exports = MyPromise;
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,242评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,769评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,484评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,133评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,007评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,080评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,496评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,190评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,464评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,549评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,330评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,205评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,567评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,889评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,160评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,475评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,650评论 2 335