Svelte 带来哪些新思想?赶紧学起来!

本文介绍

点赞 + 关注 + 收藏 = 学会了


Svelte 是我用过最爽的框架,就算 Vue 和 React 再强大,生态再好,我还是更喜欢 Svelte,因为它开发起来真的很爽。


其实在很久之前我就注意到 Svelte ,但一直没把这个框架放在心上。

因为我之前的工作主要使用 Vue,偶尔也会接触到一些 React 项目,但完全没遇到过使用 Svelte 的项目。

直到 Vite 的出现,我才开始开始重视 Svelte。


Vite文档 里可以看到它支持这些模板:

JavaScript TypeScript
vanilla vanilla-ts
vue vue-ts
react react-ts
preact preact-ts
lit lit-ts
svelte svelte-ts


能让祖师爷也重视的框架,不简单不简单~


我喜欢用 Demo 的方式学习新技术,Svelte 官方入门教程 就提供了这种方式。

这是我觉得入门比较舒服且方便日后搜索的学习方式。

虽然 Svelte 官方入门教程 已经给出很多例子,而且 Svelte中文网 也有对应的翻译,但有些翻译看上去是机译,而且部分案例可能不太适合新手学习~


本文的目的是把 Svelte 的学习流程梳理出来,让第一次接触 Svelte 的工友能顺利上手。


本文适合人群:有 HTMLCSSJS 基础,知道并已经安装了 Node

如果你是打算从0开始学习前端,那本文暂时还不适合你阅读。



Svelte 简介

Svelte 是一个构建 web 应用程序的工具。

传统框架如 React 和 Vue 在浏览器中需要做大量的工作,而 Svelte 将这些工作放到构建应用程序的编译阶段来处理。


需要注意,Svelte 是一款编译器。它可以将按照规定语法编写的代码打包成浏览器能运行的项目。

和其他前端框架一样,同样也是使用 HTMLCSSJavaScript 进行开发。


作者

在学习 Svelte 之前先了解一下它的父亲(作者)。

Svelte 的作者叫 Rich Harris ,正在吃东西的这位就是他。

可能国内大多数工友对他不是很熟悉(我也完全不熟),但应该听过 Rollup

没错,他也是 Rollup 的爸爸。

他在开发 Svelte 之前还开发过 Ractive.js ,听说 Vue 的部分实现也是受到了 Ractive 的启发。


关于 Rich Harris 的介绍还有很多,我搜到的资料上这样介绍到:

  • 大学专业是学哲学的
  • 在纽约时报调查组工作的图形编辑,身兼记者和开发者职位


还有更多关于他和 Svelte 的介绍,可以看看 《Svelte - The magical disappearing UI framework - Interview with Rich Harris》


Svelte 的优势

Svelte 翻译成中文就是“苗条”的意思,侧面表明它打包出来的包非常小。

Svelte 主要优势有以下几点。


1. 编译器

在打开Svelte官网时就能看到这样的介绍。

Svelte 是一种全新的构建用户界面的方法。传统框架如 React 和 Vue 在浏览器中需要做大量的工作,而 Svelte 将这些工作放到构建应用程序的编译阶段来处理。

Svelte 组件需要在 .svelte 后缀的文件中编写,Svelte 会将编写好的代码翻编译 JSCSS 代码。


2. 打包体积更小

Svelte 在打包会将引用到的代码打包起来,而没引用过的代码将会被过滤掉,打包时不会加入进来。

《A RealWorld Comparison of Front-End Frameworks with Benchmarks (2019 update)》 报告中,对主流框架进行了对比。

在经过 gzip 压缩后生成的包大小,从报告中可以看出,Svelte 打包出来的体积甩开 Vue、React 和 Angular 几条街。


这是因为经过 Svelte 编译的代码,仅保留引用到的部分。


3. 不使用 Virtual DOM

Virtual DOM 就是 虚拟DOM,是用 JS 对象描述 DOM 节点的数据,由 React 团队推广出来的。

虚拟DOM 是前端的网红,因此也有很多开发者开始研究和搞辩论赛。

网上有一张图对比了 Svelte 和 React 在数据驱动视图的流程

其实主要对比了使用虚拟DOM和直接操作真实DOM的区别。

在 React 中实现数据驱动视图大概流程是这样的:

数据发生变化 -> 通过diff算法判断要更新哪些节点 -> 找到要更新的节点 -> 更新真实DOM

Vue 的数据更新原理其实也差不多,只是实现方式和使用语法会有所不同。

diff算法 会根据数据更新前和更新后生成的虚拟DOM进行对比,只有两个版本的虚拟DOM存在差异时,才会更新对应的真实DOM。

使用虚拟DOM对比的方式会比直接对比真实DOM的效率高。

而且真实DOM身上挂载的属性和方法非常多,使用虚拟DOM的方式去描述DOM节点树会显得更轻便。


但这也意味着每次数据发生变化时都要先创建一个虚拟DOM,并使用 diff算法 将新虚拟DOM与旧虚拟DOM进行比对,这个步骤会消耗一点性能和需要一点执行时间。


而 Svelte 在未使用虚拟DOM的情况下实现了响应式设计。

我以粗暴的方式理解:Svelte 会监听顶层组件所有变量,一旦某个变量发生变化,就更新使用过该变量的组件。这就仅仅只需更新受影响的那部分DOM元素,而不需要整个组件更新。


综上所述,在我的理解力,虚拟DOM的思想很优秀,也是顺应时代的产物,但虚拟DOM并不是最快的,JS 直接操作 DOM 才是最快。


《Virtual DOM is pure overhead》 是 Svelte 官网上的一篇博客,专门讨论虚拟DOM。有兴趣的工友可以看看~


4. 更自然的响应式

这也是我刚接触 Svelte 时立刻喜欢上的理由。

这里说的响应式设计是只关于数据的响应,而不是像 Bootstrap 的响应式布局。


现在流行的前端框架基本都使用 数据驱动视图 这个概念,像 Vue 和 React 这些框架,都有响应式数据的概念。

但 Vue 和 React 在数据响应方面还是有点“不那么自然”,我简单举几个例子:

  • 在 React 中,如果需要更新数据并在视图中响应,需要使用 setState 方法更新数据。
  • 在 Vue2 中,响应式数据要放在 data 里,在 methods 中使用 this.xxx 来更新数据。
  • 在 Vue3 的 Composition API 语法中,需要使用 ref 或者 reactive 等方法包裹数据,使用 xxx.value 等方式修改数据。


上面这几种情况,感觉多少都添加了点东西才能实现响应式数据功能(至少在普通开发者开发时是这样)。


在 Svelte 的理念中,响应式应该给开发者一种无感体验,比如在 Excel 中,当我规定 C1 单元格的值是 A1 + B1 的和,设置好规则后,用户只需要修改 A1 和 B1 即可,C1 会自动响应,而不需再做其他操作。

在这方面,Svelte 我认为在现阶段是做得最自然的。

{name}

上面的代码中,1秒后修改 name 的值,并更新视图。

从代码就能看出,在使用 Svelte 开发项目时,开发者一般无需使用额外的方法就能做到和 Vue、React 的响应式效果。


如果你对 Svelte 响应式原理感兴趣,推荐阅读 FESKY《Svelte 响应式原理剖析 —— 重新思考 Reactivity》

也可以看看 《Rethinking reactivity》,看看官方对 reactivity 的思考。


5. 性能强

Stefan Krause 给出一份 性能测试报告(点击可查看) 对比里多个热门框架的性能。从 Svelte 的性能测试结果可以看出,Svelte 是相当优秀的。


6. 内存优化

性能测试报告(点击可查看) 也列出不同框架的内存占用程度,Svelte 对内存的管理做到非常极致,占用的内存也是非常小,这对于配置不高的设备来说是件好事。


第5和6点,由于测试报告比较长,我没截图放进文中。大家有兴趣可以点开链接查看测试报告

7. 更关注无障碍体验

在使用 Svelte 开发时会 自动对无障碍访问方面的体验进行检测,比如 img 元素没有添加 alt 属性,Svelte 会向你发出一条警告。无障碍体验对特殊人事来说是很有帮助的,比如当你在 img 标签中设置好 alt 属性值,使用有声浏览器会把 alt 的内容读出来。


在此我还要推荐2本关于设计体验的书。

  • 《点石成金:访客至上的Web和移动可用性设计秘笈》
  • 《包容性Web设计》

它们的封面长分别这个样子


Svelte 的优势肯定还有很多,但由于我开发经验不足,只能总结出以上这些了。如果你对 Svelte 有更多理解,欢迎在评论区补充~



Svelte 的不足

  1. Svelte 对 IE 是非常不友好的,但我并不把这放在眼里。如果想兼容 IE 我还是推荐使用 jQuery。
  2. Svelte 的生态不够丰富。由于是“新宠”,生态方面肯定是不如 Vue 和 React 的。


与 Svelte 相关的库

Sapper

Sapper 官网地址

Sapper 是构建在 Svelte 上的框架,Sapper 提供了页面路由、布局模板、SSR等功能。


Svelte Native

Svelte Native 官网地址

Svelte Native 是建立在 NativeScript 之上的产物,可以开发安卓和iOS应用,是一个跨端技术。

有点类似于 React Native 和 Weex 之类的东西。


svelte-gl

svelte-gl 仓库

svelte-gl 还没正式发布,但这是个很有趣的工具,它和 three.js 类似,专门做 3D应用的。

虽然现在 github 上的 Star 还不是很多,但也可以写些 demo 玩玩。



创建项目

在开始之前,你需要在电脑上安装 Node 环境。

编辑工具我使用了 VS Code ,同时安装了 Svelte for VS Code 扩展插件

使用 Svelte 前,必须有一个开发环境。

创建或使用开发环境有以下几种方式:

  1. REPL
  2. Rollup
  3. Webpack
  4. Parcel
  5. Vite

本文使用的是 Vite 创建项目,但上面列出的所有方式我都会逐一说说。


REPL

REPL 是 Svelte 提供的一个线上环境,打开 Svelte 官网 可以看到顶部导航栏上面有个 REPL 的选项。点击该选项就可以跳转到 Svelte 线上开发环境了。

REPLread(读取)evaluate(执行)print(打印)loop(循环) 这几个单词的缩写。


如果你只是想尝试 Svelte 的某些功能或者测试小型代码,可以使用这款线上工具。

REPL 还提供了多组件开发,按左上角的 +号 可以创建新组件。组件的内容稍后会说到。


界面右侧,顶部有3个选项:

  • Result: 运行结果。
  • JS output: Svelte 编译后的 JS 代码。
  • CSS output: Svelte 编译后的 CSS 代码。


REPL 界面右上角还有一个下载按钮。

当你在线上环境写好代码,可以点击下载按钮把项目保存到本地,下载的文件是一个 zip,需要自己手动解压。

然后使用以下命令初始化项目并运行即可。

# 1、初始化项目
npm install

# 2、运行项目
npm run dev

# 3、在浏览器访问 http://localhost:5000


运行结果:



Rollup 版

Svelte 官方也提供了一个命令,可以下载 Svelte 项目到本地。

命令最后需要输入你的项目名称。

# 1、下载模板
npx degit sveltejs/template 项目名称

# 2、安装依赖
npm install

# 3、运行项目
npm run dev

# 4、在浏览器访问 http://localhost:8080


运行结果:


这是官方提供的创建项目方式,这个项目是使用 Rollup 打包的。

Rollup 和 Svelte 都是同一个作者(Rich Harris )开发的,用回自家东西很正常。



Webpack 版

如果你不想使用 Rollup 打包项目,可以尝试使用 Webpack。

# 1、下载模板
npx degit sveltejs/template-webpack 项目名称

# 2、安装依赖
npm install

# 3、运行项目
npm run dev

# 4、在浏览器访问 http://localhost:8080/


运行结果:



Parcel 版

我并 不推荐使用 该方法创建项目,因为 Svelte 并没有提供使用 Parcel 打包工具的模板。但 GitHub 上有第三方的解决方案(点击访问仓库)

DeMoorJasper/parcel-plugin-svelte 的代码下载下来。

# 1、进入 `packages/svelte-3-example` 目录

# 2、安装依赖
npm install

# 3、运行项目
npm run start

# 4、在浏览器访问 http://localhost:1234/


运行结果:



Vite 版

本文接下来所有例子都是使用 Vite 创建 Svelte 项目进行开发的。

使用 Vite 创建项目的原因是:快!

# 1、下载模板的命令
npm init vite@latest

# 2、输入项目名

# 3、选择 Svelte 模板(我没选ts)

# 4、进入项目并安装依赖
npm install

# 5、运行项目
npm run dev

# 6、在浏览器访问 http://127.0.0.1:5173/


运行结果:


本文使用 Vite 创建项目,目录结构和 Rollup版 创建出来的项目结构稍微有点不同,但开发逻辑是一样的。



起步

index.htmlsrc/main.jssrc/App.svelte 这三个是最主要的文件。

index.html 是项目运行的入口文件,它里面引用了 src/main.js 文件。

src/main.js 里引入了 src/App.svelte 组件,并使用以下代码将 src/App.svelte 的内容渲染到 #app 元素里。

const app = new App({
  target: document.getElementById('app')
})

target 指明目标元素。


我们大部分代码都是写在 .svelte 后缀的文件里。

.svelte 文件主要保安 多个 HTML 元素1个 script 元素1个 style 元素 。这3类元素都是可选的。


我们主要的工作目录是 src 目录。

为了减轻学习难度,我们先做这几步操作。

1、清空全局样式

如果你使用 Rollup版 创建项目,不需要做这一步。


在使用 Vite 创建的 Svelte 项目中,找到 src/app.css 文件,并把里面的内容清空掉。


2、改造 src/App.svelte

src/App.svelte 文件改成以下内容



Hello {name}

此时点击按钮,页面上的 “雷猴” 就会变成 “鲨鱼辣椒”


上面的代码其实和 Vue 有点像。