浅谈vu3中的多种组件形式

前言

自从vue3彻底支持函数式编程后,终于支持jsx的写法了。写过react的朋友可能更容易理解无状态组件与有状态组件的区别。对写惯了vue2对象式编程的人来说,可能对于函数式组件还有些一知半解,时常不知道普通函数与jsx函数有什么区别,毕竟习惯了在.vue文件中的template一把梭。分不清在函数组件中直接返回jsx模板与返回一个对象又有什么区别。这篇文章我将浅谈一下自己的理解,希望为你解惑。

*基础好的朋友可直接阅读第四章

一、脚手架的.vue文件是什么

这里我们不讨论脚手架具体是如何处理.vue文件的,这个涉及到打包器的知识,感兴趣的朋友可以自行拓展了解。我们在基于vite或webpack的脚手架中开发vue时,能够很方便的在.vue文件中开发,一个.vue文件的基本三要素就是“template”、“style”以及“script”。在其他vue文件中使用时,只需要 import A from 'A.vue',然后在template模板中写上 <A />即可渲染。这是一个大家再熟悉不过的流程,可.vue文件的本质是什么呢?我们不妨输出一下我们引入的组件:

Test.vue
<template>
  <div>测试</div>
</template>
<script setup lang="ts"></script>
<style lang="scss" scoped></style>

在App.vue中引入Test.vue,然后输出Test
<script setup lang="ts">
    import Test from './Test.vue';
    console.log(Test);
</script>
查看浏览器中的输出结果
image.png

我们可以很清楚的看到,我们打印的Test输出了一个对象,并且对象中存在一个render函数,其他两个带下划线的我们暂且忽略。这里最重要的是一个render对象。 我们由此可见,在模板中引入的vue组件,实际上是一个带有render函数的对象。也就是说,vue在执行我们的组件时,会去对象中找render函数并且执行它,render函数会返回一个Vnodes渲染树,Vue就能根据这个树进行渲染显示。

二、render函数是什么

在vue中,render函数实际上返回的是一个h函数返回的虚拟dom树,实际上就是一个对象。它里面存储了很多dom的节点信息,这些信息都是虚拟的。这里我们不考虑vue的template,我们暂时把它忘掉。这里我们考虑一个场景,我给定你一个对象,例如

const obj = {
  div:{
    style:"color:red;font-size:13px"
  }
}

我的需求是,希望你用这个对象,使用你知道的所有javascript知识,将它渲染到网页里。
这里我就不贴代码了。
我相信对你来说这是一件很简单的事情,不就是读取这个obj对象,然后创建相对应的div标签,接着将style设置为对象中的style,最后append到dom容器里不就好了吗?
是的,你的做法以及思想都完全没有错,而且这也不算很有难度。那么,如果我希望在div下再渲染一个span呢?像这样:

const obj = {
  div: {
    style: 'color:red;font-size:13px',
    children: [
                   { span: { style: 'color:red', content: '内容' } },
                   { span: { style: 'color:red', content: '内容2' } }
              ]
          },
      };

你可能注意到,我在div下增加了一个叫children的数组,为什么是数组?因为div下不可能只有一个元素嘛!那么现在我希望在div下渲染出这两个span,并且content 内容也要显示在标签里,你会怎么做呢?
我觉得这个对于看我文章的人来说,应该也是毫无压力的,无非就是处理div的时候,顺便找一下有没有children,有的话就一起处理,最后一起append到dom容器里就好了嘛!

你真是个天才!根本难不倒你,你的想法完全可行。

可是content内容不可能一直不变,如果我需要修改span中的内容,那么你又得重新执行一遍你刚刚的处理的流程,我假设你很聪明的把刚刚对于obj的处理写了一个函数,那就意味着,你会想办法监听到我对obj到更改。只要有更改,你就会重新执行,确保页面显示的跟我的obj对象是一致的。

如果你真的这么做了,当obj这个对象所表达的标签足够多时,可能只是其中一个标签中的content改变,你就需要重新运行整个函数,然后整个处理完再全部append。你的页面性能将会迎来瓶颈,说人话就是卡到爆。

你会如何处理这个性能问题呢?

我觉得可以自己悄悄再写一个函数,专门用来缓存上次保存的obj对象,当原始obj对象更改时,我将缓存里的obj对象和更改的obj对象先进行一次比对,找出实际被更改的那个标签,然后单独对这个标签进行append。这样,我就可以做到只对有更改的标签进行append到页面上,做到了局部的DOM更新。 很棒对吧?!

如果你也觉得上面这个方案很棒,尤雨溪也是这么觉得的,这就是我们所谓的diff算法。

看到这里,实际上我们已经实现了最基本的数据驱动视图,我们只需要控制obj这个对象,具体的渲染更新我们交给了事先封装好的针对处理obj的函数(我假设你封装好了)。

我们刚刚讨论的那个obj,里面的结构完全是我定义的,你是被动的。虽然可以实现想要的效果,但每个人都有自己的想法。比如那个content,你偏偏改成text或者value来表达不行吗? 那当然行啦!谁能犟过你啊。 但是无规矩不成方圆,既然这个方案很好用,能实现数据驱动视图的最基本方案。是不是应该大力推广呢?那用的人一多,那就必须有一套规范了,可不能乱来。

这套规范叫hyperscript,简称为h,就是我们熟悉的h函数啦。实际上,h函数就是我们上面讨论规范后的结构。具体的可以参考vue官方文档:https://cn.vuejs.org/guide/extras/render-function.html#creating-vnodes

为什么是h函数而不是对象?
在实际开发中,框架除了要处理我们的模板外,还要处理很多其他的逻辑。比如说,你写了一个虚拟DOM表达对象:{type:"div",....},但一个成熟可用的框架要做的事情很多,例如vue可能还需要维护虚拟DOM的唯一性,需要往对象里插入更多属性配合性能优化。所以vue自身提供了一个h函数,我们可以把虚拟dom表达在函数中。vue调用h函数会经过一层处理,再次返回一个可用的虚拟DOM树,也就是VNode。同样,在vue3中使用jsx时,它的逻辑也是h函数的语法糖,最终vue拿到的都需要是一个Vnode。也就是一个实际可用的虚拟DOM树。

写了这么多,终于可以回到主题了,什么是render函数? 其实render函数返回的就是我们上面写的obj对象,只不过vue自身封装了一个h函数,h函数基于hyperscript规范同时又增加了vue中特有的字段,比如props。最终h函数会返回一个vue实际可用的Vnode虚拟DOM对象。总的来说,render是返回一个可执行的h函数,h函数返回一个Vnode对象,对象结构里表达了各种标签渲染规则。

还记得一开始我打印的组件吗?留给大家一个小问题:为什么vue调用render返回h函数,为什么不直接命名为h而是命名为render呢?

三、什么是JSX

我们上面聊过h函数,在vue中,它就是接收hyperscript基础规范语法的函数。并且h函数返回一个vue可用的虚拟DOM树。对象里描述了将来渲染真实html的规则。又由于对象可以层层嵌套描述更多html规则,所以我们把这种对象称之为虚拟DOM树。为什么说是虚拟?因为它是一个js对象,并不是真实dom,真实dom是需要你根据这个对象再去处理渲染的。又因为它层层嵌套很像一棵树分叉,所以我们称之为虚拟DOM树。
说这么多,那什么是jsx呢?要不说程序员偷起懒来,就会促进科技的进步呢。实际上JSX就是一种偷懒的结果。我们先看下面这个示例:

const test = h('div', { id: 'foo' }, 'hello')

这是一个很简单的h函数,里面描述了一个div,它代表根节点。自身id为"foo",并且这个div内有一个文本叫"hello"。我把这个h函数用test包装起来,Vue的h函数将会为我们返回vue可执行的虚拟DOM树,将来它渲染出html就是以下样式:

<div id="foo">
  hello
</div>

看起来结果很好,但现在我们只有一个div。我们知道在实际开发中,要写完整个页面可不止这一个div。那总不可能所有写页面过程都面向这个h函数吧,要一个个手写对象去描述html结构,想想就头大了。 所以这个时候jsx来了:

const test = <div id="foo">hello</div>

是的你没看错,我把html丢到了变量里。那有朋友就要问了:这么写js不会报错吗?
会!而且大错特错!js才不认识你这堆玩意儿呢。js里有这种奇葩的做法吗?再不济写到js里也应该用字符串包裹一下吧,你这直接扔过来算什么?报错是必然的。
所以,JSX它注定要经过一层编译处理。无论你是在原生scirpt里写jsx语法还是在脚手架里写jsx语法。要么你就用scirpt引入解析jsx的插件,要么你就是在脚手架里配置好jsx的插件。无论如何,它都必须要经过一层处理。
处理什么?如何处理?实际上,就是把jsx的语法转换为h函数,然后再交给实际要处理h函数渲染的逻辑去处理。
说到这里,肯定有朋友要问了:那既然jsx还得转换成h函数才能执行,那不是比直接写h函数执行效率更低了吗?
我说朋友,你要这么想,也没错。但是jsx我们一般是在开发环境使用,毕竟一般开发react和vue的都上脚手架了是吧。实际上在我们最后执行打包操作后,打包完的代码里是没有jsx的,所以在打包后的运行代码是没有性能消耗的,也就是开发运行时耗一点性能。再说这都2024了,你是要jsx呢?还是要自己手写h函数里的DOM结构描述对象?
jsx作为h函数的语法糖,它并不只是带给我们布局更方便的好处。由于我们将布局揉合到了js里,也就意味着可以在jsx模板里一起把变量渲染的逻辑做了。
比如:

const name = '张三'
const test = <div id="foo">{name}</div>

渲染结果:

<div id="foo">
 张三
</div>

是不是很方便?那有同学说了,如果我变量更新了,如何保证重新渲染呢?
我们之前讨论过,jsx只不过是h函数的语法糖,而h函数也只不过是返回一个标准化的虚拟DOM树而已。说人话就是 只是统一了数据规范。至于如何处理jsx中的逻辑,不同的框架语言有自己的优化逻辑。比如react能搭配hooks和state对虚拟DOM树进行动态更新渲染。同样vue3中的ref也可以搭配jsx使用,就像vue3的h函数在返回可用虚拟DOM之前,会往对象里插入更多其他的属性,这些都是vue的一部分。
总之,jsx是h函数的语法糖。除此之外 我们需要搭配框架语言本身提供的功能使用。也就是说,在vue3里写jsx跟react里最简单的不同点是,你们可以在jsx基本模板语法保持一致的情况下,可以使用自身框架的状态管理和hook来编写代码。

四、VUE多种组件形式之:函数式组件(无状态组件)

函数式组件通常用来做无状态组件和有状态组件,这里并不是说.vue文件那种不能写无状态和有状态无状态组件,只是在无状态组件比较简单时,更优先采用函数式。在vue3中,平时当我们编写组件时,我们会在组件内写很多响应式变量,可能还会写点击事件改变某些变量从而重新渲染页面。这种有自己内部响应式变量需要管理维护的组件叫有状态组件。而相反,如果一个组件它只是接收一个参数,然后渲染到页面,自身并没有定义响应式变量和方法来异步控制页面更新,那么这种就叫无状态组件。 类似组件比如时间转换组件,可能你有一个组件接收一个时间props,处理后展示到页面上,仅此而已没有其他变量状态需要管理。这种我们就可以称之为无状态组件,它就两件事 1.接收变量2.处理变量渲染。也有可能就一件事:渲染静态。
没有任何自身其他的响应式变量需要管理。

一般无状态组件,很适合编写一个函数,然后函数内返回一个jsx模板,比如下面的示例:

假设有个index.jsx有以下代码:
export default  function HandleTime(){
  return <div>2024年x月x日<div>
}

这是一个很普通的函数式组件,那如何使用这个组件呢?
假设我们有个index.vue:

<templat>
      <HanldeTime />
</template>
<script  setup>
import HanldeTime from 'index.jsx'
</script>

以上代码会在页面渲染出以下结果

<div>2024年x月x日</div>

这是很显而易见的结果,但是 我编写的HandleTime不是一个函数吗?为什么vue能把我的函数当作模板渲染?
在第一小节中,我打印了一个.vue后缀的组件,在浏览器中能看到它实际上是一个对象,里面有个render方法。所以我们其实可以得到一个结论:vue的.vue后缀文件被解析成一个对象,并且在解析过程中,template会被解析成render函数。那render函数又返回什么呢?
在第二小节我已经给了解答,实际上render返回的是一个h函数。而h函数又是返回一个包装好的虚拟DOM树,也就是Vnode,vue拿到Vnode就能执行渲染逻辑了。

那到底为什么vue可以把HanldeTime函数当作模板渲染?
因为一个我编写的函数组件,采用的是jsx的方式返回

 return <div>2024年x月x日<div>

这一段return实际上就是jsx,然而我们前面总结过,在vue中jsx其实就是h函数的语法糖。 而我们又看到一个组件输出的是一个对象里面包含render,调用render仍然也是为了拿到h函数的结果得到Vnode。所以如果我们函数里直接返回了jsx也就意味着直接返回了Vnode,所以它可以直接放在模板中当作组件渲染。

既然如此,如果项目中没有用到jsx,应当如何返回一个无状态组件?
那就不得不手动引入vue提供的h函数了,你可以用以下写法

import { h } from 'vue'
export default  function HandleTime(){
  return h('div','2024年x月x日')
}

引入这个HandleTime,渲染的效果跟用jsx是一样的,因为jsx就是h函数的语法糖。在没有jsx的情况下,用h函数来应急也是不错的选择,但不建议大量在项目中使用,是真的很难维护。

五、VUE多种组件形式之:函数式组件(有状态组件)

我们了解过无状态函数组件后,总会发现一个组件不能传参、不能管理自己的变量的话,使用场景会变得很局限。所以如何给一个函数式组件传参?
其实很简单:

<templat>
      <HandleTime name='张三' />
</template>
<script  setup>
import HandleTime from 'index.jsx'
</script>  

我们可以把vue的模板使用当作函数的调用来看待,这里其实就是调用了HandleTime这个函数,并且将name当作函数参数传递进去了。

export default  function HandleTime(props){
  return <div>{props.name}<div>
}

参数会变成一个对象传递进来,在HandleTime这个函数中接收,直接使用这个name就可以了。
那有人会问了,如果外面的name变化了怎么办?
其实,在vue3中,只需要保证你传递的这个name是响应式变量,当name更新时,HandleTtime组件就会重新渲染。
比如你可以这么写:

<templat>
      <HandleTime :name='name' />
</template>
<script  setup>
import {ref} from 'vue
import HandleTime from 'index.jsx'
const name = ref('张三')
</script> 

我把name用ref包裹,并且传递到HandleTime组件中,后续只要name更新,组件内也会跟着一起重新渲染。

到此为止,我们其实还算是无状态组件,无非就是传递了一个参数进去,如果Handle需要自己的响应式变量怎么办?
其实也很简单,直接引用vue的函数就好了,这也是vue3的特点:

import { ref}  from 'vue'
export default  function HandleTime(props){
  const count = ref(0)
  return <div onClick=()=>{ count++ }>{props.name}<div>
}

这就是一个最基础的有状态组件,我们可以引入Vue的hook来搭配使用。是不是很简单?

*问题解答:为什么vue调用render返回h函数,为什么不直接命名为h而是命名为render呢?

因为.vue的文件中,render只是转换了你的template模板代码,render实际上是不包含你的业务代码的,包括生命周期也不包含在内。render负责渲染html那么就只是需要返回一个h函数的Vnode即可。 除了渲染工作,还有其他的事情要做,所以重新定义一个render函数,可以在render函数中做很多中间处理,最后只需要返回一个可用的Vnode即可。类似于:

{
  render(){
    return h('div')
  }
}

而在render中还可以做很多中间操作

{
  render(){
    const name = 'content'
    return h('div',name)
  }
}

vue在渲染时,调用render即可拿到需要渲染的Vnode信息,而除了render,vue还可以在这个对象里存储其他的数据,比如生命周期或一些私有变量。
这里也就意味着,我们直接用函数式组件 等于就直接返回了Vnode,不需要再次使用render来包装,我们打印一个函数式组件会发现控制台有如下输出:


image.png

这就可以证明,函数式组件确实没有render,而是直接返回的一个可执行的函数,并且里面就是Vnode,执行就可以渲染。

但如果在特殊情况下,我们的jsx渲染或者组件不得不自己写在对象里,也是可以起名叫render的,毕竟这个单词它好理解,直观。

六、VUE多种组件形式之:对象式组件

细心的朋友可能发现了,讲了半天函数式组件,生命周期呢?昂?生命周期呢?
不好意思,函数式组件没办法使用生命周期hook。
想要在组件中使用生命周期,我们先看文档是怎么说的:


image.png

什么意思呢?意思就是说,生命周期必须要在setup周期内执行,生命周期钩子也可以封装到其他函数里,但前提是函数调用必须在setup周期内。但请大家注意了,它这里指的是.vue模板开发中,你可以把生命周期钩子写在别的函数里,在setup作用域使用即可。 但我们这个函数式组件是直接给模板使用了,不存在什么setup,在函数式组件里写生命周期是无效的。

但同时它也给出了答案,在setup上下文中使用即可。
还记得vue的对象式编程刚转vue3的时候吗?是不是用setup函数代替的?那时候还没有script setup这个用法。这里,我们也可以这样使用:

在index.jsx中
import {onMounted} from 'vue'
export default {
  setup(){
        onMounted(()=>{
              console.log('生命周期加载')
          })
        return '<div>你好</div>'
    }
}

同样在模板中

<template>
  <HandleTime />
</template>
<script setup>
  import  HandleTime from 'index.jsx'
<script>

有朋友会细心的发现,不对啊,setup最终return不是代表对外暴露的变量吗?
的确是,我们回想一下,在vue3刚出来的时候,我们是不是可以在写template的情况下,然后对象上加一个setup就好了? 但是我们这边是不是没有template了,我们也没办法使用template,因为我们这不是约束在.vue模板中了,那没有template了怎么办?
这里的问题还是,template是什么?我们前面讨论过,templte就是render函数的语法糖,而render返回的是h函数。h函数又会返回Vnode。没错吧?所以没有了template,我们需要手动写上render,所以以下的代码也是可用的:

import {onMounted} from 'vue'
export default {
    render(){
      return '<div>你好</div>'
    },
    setup(){
        onMounted(()=>{
              console.log('生命周期加载')
          })
      
    }
}

而以上写法,会使得变量的传递显得异常的麻烦,因为上面的写法不多,这里不再过多赘述,我们了解一下本质就可以。

因为变量传递过于麻烦,所以vue3还支持在setup中直接return返回模板,至于变量我们在setup中做就可以了,生命周期也能用了。

那么问题来了,本身setup返回的是对外暴露的变量,这么一写,还怎么暴露变量?现在都默认暴露的是JSX了,也就是Vnode。 怎么办?

我们可以使用expose函数,它不是一个需要显示导入的hook,以下是官方示例用法:

export default {
  setup(props, { expose }) {
    // 让组件实例处于 “关闭状态”
    // 即不向父组件暴露任何东西
    expose()

    const publicCount = ref(0)
    const privateCount = ref(0)
    // 有选择地暴露局部状态
    expose({ count: publicCount })
  }
}

此时我们可以用expose替代原本的return暴露属性,return我们就用来返回一个jsx。 具体的细节可以查看官方文档:https://cn.vuejs.org/api/composition-api-setup.html#usage-with-render-functions

不知道大家注意到没有,我导出的是一个对象。而函数组件我导出的是一个函数,无论是函数还是对象,vue都可以正常解析,这里面的基础逻辑是什么呢?可以总结为:
当vue模板渲染时,如果拿到的是一个函数,会尝试执行它,如果返回的不是一个Vnode,将会在控制台报错,如果拿到的是Vnode,将会正常渲染成html。
如果拿到的是一个对象,vue会去找这个对象的render函数,如果没有render函数将会执行setup,最终拿到setup返回的Vnode。

七:VUE多种组件形式之:辅助函数包装组件:defineComponent

辅助函数包装组件这个名字是我自己取的,我也不知道具体如何称呼它,但也是字面意思,它通过vue提供的辅助函数编写组件。
它的作用主要是为Ts提供类型推导,以下是官方语法:

import { ref, h } from 'vue'

const Comp = defineComponent(
  (props) => {
    // 就像在 <script setup> 中一样使用组合式 API
    const count = ref(0)

    return () => {
      // 渲染函数或 JSX
      return h('div', count.value)
    }
  },
  // 其他选项,例如声明 props 和 emits。
  {
    props: {
      /* ... */
    }
  }
)

它接收两个参数,参数1可以当setup函数使用,最终返回一个Vnode,也就是JSX。但它这里没有基于jsx,所以返回的是h函数。如果有引入jsx这里直接返回jsx就可以。
第二个参数是组件的一些选项,这里不再过多赘述了,就是提供多了一些TS支持,如果有需要可以用defineComponent包装一下,它既支持生命周期又支持TS推导,也是一个不错的选择,不过我比较少用。
具体的可看文档了解:https://cn.vuejs.org/api/general.html#definecomponent

最后

这篇文章我尝试从vue的h函数开始到JSX到templat模板语法,接着给大家介绍基于这些概念的几种组件编写方式。知识点多而杂,如有理解错误的地方 还请大家指出,多多包涵。感谢你的耐心阅读。

完。

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

推荐阅读更多精彩内容