在微信小程序中直接运行React组件

广告位招租
扫码页面底部二维码联系

在研究跨端开发时,我的一个重要目标,是可【作者:唐霜】【本文受版权保护】以让react组件跑在微信小程序中。在这【作者:唐霜】原创内容,盗版必究。个过程中,我探索了微信小程序的架构,并且【转载请注明来源】【转载请注明来源】引发了很多思考。而作为跨端开发,实际上很原创内容,盗版必究。本文版权归作者所有,未经授权不得转载。难做到 write once,run a转载请注明出处:www.tangshuang.net【本文首发于唐霜的博客】nywhere,因为每个平台所提供的能力【访问 www.tangshuang.net 获取更多精彩内容】【访问 www.tangshuang.net 获取更多精彩内容】是不一样的,例如微信小程序提供了原生的能转载请注明出处:www.tangshuang.net未经授权,禁止复制转载。力,例如调起摄像头或其他需要原生环境支持本文作者:唐霜,转载请注明出处。【作者:唐霜】的能力,在微信小程序中开发虽然也是在we【版权所有,侵权必究】【本文受版权保护】bview中开展,但是,却需要一些原生的本文版权归作者所有,未经授权不得转载。【访问 www.tangshuang.net 获取更多精彩内容】思维。所以,要做到 write once【作者:唐霜】转载请注明出处:www.tangshuang.net 就必须有一些限制,这些限制注定了我们无原创内容,盗版必究。未经授权,禁止复制转载。法完全利用小程序的能力,仅仅只用到一些布转载请注明出处:www.tangshuang.net本文作者:唐霜,转载请注明出处。局的能力而已。所以,奉劝各位,在做跨端开【原创不易,请尊重版权】【本文受版权保护】发时,要有个心理准备。但如果跳出跨端开发【原创不易,请尊重版权】【访问 www.tangshuang.net 获取更多精彩内容】,我现在只开发小程序,那我能否用我熟悉的原创内容,盗版必究。转载请注明出处:www.tangshuang.netreact来开发呢?甚至,能否用我开发的nautil框架著作权归作者所有,禁止商业用途转载。来开发呢?答案是可以的,本文将带你一步一【作者:唐霜】本文版权归作者所有,未经授权不得转载。步实现自己的react小程序开发之路,帮原创内容,盗版必究。【访问 www.tangshuang.net 获取更多精彩内容】助你在某些特定的场景下,完成react项【关注微信公众号:wwwtangshuangnet】未经授权,禁止复制转载。目往小程序迁移的目标。

【作者:唐霜】未经授权,禁止复制转载。【转载请注明来源】原创内容,盗版必究。本文作者:唐霜,转载请注明出处。

小程序运行React的方案对比【作者:唐霜】

【原创内容,转载请注明出处】【版权所有】唐霜 www.tangshuang.net【本文受版权保护】【版权所有】唐霜 www.tangshuang.net本文作者:唐霜,转载请注明出处。

目前业界能够比较好支持小程序(没有特别注本文作者:唐霜,转载请注明出处。本文作者:唐霜,转载请注明出处。明的情况下,小程序特指微信小程序)运行R本文作者:唐霜,转载请注明出处。原创内容,盗版必究。eact组件的,有3套方案,分别是京东凹【本文受版权保护】【转载请注明来源】凸实验室的taro,蚂蚁金服某团队(未找【版权所有,侵权必究】著作权归作者所有,禁止商业用途转载。到具体团队名)的remax,微信某团队的【版权所有,侵权必究】【本文首发于唐霜的博客】kbone。

转载请注明出处:www.tangshuang.net原创内容,盗版必究。【作者:唐霜】【版权所有】唐霜 www.tangshuang.net
方案【作者:唐霜】 思路【版权所有,侵权必究】 优势未经授权,禁止复制转载。 劣势本文版权归作者所有,未经授权不得转载。
Taro【本文首发于唐霜的博客】 编译,新版本也基于运行时未经授权,禁止复制转载。 解析为wxml+js原创内容,盗版必究。 老牌,不断发展,全平台支持,持续迭代【原创不易,请尊重版权】
Remax【本文首发于唐霜的博客】 运行时,带编译宏【作者:唐霜】 基于reconciler【本文受版权保护】 最优雅,增量更新【版权所有】唐霜 www.tangshuang.net 不够成熟,后续发展未知本文版权归作者所有,未经授权不得转载。
Kbone本文版权归作者所有,未经授权不得转载。 运行时,依赖webpack【访问 www.tangshuang.net 获取更多精彩内容】 自己实现一套DOM API【访问 www.tangshuang.net 获取更多精彩内容】 可兼容vue,甚至任意基于DOM渲染的框转载请注明出处:www.tangshuang.net本文作者:唐霜,转载请注明出处。 性能问题(全量检查),几乎停更【原创不易,请尊重版权】

3套方案各有不同,而且在各自的思路上都是【转载请注明来源】转载请注明出处:www.tangshuang.net独树一帜。就我个人而言,如果不考虑跨端开【关注微信公众号:wwwtangshuangnet】本文作者:唐霜,转载请注明出处。发,自己实现一套DOM API这种方案是【访问 www.tangshuang.net 获取更多精彩内容】转载请注明出处:www.tangshuang.net非常有价值的,因为DOM接口是HTML标【访问 www.tangshuang.net 获取更多精彩内容】著作权归作者所有,禁止商业用途转载。准,你不需要自己去发明一套标准出来,而一【本文首发于唐霜的博客】本文作者:唐霜,转载请注明出处。旦实现了DOM API,那么所以其他基于著作权归作者所有,禁止商业用途转载。【原创内容,转载请注明出处】DOM实现的应用理论上都支持在这上面跑。【本文受版权保护】【访问 www.tangshuang.net 获取更多精彩内容】但是,它的不足就是你每换一个平台,就要针【原创不易,请尊重版权】未经授权,禁止复制转载。对这个平台去实现一套DOM API,这个【原创内容,转载请注明出处】【版权所有】唐霜 www.tangshuang.net成本是非常大的,因为DOM接口标准极其庞【版权所有,侵权必究】【转载请注明来源】大,实现的时候也很容易出bug。在我看来【版权所有】唐霜 www.tangshuang.net转载请注明出处:www.tangshuang.net,最优雅的实现还是Remax的那种思路,【版权所有,侵权必究】【转载请注明来源】基于react-reconciler做一原创内容,盗版必究。本文作者:唐霜,转载请注明出处。个渲染器,这个渲染器将react组件实例原创内容,盗版必究。著作权归作者所有,禁止商业用途转载。抽象为一个统一的DSL,在不同的平台上,转载请注明出处:www.tangshuang.net著作权归作者所有,禁止商业用途转载。去解析渲染这个DSL。

本文作者:唐霜,转载请注明出处。本文版权归作者所有,未经授权不得转载。【原创内容,转载请注明出处】【未经授权禁止转载】

但是remax迭代更新之后,它开始强依赖本文版权归作者所有,未经授权不得转载。本文作者:唐霜,转载请注明出处。自己的编译工具,这直接导致我放弃在项目中未经授权,禁止复制转载。【关注微信公众号:wwwtangshuangnet】使用它。因为对于我们自己的项目而言,我们【访问 www.tangshuang.net 获取更多精彩内容】【关注微信公众号:wwwtangshuangnet】其实有可能不需要它的全部,我们只是使用r本文作者:唐霜,转载请注明出处。未经授权,禁止复制转载。eact来完成我们整个小程序中的某些部分原创内容,盗版必究。【关注微信公众号:wwwtangshuangnet】(比如有些已经用react写好的h5我们【本文首发于唐霜的博客】【版权所有】唐霜 www.tangshuang.net想要渲染到小程序,其他部分我们还是在原来转载请注明出处:www.tangshuang.net【本文受版权保护】的项目中跑)。如果对它的编译工具有依赖,【原创内容,转载请注明出处】【本文受版权保护】我们就不得不把整个项目迁移到它的编译工具本文版权归作者所有,未经授权不得转载。转载请注明出处:www.tangshuang.net,那我还不如直接使用taro这个老牌比较【作者:唐霜】【版权所有】唐霜 www.tangshuang.net稳定的工具。

本文版权归作者所有,未经授权不得转载。【未经授权禁止转载】本文版权归作者所有,未经授权不得转载。【访问 www.tangshuang.net 获取更多精彩内容】原创内容,盗版必究。

整体实现思路【原创不易,请尊重版权】

【本文受版权保护】【本文受版权保护】【原创内容,转载请注明出处】【关注微信公众号:wwwtangshuangnet】本文作者:唐霜,转载请注明出处。

经过一番研究之后,我决定采用remax的本文版权归作者所有,未经授权不得转载。【访问 www.tangshuang.net 获取更多精彩内容】思路,也就是基于react-reconc【原创不易,请尊重版权】【作者:唐霜】iler实现一个渲染器,生成一个DSL,著作权归作者所有,禁止商业用途转载。原创内容,盗版必究。再创建一个小程序组件,去解析和渲染这个D【转载请注明来源】【版权所有,侵权必究】SL。在完成实现之后,我把所有这些逻辑构转载请注明出处:www.tangshuang.net【本文首发于唐霜的博客】建为最终产物,并以npm的形式发布产物,著作权归作者所有,禁止商业用途转载。【作者:唐霜】对于小程序开发者而言,只需要npm安装之本文作者:唐霜,转载请注明出处。【转载请注明来源】后,执行开发者工具中的构建npm即可,之【本文首发于唐霜的博客】本文作者:唐霜,转载请注明出处。后在自己的页面中引入这个包,利用api即【转载请注明来源】【作者:唐霜】可完成开发,而不在需要使用另外的编译工具转载请注明出处:www.tangshuang.net【原创不易,请尊重版权】

【原创内容,转载请注明出处】未经授权,禁止复制转载。原创内容,盗版必究。转载请注明出处:www.tangshuang.net

这一方案的最大好处是,对编译工具的弱(无【原创内容,转载请注明出处】本文版权归作者所有,未经授权不得转载。)依赖,这样就可以让我们的这套方案可以在【版权所有,侵权必究】【作者:唐霜】任意的项目中去跑,而不需要额外引入编译工转载请注明出处:www.tangshuang.net【转载请注明来源】具切换工具栈。另外,因为reconcil【转载请注明来源】未经授权,禁止复制转载。er的部分已经打包进npm包了,所以它是转载请注明出处:www.tangshuang.net未经授权,禁止复制转载。一个可以独立运行的模块,所以,你甚至可以【版权所有】唐霜 www.tangshuang.net【作者:唐霜】在mpvue等vue风格或小程序原生风格【版权所有】唐霜 www.tangshuang.net本文版权归作者所有,未经授权不得转载。项目中使用这个npm包来渲染react的【版权所有】唐霜 www.tangshuang.net【本文首发于唐霜的博客】组件。

【版权所有,侵权必究】原创内容,盗版必究。【作者:唐霜】【作者:唐霜】原创内容,盗版必究。

微信小程序中运行react组件的思路【版权所有】唐霜 www.tangshuang.net

本文作者:唐霜,转载请注明出处。【未经授权禁止转载】【关注微信公众号:wwwtangshuangnet】【版权所有,侵权必究】转载请注明出处:www.tangshuang.net

如上图所示,我们将一个react组件通过【关注微信公众号:wwwtangshuangnet】本文版权归作者所有,未经授权不得转载。基于react-reconciler的渲【版权所有】唐霜 www.tangshuang.net【关注微信公众号:wwwtangshuangnet】染器,创建了一个DSL的纯对象(包含回调【未经授权禁止转载】【原创不易,请尊重版权】函数),我们在page的js文件中,通过本文作者:唐霜,转载请注明出处。【版权所有,侵权必究】this.setData把这个对象发送给【本文受版权保护】本文作者:唐霜,转载请注明出处。渲染线程,在wxml中使用了我们提供的一【访问 www.tangshuang.net 获取更多精彩内容】【未经授权禁止转载】个自引用嵌套的组件对DSL进行渲染。这里【原创不易,请尊重版权】【版权所有,侵权必究】需要注意一个点,react-reconc原创内容,盗版必究。【原创内容,转载请注明出处】iler会在组件更新的时候,触发对应的钩【原创不易,请尊重版权】【作者:唐霜】子,此时,会再次生成新的DSL,并再次通【原创内容,转载请注明出处】【原创不易,请尊重版权】过this.setData发送渲染。所以转载请注明出处:www.tangshuang.net【本文首发于唐霜的博客】,这个渲染器和单纯使用createEle【转载请注明来源】原创内容,盗版必究。ment的结果是不同的,渲染器支持hoo本文版权归作者所有,未经授权不得转载。【原创内容,转载请注明出处】ks等react内置的功能。

【未经授权禁止转载】本文版权归作者所有,未经授权不得转载。【版权所有】唐霜 www.tangshuang.net

接下来,我将对其中的具体细节进行讲解,以【转载请注明来源】转载请注明出处:www.tangshuang.net让你尽可能自己可以手写出本文所阐述的代码【关注微信公众号:wwwtangshuangnet】本文作者:唐霜,转载请注明出处。,以让你在自己的项目中可以实现本文一致的【转载请注明来源】【未经授权禁止转载】效果。你可以克隆这个仓库到本地本文版权归作者所有,未经授权不得转载。,运行效果看看,研究它的整个实现过程。转载请注明出处:www.tangshuang.net

著作权归作者所有,禁止商业用途转载。【原创不易,请尊重版权】【版权所有】唐霜 www.tangshuang.net

将react组件渲染为纯JS对象【本文受版权保护】

【作者:唐霜】【作者:唐霜】【本文受版权保护】未经授权,禁止复制转载。

react的渲染器本质上是一个基于rea著作权归作者所有,禁止商业用途转载。【本文受版权保护】ct调度系统的副作用执行器,副作用的结果本文版权归作者所有,未经授权不得转载。【作者:唐霜】在web环境下就是DOM的操作,在nat本文作者:唐霜,转载请注明出处。著作权归作者所有,禁止商业用途转载。ive环境下就是调用渲染引擎光栅化图形,本文版权归作者所有,未经授权不得转载。未经授权,禁止复制转载。在art环境下就是调用声卡播放声音,而在本文作者:唐霜,转载请注明出处。著作权归作者所有,禁止商业用途转载。我们这次的计划中,我们需要渲染器生成一个【本文受版权保护】【作者:唐霜】纯js对象,以方便交给小程序在小程序的两本文作者:唐霜,转载请注明出处。本文版权归作者所有,未经授权不得转载。个线程之间作为消息体进行传递,并基于这个本文作者:唐霜,转载请注明出处。原创内容,盗版必究。对象在小程序中渲染界面。

本文作者:唐霜,转载请注明出处。本文作者:唐霜,转载请注明出处。原创内容,盗版必究。

有同学对我发出疑问:jsx编译之后Rea【原创内容,转载请注明出处】【本文首发于唐霜的博客】ct.createElement的执行结【本文首发于唐霜的博客】【作者:唐霜】果不就是纯JS的对象么?这里需要了解re本文作者:唐霜,转载请注明出处。【原创内容,转载请注明出处】act的本质。react的组件,实际上为【版权所有】唐霜 www.tangshuang.net未经授权,禁止复制转载。react提供了一套描述系统,它描述了r【原创不易,请尊重版权】【未经授权禁止转载】eact所表达的具体对象的结构。但是,这【转载请注明来源】转载请注明出处:www.tangshuang.net个描述是抽象的,只有当你把它实例化,运行【本文受版权保护】【本文受版权保护】起来时,它才有意义。我们在组件中所做的描【原创不易,请尊重版权】【原创不易,请尊重版权】述,可不单单只有jsx的部分,它还包括业本文版权归作者所有,未经授权不得转载。原创内容,盗版必究。务和程序层面的逻辑。比如很多场景下,我们本文作者:唐霜,转载请注明出处。【访问 www.tangshuang.net 获取更多精彩内容】需要根据组件状态来决定返回那一部分jsx【版权所有】唐霜 www.tangshuang.net转载请注明出处:www.tangshuang.net,从而渲染不同的界面。而这部分内容,需要【关注微信公众号:wwwtangshuangnet】著作权归作者所有,禁止商业用途转载。依赖一个环境来执行,也就是react渲染【未经授权禁止转载】【版权所有,侵权必究】器。

本文作者:唐霜,转载请注明出处。【访问 www.tangshuang.net 获取更多精彩内容】原创内容,盗版必究。本文版权归作者所有,未经授权不得转载。本文版权归作者所有,未经授权不得转载。

在以前,我们只能模拟react-dom,【原创内容,转载请注明出处】【访问 www.tangshuang.net 获取更多精彩内容】按照它的运行逻辑,自己手写一个渲染器。而【版权所有】唐霜 www.tangshuang.net本文作者:唐霜,转载请注明出处。现在,react把它的调度器专门做了一个著作权归作者所有,禁止商业用途转载。转载请注明出处:www.tangshuang.net库,react-reconciler,帮【本文受版权保护】【转载请注明来源】助开发者快速接入react的调度系统,从本文版权归作者所有,未经授权不得转载。【关注微信公众号:wwwtangshuangnet】而可以构建自己的渲染器。这里有一个视频(自备梯子),介绍了rea【版权所有】唐霜 www.tangshuang.net【原创不易,请尊重版权】ct-reconciler的基本用法和使原创内容,盗版必究。【原创内容,转载请注明出处】用效果转载请注明出处:www.tangshuang.net

转载请注明出处:www.tangshuang.net未经授权,禁止复制转载。【本文受版权保护】
import Reconciler from 'react-reconciler'

const container = {}
const HostConfig = {
  // ... 极其复杂的一个配置
}
const reconcilerInstance = Reconciler(HostConfig)

let rootContainerInstance = null

export function render(element, { mounted, updated, created }) {
  if (!rootContainerInstance) {
    rootContainerInstance = reconcilerInstance.createContainer(container, false, false)
  }
  return reconcilerInstance.updateContainer(element, rootContainerInstance, null, () => {
    notify = { mounted, updated, created }
    created && created(container)
    mounted(container.data)
  })
}

上面代码中,没有给出的HostConfi【转载请注明来源】著作权归作者所有,禁止商业用途转载。g的具体内容是关键,它用于配制一个Rec未经授权,禁止复制转载。【本文首发于唐霜的博客】onciler,从代码的角度,它就是一个未经授权,禁止复制转载。【版权所有】唐霜 www.tangshuang.net钩子函数的集合,我们需要在每个钩子函数内【未经授权禁止转载】【版权所有,侵权必究】部写一些副作用来操作container,【本文受版权保护】【本文受版权保护】你可以看到,在不同的时刻,我们传入的cr转载请注明出处:www.tangshuang.net【本文首发于唐霜的博客】eated, mounted, upda转载请注明出处:www.tangshuang.net本文版权归作者所有,未经授权不得转载。ted会被调用,而它们接收被操作过的co著作权归作者所有,禁止商业用途转载。【版权所有,侵权必究】ntainer,从而让我们获得这个js对【版权所有,侵权必究】【关注微信公众号:wwwtangshuangnet】象(container上还有一些函数,但【作者:唐霜】转载请注明出处:www.tangshuang.net我们可以不用理会,因为this.setD转载请注明出处:www.tangshuang.net【本文受版权保护】ata会自动清除这些函数)。

【原创内容,转载请注明出处】【版权所有】唐霜 www.tangshuang.net本文作者:唐霜,转载请注明出处。【关注微信公众号:wwwtangshuangnet】

由于这一配置内容太过复杂,要讲解清楚需要本文作者:唐霜,转载请注明出处。转载请注明出处:www.tangshuang.net花费比较大的篇幅,所以我直接把源码地址贴在这里,你可以通过阅读【未经授权禁止转载】【转载请注明来源】源码来了解它都有哪些配置项,并且你可以把这部分代码拆分出来后,运行【本文受版权保护】【原创内容,转载请注明出处】一个自己的组件,通过console.lo原创内容,盗版必究。【未经授权禁止转载】g来观察它们被调用的时机以及顺序。

【原创内容,转载请注明出处】本文版权归作者所有,未经授权不得转载。著作权归作者所有,禁止商业用途转载。著作权归作者所有,禁止商业用途转载。

总而言之,这些接口都是知识层面的,不是什【版权所有】唐霜 www.tangshuang.net【本文首发于唐霜的博客】么复杂的逻辑,了解每一个配置项的作用和执本文作者:唐霜,转载请注明出处。【版权所有】唐霜 www.tangshuang.net行时机之后,你就能写出自己的渲染器。理论本文作者:唐霜,转载请注明出处。【访问 www.tangshuang.net 获取更多精彩内容】上,它没有什么难度。

原创内容,盗版必究。本文版权归作者所有,未经授权不得转载。【原创内容,转载请注明出处】【关注微信公众号:wwwtangshuangnet】原创内容,盗版必究。

基于react-reconciler,我原创内容,盗版必究。【关注微信公众号:wwwtangshuangnet】在react运行时的每一个环节都做了一些【版权所有】唐霜 www.tangshuang.net本文作者:唐霜,转载请注明出处。副作用操作,这些副作用的本质,就是修改一【作者:唐霜】著作权归作者所有,禁止商业用途转载。个纯js对象,当react被运行起来时,【版权所有】唐霜 www.tangshuang.net【关注微信公众号:wwwtangshuangnet】它会经历一个生命周期,这在我的一个视频中有讲到react的生命周期【本文受版权保护】著作权归作者所有,禁止商业用途转载。的具体过程。你也可以关注我的个人微信公众号 www【关注微信公众号:wwwtangshuangnet】【原创内容,转载请注明出处】tangshuangnet 和我讨论相关未经授权,禁止复制转载。【作者:唐霜】的问题。在每一个生命周期节点上,调度器就著作权归作者所有,禁止商业用途转载。【作者:唐霜】会执行一个副作用,即修改我提供的那个纯j本文作者:唐霜,转载请注明出处。【版权所有】唐霜 www.tangshuang.nets对象。

转载请注明出处:www.tangshuang.net【访问 www.tangshuang.net 获取更多精彩内容】转载请注明出处:www.tangshuang.net【原创内容,转载请注明出处】

我提供了两个方法,用于在小程序的渲染器中【版权所有,侵权必究】【版权所有】唐霜 www.tangshuang.net,获得生成好的js对象。得到这个js对象【作者:唐霜】【转载请注明来源】之后,就可以调用小程序的this.set本文作者:唐霜,转载请注明出处。【本文首发于唐霜的博客】Data,把这个对象发送到渲染线程进行渲转载请注明出处:www.tangshuang.net【本文受版权保护】染。

【转载请注明来源】【版权所有】唐霜 www.tangshuang.net本文版权归作者所有,未经授权不得转载。转载请注明出处:www.tangshuang.net

利用react渲染器得到的纯对象上存在一【未经授权禁止转载】本文版权归作者所有,未经授权不得转载。些函数,调用这些函数会触发它们对应的逻辑【版权所有,侵权必究】原创内容,盗版必究。(比如调用setState触发hooks【访问 www.tangshuang.net 获取更多精彩内容】转载请注明出处:www.tangshuang.net状态更新),从而触发调度器中的钩子函数执【关注微信公众号:wwwtangshuangnet】【版权所有】唐霜 www.tangshuang.net行,container对象再次被修改,u未经授权,禁止复制转载。未经授权,禁止复制转载。pdated被再次调用,this.set本文作者:唐霜,转载请注明出处。【版权所有】唐霜 www.tangshuang.netData被再次执行,这样,就实现了真正的【作者:唐霜】【本文受版权保护】react运行时在小程序中的植入。

【版权所有】唐霜 www.tangshuang.net本文版权归作者所有,未经授权不得转载。【作者:唐霜】

嵌套递归自引用组件【版权所有】唐霜 www.tangshuang.net

【原创内容,转载请注明出处】本文版权归作者所有,未经授权不得转载。本文版权归作者所有,未经授权不得转载。【访问 www.tangshuang.net 获取更多精彩内容】

渲染线程接收到this.setData发本文作者:唐霜,转载请注明出处。本文作者:唐霜,转载请注明出处。送过来的js对象后,如何将这个对象作为布原创内容,盗版必究。原创内容,盗版必究。局的信息,渲染到界面上呢?由于小程序的特【关注微信公众号:wwwtangshuangnet】【原创内容,转载请注明出处】殊架构,它为了安全起见,渲染线程中无法执【本文首发于唐霜的博客】【未经授权禁止转载】行可操作界面的脚本,所有的渲染,都得依靠【作者:唐霜】本文作者:唐霜,转载请注明出处。模板语法和少量的wxs脚本。所以,要怎么【关注微信公众号:wwwtangshuangnet】【本文首发于唐霜的博客】做呢?

转载请注明出处:www.tangshuang.net【本文首发于唐霜的博客】【本文受版权保护】【版权所有】唐霜 www.tangshuang.net【作者:唐霜】

小程序提供了自定义组件的功能,在app.【版权所有,侵权必究】【作者:唐霜】json或对应的page.json中,通【版权所有】唐霜 www.tangshuang.net【访问 www.tangshuang.net 获取更多精彩内容】过usingComponents来指定一【作者:唐霜】【原创不易,请尊重版权】个路径,从而可以在wxml中使用这个组件转载请注明出处:www.tangshuang.net原创内容,盗版必究。。而有趣的地方在于,组件本身也可以在组件本文作者:唐霜,转载请注明出处。【本文首发于唐霜的博客】自己的component.json中使用【本文首发于唐霜的博客】【原创内容,转载请注明出处】usingComponents这个配置,【作者:唐霜】【作者:唐霜】而这个配置的内容,可以直接指向自己,例如本文版权归作者所有,未经授权不得转载。【版权所有,侵权必究】,我在自己的组件中,这样自引用:

本文作者:唐霜,转载请注明出处。【原创不易,请尊重版权】原创内容,盗版必究。【本文首发于唐霜的博客】
// dynamic.json
{
  "usingComponents": {
    "dynamic": "./dynamic"
  }
}

自己引用自己作为组件之后,在其wxml中【关注微信公众号:wwwtangshuangnet】【本文受版权保护】,我们就可以使用组件自己去渲染子级数据,【原创不易,请尊重版权】【版权所有】唐霜 www.tangshuang.net即一种嵌套递归的形式进行渲染。

本文作者:唐霜,转载请注明出处。【关注微信公众号:wwwtangshuangnet】转载请注明出处:www.tangshuang.net

我规定了一种特别的数据结构,大致如下:【访问 www.tangshuang.net 获取更多精彩内容】

【转载请注明来源】本文版权归作者所有,未经授权不得转载。本文版权归作者所有,未经授权不得转载。
{
  type: 'view',
  props: {
    class: 'shadow-component',
    bindtap: (e) => { ... },
  },
  children: [
    {
      type: 'view',
      props: {},
      children: [
        ...
      ],
    },
  ],
}

模板中,通过对type的判断,选择不同的【未经授权禁止转载】【关注微信公众号:wwwtangshuangnet】模板代码进行渲染。

原创内容,盗版必究。【版权所有,侵权必究】【访问 www.tangshuang.net 获取更多精彩内容】【关注微信公众号:wwwtangshuangnet】【原创不易,请尊重版权】
<block wx:if="{{ type === 'view' }}">
  <view class="{{ props.class }}" bindtap="bindtap">
    <block wx:if="{{ children.length }}" wx:for="{{ children }}">
      <dynamic data="{{ item }}" /> <!-- 嵌套递归 -->
    </block>
  </view>
</block>

在wxml中把所有组件通过这种形式枚举出【访问 www.tangshuang.net 获取更多精彩内容】【作者:唐霜】来之后,这个组件就能按照上述的数据结构递【未经授权禁止转载】【版权所有,侵权必究】归渲染出整个结构。

【作者:唐霜】【转载请注明来源】【原创不易,请尊重版权】【访问 www.tangshuang.net 获取更多精彩内容】未经授权,禁止复制转载。

当然,这里还需要处理一些细节,例如响应d【访问 www.tangshuang.net 获取更多精彩内容】【关注微信公众号:wwwtangshuangnet】ata的变化,事件响应函数等,你可以通过源码了解具体要怎么处理【原创内容,转载请注明出处】。另外,微信小程序this.setDat本文版权归作者所有,未经授权不得转载。本文作者:唐霜,转载请注明出处。a限制在1M以内,我虽然还没有尝试过很大著作权归作者所有,禁止商业用途转载。本文版权归作者所有,未经授权不得转载。的数据,但是,这个限制肯定在将来是一个风【关注微信公众号:wwwtangshuangnet】【访问 www.tangshuang.net 获取更多精彩内容】险点,我现在还没有解决,还在思考应该怎么【本文首发于唐霜的博客】【作者:唐霜】最小化更新粒度。

【本文首发于唐霜的博客】【未经授权禁止转载】本文作者:唐霜,转载请注明出处。未经授权,禁止复制转载。

不支持直接JSX的变通方法著作权归作者所有,禁止商业用途转载。

转载请注明出处:www.tangshuang.net【本文受版权保护】转载请注明出处:www.tangshuang.net【版权所有】唐霜 www.tangshuang.net【版权所有,侵权必究】

小程序的编译,没有办法自己配置支持新语法本文作者:唐霜,转载请注明出处。【版权所有,侵权必究】,所以如果我们在小程序代码中使用jsx,【本文首发于唐霜的博客】本文作者:唐霜,转载请注明出处。就必须先走一遍自己的编译逻辑。有两种解决【版权所有】唐霜 www.tangshuang.net【转载请注明来源】办法,一种是不使用jsx语法,而是使用h【作者:唐霜】著作权归作者所有,禁止商业用途转载。yperscript标记语法,比如:

转载请注明出处:www.tangshuang.net著作权归作者所有,禁止商业用途转载。【作者:唐霜】【作者:唐霜】【作者:唐霜】
import { createElement as h } from 'react'

function Some() {
  return h(
    'view',
    { class: 'some-component' },
    h(
      'view',
      { class: 'sub-view' },
      '一段文字',
    ),
    '一段文字',
  )
}

这样的写法显然没有直接写jsx来的方便,【本文首发于唐霜的博客】未经授权,禁止复制转载。但是阅读上没有什么障碍,且不需要将jsx【本文受版权保护】【访问 www.tangshuang.net 获取更多精彩内容】编译的过程。

【原创不易,请尊重版权】著作权归作者所有,禁止商业用途转载。【未经授权禁止转载】【本文受版权保护】

另一种办法是走一遍编译,在小程序的页面目本文版权归作者所有,未经授权不得转载。【转载请注明来源】录下,创建一个页面同名的.jsx文件,再本文版权归作者所有,未经授权不得转载。【原创不易,请尊重版权】利用bebel将它编译为.js文件。但是本文作者:唐霜,转载请注明出处。【原创内容,转载请注明出处】这样的话,你需要在发布小程序的时候,忽略【转载请注明来源】未经授权,禁止复制转载。掉所有的.jsx文件。另外,还有一个坑是本文作者:唐霜,转载请注明出处。【原创不易,请尊重版权】,小程序的编译不提供process.en本文作者:唐霜,转载请注明出处。未经授权,禁止复制转载。v,所以编译react的结果用的时候会报本文作者:唐霜,转载请注明出处。转载请注明出处:www.tangshuang.net错。解决办法是把react的cjs/re转载请注明出处:www.tangshuang.net【版权所有,侵权必究】act.production.min.j【原创内容,转载请注明出处】【版权所有,侵权必究】s作为react的入口文件,通过小程序的【版权所有】唐霜 www.tangshuang.net【版权所有,侵权必究】构建npm的相关配置逻辑,指定react【作者:唐霜】【原创不易,请尊重版权】构建的文件。

【转载请注明来源】【作者:唐霜】【作者:唐霜】转载请注明出处:www.tangshuang.net本文作者:唐霜,转载请注明出处。

结语转载请注明出处:www.tangshuang.net

本文版权归作者所有,未经授权不得转载。【访问 www.tangshuang.net 获取更多精彩内容】【访问 www.tangshuang.net 获取更多精彩内容】【本文受版权保护】【版权所有】唐霜 www.tangshuang.net

本文详细讲解了如何在微信小程序中直接运行【原创不易,请尊重版权】【作者:唐霜】react组件的思路,同时,你可以参考这个仓库著作权归作者所有,禁止商业用途转载。,运行效果看看,研究它的整个实现过程。总转载请注明出处:www.tangshuang.net【版权所有】唐霜 www.tangshuang.net结而言,这个方法分为3个部分:1. 基于【版权所有】唐霜 www.tangshuang.net原创内容,盗版必究。react-reconciler实现一个【版权所有】唐霜 www.tangshuang.net【本文首发于唐霜的博客】把react组件渲染为纯js对象的渲染器【转载请注明来源】原创内容,盗版必究。,之所以需要纯js对象,是因为小程序发送【作者:唐霜】未经授权,禁止复制转载。到渲染线程的数据必须是纯对象。2. 利用【本文受版权保护】【未经授权禁止转载】小程序的自定义组件,实现自引用嵌套递归的【本文受版权保护】【本文首发于唐霜的博客】组件,用于利用上一步得到的js对象渲染成转载请注明出处:www.tangshuang.net【原创内容,转载请注明出处】真正的界面。3. 解决jsx问题,将前两【未经授权禁止转载】【原创不易,请尊重版权】步的结果,在page中进行实施,以真正完转载请注明出处:www.tangshuang.net【版权所有,侵权必究】成在小程序中渲染react组件的效果。当【本文首发于唐霜的博客】【访问 www.tangshuang.net 获取更多精彩内容】然,本文阐述过程,仅仅提供了这套思路,在【访问 www.tangshuang.net 获取更多精彩内容】【关注微信公众号:wwwtangshuangnet】真正用到项目中时,使用过程中肯定还会遇到原创内容,盗版必究。【版权所有】唐霜 www.tangshuang.net一些坑,仅能作为原有小程序开发项目的补充原创内容,盗版必究。著作权归作者所有,禁止商业用途转载。手段,比如之前写好的react组件不想重著作权归作者所有,禁止商业用途转载。【版权所有】唐霜 www.tangshuang.net新写成小程序版本,那么就可以使用这个方法原创内容,盗版必究。【转载请注明来源】,同时在渲染组件的地方,把DOM的标签,原创内容,盗版必究。著作权归作者所有,禁止商业用途转载。映射为小程序的标签,就可以在一定程度上解本文作者:唐霜,转载请注明出处。【本文受版权保护】决原有react代码复用的问题。如果你在本文版权归作者所有,未经授权不得转载。【访问 www.tangshuang.net 获取更多精彩内容】实操过程中遇到什么问题,欢迎在本文下方留【未经授权禁止转载】【访问 www.tangshuang.net 获取更多精彩内容】言讨论~

【版权所有,侵权必究】未经授权,禁止复制转载。本文版权归作者所有,未经授权不得转载。本文版权归作者所有,未经授权不得转载。

2021-06-19 7004 ,

为价值买单,打赏一杯咖啡

本文价值70.04RMB
已有1条评论
  1. 1188 2021-06-21 10:23

    刚好在学小程序, 参考学习下