让你的react代码跑在svelte引擎上

简介: 借助ast魔法让现有react业务接近无成本收获svelte带来的收益

作者:闲鱼技术——羲凡

背景

Svelte UI框架是一款类似react、vue一样的UI框架,有兴趣的同学可以自行查阅。Svelte有着很多不一样的特质,其中我们最为关注的是它的运行前编译。像更简洁灵活的响应式写法、更小的运行包体积这些都得益于运行前编译;但同样也存在它的局限性,Svelte无法像React一样做到高度灵活的模板嵌套、封装以及独立上下文。早期在小项目中我们也体验过,在包体积上收益颇大;但目前整个淘系前端还是有着自己统一的一套基于React的Rax framework,如果说需要将Svelte的收益快速落地最好的方式肯定不是重构所有的业务与生态。

主体思路

要实现dsl层无感换引擎也就只有在ast层做转换,所以我们从开始的思路就是React ast -> Svelte ast,为了保证这一思路的清晰我们暂且把框架的特性(hook、函数jsx)丢掉,保留最干净的三部分jsx、js、css进行转换。主体思路如下
image.png

JSX转换

jsx可以说是这两个UI框架最大差异的体现;React的jsx是运行时,原汁原味保留着原生js运行时的灵活性,而Svelte的html是运行前编译,意味着运行前就会敲定html中的逻辑与依赖变量。我暂且将jsx本身拆分为三部分:element节点attribute属性mustache模板系统函数JSX

element与attribute

这部分主要是常规的节点属性映射能力
image.png

mustache模板系统

在React的模板系统中可以有以下几种可能:
1、变量输出 { xxx }
2、与运算jsx输出 { xx && <view> }
3、三元运算jsx输出 { xx ? <n> : <m> }
4、函数调用jsx输出 { jsxFn() }
其中第1种情况不需要处理,Svelte的模板系统也是能够支持常规的变量输出;而第4种属于函数jsx;第2、3种可以转换成Svelte提供的{#if ...}语法糖。
image.png
至此,简单jsx的ast就能编译成svelte html的ast。

函数JSX

函数jsx背后折射的是js运行时,意味着可以封装独立的运行上下文(scope),可以做到环境变量隔离。先来看看一个常规的函数jsx,再进一步抽象成三部分:上下文逻辑控制JSX
image.png
其中JSX部分的转换可以复用上文讲到的element与attribute部分。我们重点来看上下文逻辑控制的转换。

上下文(scope)

尝试使用函数模拟出完整的执行上下文
image.png
优化调用次数
image.png

逻辑控制

这里我指的逻辑控制是控制jsx输出的逻辑(非jsx输出的控制逻辑不需要关心),在一个jsx函数中,逻辑控制语句可有、可无、可组合、可嵌套,在js中大致常用的逻辑控制语句大致有以下几种:
1、if else
2、switch case
3、for/while/do while
4、return is ? <viewa> : <viewb>
5、return is && <view>
根据Svelte提供的可编译html逻辑控制语法糖来做个映射:
image.png
逻辑控制语句转换过程中较为麻烦的是通盘的识别,例如下面例子,我们怎么识别出 if (!gVari3)if (!gVari4) 是属于jsx的逻辑控制语句?这件事就好比这段代码交给一个开发者来重构,他是怎么判别的;所以为了让结果准确必须拉通全盘来识别。
image.png

转换大致结果

image.png
看着产物有些恐怖,极其膨胀,但这也给后面优化留够了空间。

JS转换

js层面的差异主要来自两方面:hook体系(不考虑类)、ecma ast差异

hook体系

hook体系的api更多是为了纯函数组件注入状态与生命周期的,在这两点上Svelte给出的方案是截然不同的,得益于运行前编译,Svelte编译器扫描了所有与UI相关的状态并注入黑魔法,使得状态的使用就是变量申明赋值那么简单,基本上开发者不需要太关心所谓的副作用;所以有一些hook接口在Svelte框架上显得是有些多余的。
但鉴于hook接口较多,我们在转换过程选择了内置svelte-hooks来简化转换逻辑,svelte-hooks是基于svelte且对标react hook来实现的一套hook接口,在使用上基本保持一样。
image.png

ecma ast差异

babel提供的parse是基于estree,但同时在基础上细化出一些类型,具体可以差异可以参考这里babel-parser,细化的这些数据类型在我们做转换推断有一定帮助,所以我们并没有使用babel提供estree插件,而且在转换之后的ast再进行一次差异抹平。
image.png

CSS转换

css转换相比上面两部分的转换就要简单很多,React的样式是标准的css,Svelte编写的样式也是标准css,不过会在基础上增加了一定的编译能力,可以理解是标准css的超集,可以直接使用。不过为了抹平jsx与Svelte html在自定义组件的类选择器上的差异,我们还是在编译阶段做了一些转换,这里就不展开了。大致流程如下
image.png

结果

一个重构过的业务(只含UI相关逻辑)
image.png
细化后的体积差异
image.png
客观对比:编译产物是要比正常打包要大的,从前面的函数jsx编译思路来看是要膨胀的,在预期内;拉开差距的是framework与依赖的UI Component,Rax是集团内部基于react进行简化后的framework,如果使用React framework做比较这个差距会进一步的拉大;而UI Component这部分的差异是得益于我们参照运行前编译的思路重新设计了一套可在编译阶段根据具体使用情况来按需打包的轻量UI组件(本文没有具体展开)。

后续

1、将运行时的jsx百分百准确编译成静态的html是不可能的,弱类型语言的变量追踪是不可靠的,非原生逻辑控制语法也无法一一在编译器中枚举;目前在转换工作中还遗留很多编译相关问题,但这些问题可以通过一些插件来补充以致逐渐完善。
2、在大型项目中包体积的现状并不乐观,Svelte借助着运行前编译将整个framework按需打包可以有效减少包体积,但编译产物本身是没有优势的,当一个页面UI交互越复杂,编译产物会越大,加上对framework依赖越多,整体包体积的优势就荡然无存了;除此之外,我们转换器为了抹平差异又给编译增加了一定复杂度,所以在编译产物体积的优化上还存在很大的空间。
3、无性能不前端,在性能方面的数据我们还是缺失的,但也从一些三方文章了解到整体Svelte性能并不是瓶颈,而且从理论上来说借助编译来实现数据驱动DOM是简洁高效的,脱离Virtual DOM理论上也让内存表现更优异;但我们还是会单独去看性能一块的具体表现。
4、运行前编译的思路不单止应用于framework,component同样受用,而且收益颇大。

相关文章
|
7月前
|
监控 前端开发 API
React代码在电脑监控软件开发中的应用
这篇文章除了介绍React在电脑监控软件开发中的应用,还展示了如何通过组件化构建监控界面,如`MonitorDashboard`、`StatusWidget`和`ActivityLog`组件。文章强调了React的生命周期方法和Hooks在实时数据获取和显示中的作用,如`SystemStatus`组件的`useEffect`钩子。此外,还讲解了如何监听和记录用户行为的`UserActivity`组件,以及利用axios自动提交监控数据的`DataSubmitter`组件。React的这些特性使得开发高效、响应式的监控软件变得更加便捷。
102 8
|
8月前
|
前端开发 JavaScript API
React API 和代码重用的演变!
React API 和代码重用的演变!
|
2月前
|
资源调度 前端开发 Android开发
如何在 React Native 中进行代码签名验证?
如何在 React Native 中进行代码签名验证?
|
3月前
|
JavaScript 前端开发 算法
前端优化之超大数组更新:深入分析Vue/React/Svelte的更新渲染策略
本文对比了 Vue、React 和 Svelte 在数组渲染方面的实现方式和优缺点,探讨了它们与直接操作 DOM 的差异及 Web Components 的实现方式。Vue 通过响应式系统自动管理数据变化,React 利用虚拟 DOM 和 `diffing` 算法优化更新,Svelte 通过编译时优化提升性能。文章还介绍了数组更新的优化策略,如使用 `key`、分片渲染、虚拟滚动等,帮助开发者在处理大型数组时提升性能。总结指出,选择合适的框架应根据项目复杂度和性能需求来决定。
|
4月前
|
前端开发 JavaScript 区块链
react18函数组件+antd使用指南-使用代码集合以及报错记录汇总
本文介绍了多个React开发中常见的问题及其解决方案,包括但不限于:1)`useForm`实例未连接到任何`Form`元素的警告及解决方法;2)监听页面滚动事件的实现方式;3)React 18与antd 5.8.6中定制主题的方法;4)React结合antd 4.x版本自定义主题色的步骤;5)解决`ResizeObserver loop`相关报错的技巧;6)处理React设计表单时遇到的CDN资源加载失败问题;7)解决onClick事件传参问题;8)修复类型错误等。每部分均提供详细分析与实用代码示例,帮助开发者快速定位并解决问题。
74 2
|
5月前
|
开发者 搜索推荐 Java
超越传统:JSF自定义标签库如何成为现代Web开发的个性化引擎
【8月更文挑战第31天】JavaServer Faces(JSF)框架支持通过自定义标签库扩展其内置组件,以满足特定业务需求。这涉及创建`.taglib`文件定义标签库及组件,并实现对应的Java类与渲染器。本文介绍如何构建和应用JSF自定义标签库,包括定义标签库、实现标签类与渲染器逻辑,以及在JSF页面中使用这些自定义标签,从而提升代码复用性和可维护性,助力开发更复杂且个性化的Web应用。
84 0
|
5月前
|
前端开发 测试技术 UED
React性能优化的神奇之处:如何用懒加载与代码分割让你的项目一鸣惊人?
【8月更文挑战第31天】在现代Web开发中,性能优化至关重要。本文探讨了React中的懒加载与代码分割技术,通过示例展示了如何在实际项目中应用这些技术。懒加载能够延迟加载组件,提高页面加载速度;代码分割则将应用程序代码分割成多个块,按需加载。两者结合使用,可以显著提升用户体验。遵循合理使用懒加载、编写测试及关注性能等最佳实践,能够更高效地进行性能优化,提升应用程序的整体表现。随着React生态的发展,懒加载与代码分割技术将在未来Web开发中发挥更大作用。
60 0
|
5月前
|
前端开发 UED 开发者
React.lazy()与Suspense:实现按需加载的动态组件——深入理解代码分割、提升首屏速度和优化用户体验的关键技术
【8月更文挑战第31天】在现代Web应用中,性能优化至关重要,特别是减少首屏加载时间和提升用户交互体验。React.lazy()和Suspense组件提供了一种优雅的解决方案,允许按需加载组件,仅在需要渲染时加载相应代码块,从而加快页面展示速度。Suspense组件在组件加载期间显示备选内容,确保了平滑的加载过渡。
194 0
|
5月前
|
XML 前端开发 JavaScript
【React新手必看】JSX,让你的代码舞动起来!
【8月更文挑战第24天】JSX(JavaScript XML)是React开发的核心,作为一种JavaScript语法扩展,它支持在代码中使用类似HTML的标签。JSX并非模板语言,在编译过程中转换为纯JavaScript,使React能高效更新DOM并渲染页面。其优势在于直观性、简洁性和灵活性:简化复杂用户界面构建、减少代码量,并支持HTML标签、React组件及JavaScript变量的表达式插入。基本用法涉及定义组件UI,如创建包含标题和段落的简单组件。
104 0
|
6月前
|
前端开发
React官网 - 井字棋 - 游戏改进参考答案(含汉化和完整范例代码)
React官网 - 井字棋 - 游戏改进参考答案(含汉化和完整范例代码)
55 1
下一篇
开通oss服务