babel 是如何实现继承以及各种类特性

转换前代码 转换后代码

首先我们写一个 ES6 语法下的继承:

class A {
  #prop = "A#prop";
  get #propgetter() {
    return "A#propgetter";
  }
  static stprop = 0;
  constructor(name) {
    this.name = name;
  }

  #prFunc() {
    return "pr";
  }

  Afunction() {
    console.log(this, this.name);
    console.log(this.#prop);
    console.log(this.#prFunc());
  }

  static stAfunc() {
    console.log(this);
    return this;
  }
}

class B extends A {
  #prop = "B#prop";
  static stprop = 1;
  static #pstprop = 1;
  constructor(name, id) {
    super(name);
    this.id = id;
  }

  Bfunction() {
    console.log(this, this.id);
    console.log(this.#prop);
  }

  static stBfunc() {
    console.log(this);
    return this;
  }
}

babel 官网上可以直接转换,本文使用了 es2015 stage-2,使用了插件 @babel/plugin-syntax-class-properties。下面是转换后的代码。

"use strict";

function _typeof(obj) {
  "@babel/helpers - typeof";
  if (typeof Symbol === "function" && typeof Symbol.iterator === "symbol") {
    _typeof = function _typeof(obj) {
      return typeof obj;
    };
  } else {
    _typeof = function _typeof(obj) {
      return obj &&
        typeof Symbol === "function" &&
        obj.constructor === Symbol &&
        obj !== Symbol.prototype
        ? "symbol"
        : typeof obj;
    };
  }
  return _typeof(obj);
}

function _inherits(subClass, superClass) {
  if (typeof superClass !== "function" && superClass !== null) {
    throw new TypeError("Super expression must either be null or a function");
  }
  subClass.prototype = Object.create(superClass && superClass.prototype, {
    constructor: { value: subClass, writable: true, configurable: true },
  });
  if (superClass) _setPrototypeOf(subClass, superClass);
}

function _setPrototypeOf(o, p) {
  _setPrototypeOf =
    Object.setPrototypeOf ||
    function _setPrototypeOf(o, p) {
      o.__proto__ = p;
      return o;
    };
  return _setPrototypeOf(o, p);
}

function _createSuper(Derived) {
  var hasNativeReflectConstruct = _isNativeReflectConstruct();
  return function _createSuperInternal() {
    var Super = _getPrototypeOf(Derived),
      result;
    if (hasNativeReflectConstruct) {
      var NewTarget = _getPrototypeOf(this).constructor;
      result = Reflect.construct(Super, arguments, NewTarget);
    } else {
      result = Super.apply(this, arguments);
    }
    return _possibleConstructorReturn(this, result);
  };
}

function _possibleConstructorReturn(self, call) {
  if (call && (_typeof(call) === "object" || typeof call === "function")) {
    return call;
  }
  return _assertThisInitialized(self);
}

function _assertThisInitialized(self) {
  if (self === void 0) {
    throw new ReferenceError(
      "this hasn't been initialised - super() hasn't been called"
    );
  }
  return self;
}

function _isNativeReflectConstruct() {
  if (typeof Reflect === "undefined" || !Reflect.construct) return false;
  if (Reflect.construct.sham) return false;
  if (typeof Proxy === "function") return true;
  try {
    Date.prototype.toString.call(Reflect.construct(Date, [], function () {}));
    return true;
  } catch (e) {
    return false;
  }
}

function _getPrototypeOf(o) {
  _getPrototypeOf = Object.setPrototypeOf
    ? Object.getPrototypeOf
    : function _getPrototypeOf(o) {
        return o.__proto__ || Object.getPrototypeOf(o);
      };
  return _getPrototypeOf(o);
}

function _instanceof(left, right) {
  if (
    right != null &&
    typeof Symbol !== "undefined" &&
    right[Symbol.hasInstance]
  ) {
    return !!right[Symbol.hasInstance](left);
  } else {
    return left instanceof right;
  }
}

function _classCallCheck(instance, Constructor) {
  if (!_instanceof(instance, Constructor)) {
    throw new TypeError("Cannot call a class as a function");
  }
}

function _defineProperties(target, props) {
  for (var i = 0; i < props.length; i++) {
    var descriptor = props[i];
    descriptor.enumerable = descriptor.enumerable || false;
    descriptor.configurable = true;
    if ("value" in descriptor) descriptor.writable = true;
    Object.defineProperty(target, descriptor.key, descriptor);
  }
}

function _createClass(Constructor, protoProps, staticProps) {
  if (protoProps) _defineProperties(Constructor.prototype, protoProps);
  if (staticProps) _defineProperties(Constructor, staticProps);
  return Constructor;
}

function _defineProperty(obj, key, value) {
  if (key in obj) {
    Object.defineProperty(obj, key, {
      value: value,
      enumerable: true,
      configurable: true,
      writable: true,
    });
  } else {
    obj[key] = value;
  }
  return obj;
}

function _classPrivateMethodGet(receiver, privateSet, fn) {
  if (!privateSet.has(receiver)) {
    throw new TypeError("attempted to get private field on non-instance");
  }
  return fn;
}

function _classPrivateFieldGet(receiver, privateMap) {
  var descriptor = privateMap.get(receiver);
  if (!descriptor) {
    throw new TypeError("attempted to get private field on non-instance");
  }
  if (descriptor.get) {
    return descriptor.get.call(receiver);
  }
  return descriptor.value;
}

var A = /*#__PURE__*/ (function () {
  function A(name) {
    _classCallCheck(this, A);

    _prFunc.add(this);

    _propgetter.set(this, {
      get: _get_propgetter,
      set: void 0,
    });

    _prop.set(this, {
      writable: true,
      value: "A#prop",
    });

    this.name = name;
  }

  _createClass(
    A,
    [
      {
        key: "Afunction",
        value: function Afunction() {
          console.log(this, this.name);
          console.log(_classPrivateFieldGet(this, _prop));
          console.log(
            _classPrivateMethodGet(this, _prFunc, _prFunc2).call(this)
          );
        },
      },
    ],
    [
      {
        key: "stAfunc",
        value: function stAfunc() {
          console.log(this);
          return this;
        },
      },
    ]
  );

  return A;
})();

var _prop = new WeakMap();

var _propgetter = new WeakMap();

var _prFunc = new WeakSet();

var _get_propgetter = function _get_propgetter() {
  return "A#propgetter";
};

_defineProperty(A, "stprop", 0);

var _prFunc2 = function _prFunc2() {
  return "pr";
};

var B = /*#__PURE__*/ (function (_A) {
  _inherits(B, _A);

  var _super = _createSuper(B);

  function B(name, id) {
    var _this;

    _classCallCheck(this, B);

    _this = _super.call(this, name);

    _prop2.set(_assertThisInitialized(_this), {
      writable: true,
      value: "B#prop",
    });

    _this.id = id;
    return _this;
  }

  _createClass(
    B,
    [
      {
        key: "Bfunction",
        value: function Bfunction() {
          console.log(this, this.id);
          console.log(_classPrivateFieldGet(this, _prop2));
        },
      },
    ],
    [
      {
        key: "stBfunc",
        value: function stBfunc() {
          console.log(this);
          return this;
        },
      },
    ]
  );

  return B;
})(A);

var _prop2 = new WeakMap();

_defineProperty(B, "stprop", 1);

var _pstprop = {
  writable: true,
  value: 1,
};

属性、方法的定义

转换之后比较长,我们一步步分析。

首先,babel 定义了 class A:

// 原代码定义
class A {
  #prop = "A#prop";
  // 注意这里即使是相当新的浏览器也不支持私有属性的getter Edge 83.0.478.44 chrome 83.0.4103.61
  get #propgetter() {
    return "A#propgetter";
  }
  static stprop = 0;
  constructor(name) {
    this.name = name;
  }

  // 同时不支持私有方法 Edge 83.0.478.44 chrome 83.0.4103.61
  #prFunc() {
    return "pr";
  }

  Afunction() {
    console.log(this, this.name);
    console.log(this.#prop);
    console.log(this.#prFunc());
  }

  static stAfunc() {
    console.log(this);
    return this;
  }
}

// 通过匿名函数进行定义
var A = /*#__PURE__*/ (function () {
  function A(name) {
    _classCallCheck(this, A);

    _prFunc.add(this);

    _propgetter.set(this, {
      get: _get_propgetter,
      set: void 0,
    });

    _prop.set(this, {
      writable: true,
      value: "A#prop",
    });

    this.name = name;
  }

  _createClass(
    A,
    [
      {
        key: "Afunction",
        value: function Afunction() {
          console.log(this, this.name);
          console.log(_classPrivateFieldGet(this, _prop));
          console.log(
            _classPrivateMethodGet(this, _prFunc, _prFunc2).call(this)
          );
        },
      },
    ],
    [
      {
        key: "stAfunc",
        value: function stAfunc() {
          console.log(this);
          return this;
        },
      },
    ]
  );

  return A;
})();

var _prop = new WeakMap();

var _propgetter = new WeakMap();

var _prFunc = new WeakSet();

var _get_propgetter = function _get_propgetter() {
  return "A#propgetter";
};

_defineProperty(A, "stprop", 0);

var _prFunc2 = function _prFunc2() {
  return "pr";
};

这里利用闭包在一个匿名函数内部创建了 A 函数,一共使用了三个辅助函数,_classCallCheck _createClass _classPrivateFieldGet。同时使用了WeakMap辅助实例化私有属性,WeakSet辅助实例化私有方法。

按照执行顺序先看 _createClass

function _createClass(Constructor, protoProps, staticProps) {
  // protoProps:
  // [
  //   {
  //     key: "Afunction",
  //     value: function Afunction() {
  //       console.log(this, this.name);
  //       console.log(this.#prop);
  //     },
  //   },
  // ]
  if (protoProps) _defineProperties(Constructor.prototype, protoProps);
  // staticProps:
  // [
  //   {
  //     key: "stAfunc",
  //     value: function stAfunc() {
  //       console.log(this);
  //       return this;
  //     },
  //   },
  // ]
  if (staticProps) _defineProperties(Constructor, staticProps);
  return Constructor;
}

// Object.defineProperty 定义对象属性
function _defineProperties(target, props) {
  for (var i = 0; i < props.length; i++) {
    var descriptor = props[i];
    descriptor.enumerable = descriptor.enumerable || false;
    descriptor.configurable = true;
    if ("value" in descriptor) descriptor.writable = true;
    Object.defineProperty(target, descriptor.key, descriptor);
  }
}

这里利用了Object.defineProperty来对对象属性和对象静态方法进行处理,并挂载到函数 A 的 prototype 和 A (静态方法)本身上。同时也使用 _defineProperties 函数在匿名函数之外对 A 的静态属性 stprop 初始化。

接下来是 A 的实例过程。

// A的定义
function A(name) {
  _classCallCheck(this, A);

  _prFunc.add(this);

  _propgetter.set(this, {
    get: _get_propgetter,
    set: void 0,
  });

  _prop.set(this, {
    writable: true,
    value: "A#prop",
  });

  this.name = name;
}

首先是 _classCallCheck

// 调用语句
_classCallCheck(this, A);

function _classCallCheck(instance, Constructor) {
  if (!_instanceof(instance, Constructor)) {
    throw new TypeError("Cannot call a class as a function");
  }
}

function _instanceof(left, right) {
  if (
    right != null &&
    typeof Symbol !== "undefined" &&
    right[Symbol.hasInstance]
  ) {
    return !!right[Symbol.hasInstance](left);
  } else {
    return left instanceof right;
  }
}

Symbol.hasInstance 是一个确定一个构造器对象识别的对象是否为它的实例的方法。被 instanceof 使用。这里就是查看对应类有没有自定义的方法供 instanceof 调用,有则优先直接使用该方法。

接下来是私有属性和方法的赋值。

_prFunc.add(this);

_propgetter.set(this, {
  get: _get_propgetter,
  set: void 0,
});

_prop.set(this, {
  writable: true,
  value: "A#prop",
});

// 这里 _prop 是全局变量 如果在原代码中也同时定义了 _prop 这里则会变成 _prop2
// WeakMap 可以简单理解为 Map
var _prop = new WeakMap();
var _propgetter = new WeakMap();
// WeakSet 简单理解为 Set
var _prFunc = new WeakSet();

var _get_propgetter = function _get_propgetter() {
  return "A#propgetter";
};

_defineProperty(A, "stprop", 0);

var _prFunc2 = function _prFunc2() {
  return "pr";
};

为什么要使用 WeakMap WeakSet 呢?这两个的共同特点就是只持有内部对象的弱引用,在没有其他引用的时候可以不干预垃圾回收。这样可以避免对象内的属性 或方法由于被外部的 map 或 set 一直引用导致无法触发垃圾回收。

接下来关注_classPrivateFieldGet _classPrivateMethodGet看私有属性和私有方法如何调用:

调用处

// 转换前
console.log(this.#prop);
console.log(this.#prFunc());
// 转换后
console.log(_classPrivateFieldGet(this, _prop));
console.log(_classPrivateMethodGet(this, _prFunc, _prFunc2).call(this));
// 获取私有属性
function _classPrivateFieldGet(receiver, privateMap) {
  // 通过实例本身作为key 获取定义
  var descriptor = privateMap.get(receiver);
  if (!descriptor) {
    throw new TypeError("attempted to get private field on non-instance");
  }
  // 如果是getter 使用getter函数取值
  if (descriptor.get) {
    return descriptor.get.call(receiver);
  }
  // 普通取值走这边
  return descriptor.value;
}

// 获取私有方法
function _classPrivateMethodGet(receiver, privateSet, fn) {
  // set收集了所有A实例的对应私有属性
  if (!privateSet.has(receiver)) {
    throw new TypeError("attempted to get private field on non-instance");
  }
  return fn;
}

继承

接下来是对继承的分析,直接从 B 的定义开始分析:

原定义

class B extends A {
  #prop = "B#prop";
  static stprop = 1;
  static #pstprop = 1;
  constructor(name, id) {
    super(name);
    this.id = id;
  }

  Bfunction() {
    console.log(this, this.id);
    console.log(this.#prop);
  }

  static stBfunc() {
    console.log(this);
    return this;
  }
}

转换后

var B = /*#__PURE__*/ (function (_A) {
  _inherits(B, _A);

  var _super = _createSuper(B);

  function B(name, id) {
    var _this;

    _classCallCheck(this, B);

    _this = _super.call(this, name);

    _prop2.set(_assertThisInitialized(_this), {
      writable: true,
      value: "B#prop",
    });

    _this.id = id;
    return _this;
  }

  _createClass(
    B,
    [
      {
        key: "Bfunction",
        value: function Bfunction() {
          console.log(this, this.id);
          console.log(_classPrivateFieldGet(this, _prop2));
        },
      },
    ],
    [
      {
        key: "stBfunc",
        value: function stBfunc() {
          console.log(this);
          return this;
        },
      },
    ]
  );

  return B;
})(A);

var _prop2 = new WeakMap();

_defineProperty(B, "stprop", 1);

var _pstprop = {
  writable: true,
  value: 1,
};

除了跟 A 定义过程中重复的步骤外,这里还多加了一个静态私有属性 static #pstprop = 1;,和私有静态方法 。
如果进行调用,则会转化为如下:

this.#pstprop;
this.#stFunc();
// 对应
console.log(_classStaticPrivateFieldSpecGet(this, B, _pstprop));
console.log(_classStaticPrivateMethodGet(this, B, _stFunc).call(this));

function _classStaticPrivateMethodGet(receiver, classConstructor, method) {
  if (receiver !== classConstructor) {
    throw new TypeError("Private static access of wrong provenance");
  }
  return method;
}

function _classStaticPrivateFieldSpecGet(
  receiver,
  classConstructor,
  descriptor
) {
  // 检查调用方是否B本身
  if (receiver !== classConstructor) {
    throw new TypeError("Private static access of wrong provenance");
  }
  // 同_classPrivateFieldGet
  if (descriptor.get) {
    return descriptor.get.call(receiver);
  }
  return descriptor.value;
}

静态私有方法比私有方法更简单,通过转换代码把私有方法定义到外部。

关注实例化过程,去掉刚刚分析的 _createClass 部分:

var B = /*#__PURE__*/ (function (_A) {
  // 注意这里的B是下面的函数B
  _inherits(B, _A);

  var _super = _createSuper(B);

  function B(name, id) {
    var _this;

    _classCallCheck(this, B);

    _this = _super.call(this, name);

    _prop2.set(_assertThisInitialized(_this), {
      writable: true,
      value: "B#prop",
    });

    _this.id = id;
    return _this;
  }

  // _createClass...

  return B;
})(A);

首先是 _inherits 函数:

function _inherits(subClass, superClass) {
  if (typeof superClass !== "function" && superClass !== null) {
    throw new TypeError("Super expression must either be null or a function");
  }

  // Object.create将父类prototype上的方法**链接**到子类prototype上

  // 首先superClass && superClass.prototype 的结果是 superClass.prototype
  // Object.create(superClass && superClass.prototype) 的结果是 {__proto__: [superClass.prototype]}

  // Object.create的第二个参数是propertiesObject 这些属性对应Object.defineProperties()的第二个参数
  // 这里的第二个参数目的是将constructor置为subClass {__proto__: [superClass.prototype], constuctor: subClass }
  subClass.prototype = Object.create(superClass && superClass.prototype, {
    constructor: { value: subClass, writable: true, configurable: true },
  });
  // 这里是设置 __proto__ 的地方
  // 简单理解 subClass.__proto__ = superClass
  if (superClass) _setPrototypeOf(subClass, superClass);
}

function _setPrototypeOf(o, p) {
  _setPrototypeOf =
    Object.setPrototypeOf ||
    function _setPrototypeOf(o, p) {
      o.__proto__ = p;
      return o;
    };
  return _setPrototypeOf(o, p);
}

参考 Object.create
这里还可以清晰的看到原型链

  1. subClass.__proto__ = superClass
  2. subClass.prototype.constructor = subClass
  3. subClass.prototype.__proto__ = superClass.prototype

接下来是 var _super = _createSuper(B);, 将在 B 实例化中触发(_this = _super.call(this, name);):

function _createSuper(Derived) {
  // 首先检查是否有定义Reflect
  var hasNativeReflectConstruct = _isNativeReflectConstruct();
  // 返回函数
  return function _createSuperInternal() {
    var Super = _getPrototypeOf(Derived),
      result;
    if (hasNativeReflectConstruct) {
      // 获取实例的constructor 即SubClass
      var NewTarget = _getPrototypeOf(this).constructor;
      // result将会是NewTarget 即SubClass类型
      result = Reflect.construct(Super, arguments, NewTarget);
    } else {
      result = Super.apply(this, arguments);
    }
    // 这里如果result 是走Reflect那一个分支 这里result是一个对象 将会在_possibleConstructorReturn中直接被返回出去
    // 下面那个分支result将会是undefined 将会在_possibleConstructorReturn走_assertThisInitialized分支检查是否已经初始化
    return _possibleConstructorReturn(this, result);
  };
}

function _possibleConstructorReturn(self, call) {
  if (call && (_typeof(call) === "object" || typeof call === "function")) {
    return call;
  }
  // 检查是否已经初始化
  return _assertThisInitialized(self);
}

function _assertThisInitialized(self) {
  if (self === void 0) {
    throw new ReferenceError(
      "this hasn't been initialised - super() hasn't been called"
    );
  }
  return self;
}

// 这个函数测试是否存在NativeReflectConstruct
function _isNativeReflectConstruct() {
  if (typeof Reflect === "undefined" || !Reflect.construct) return false;
  if (Reflect.construct.sham) return false;
  // 这里检查 Proxy 是因为 Reflect 是一个内置的对象,它提供拦截 JavaScript 操作的方法。这些方法与proxy handlers的方法相同。
  if (typeof Proxy === "function") return true;
  try {
    Date.prototype.toString.call(Reflect.construct(Date, [], function () {}));
    return true;
  } catch (e) {
    return false;
  }
}

function _getPrototypeOf(o) {
  _getPrototypeOf = Object.setPrototypeOf
    ? Object.getPrototypeOf
    : function _getPrototypeOf(o) {
        return o.__proto__ || Object.getPrototypeOf(o);
      };
  return _getPrototypeOf(o);
}

Reflectconstruct 方法类似 new 操作符:

function OneClass() {
  this.name = "one";
}

function OtherClass() {
  this.name = "other";
}

// 创建一个对象:
var obj1 = Reflect.construct(OneClass, /* args */ [], OtherClass);
// obj1: OtherClass {name: "one"}

// 与上述方法等效:
var obj2 = Object.create(OtherClass.prototype);
OneClass.apply(obj2, /* args */ []);
// obj2: OtherClass {name: "one"}

obj1 instanceof OneClass; // false
obj2 instanceof OneClass; // false

接下在 A 实例化过程中:

function B(name, id) {
  var _this;

  ...

  // 这里将会调用上面所准备的_super函数
  _this = _super.call(this, name);

  ...

  _this.id = id;
  // 注意这里返回了_this
  return _this;
}

函数 B 最后返回 _this 会让人感觉很奇怪。因为我们调用 B 函数使用的是 new 关键字,但是实际上 new 所触发的函数如果返回值是一个对象,则 new 整个式子也会返回这个对象。

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