Svelte 码半功倍

你未注意到的最重要的指标。

注意:原文发表于2019-04-20,随着框架不断演进,部分内容可能已不适用。

所有代码都有 BUG,你写的越多,BUG 越多,这很合情合理。

同时,写的越多,费时越多,留给其他事情的时间就更少,比如代码优化、功能完善或者去户外潇洒而不是蜷缩在笔记本面前。

其实众所周知,项目研发时间BUG 的数量,会随着代码库的膨胀呈二次增长,而非线性增长。

这也与我们的直觉相吻合:一个只有 10 行代码的 Pull Request 和 100 行的对比,其适用的审查级别就有所不同。

一旦某个模块肥胖到单屏无法显示完整的话,那么理解这个模块所需的认知和背负的压力便会剧增。然后我们只能通过重构和添加注释(几乎总是导致写更多的代码)来补偿。

这是一个恶性循环。

我们沉溺在性能指标、打包大小以及其他任何可以衡量的指标。

唯独很少关注代码量。

可读性至关重要

我的本意当然并非是将代码压缩为近乎紧凑形式的这类不惜以牺牲可读性为代价的技巧,也不是说千方百计减少代码行数就是可取的目标。

因为这样相当于鼓励原本还算可读的代码 ……

for (let i = 0; i <= 100; i += 1) {
  if (i % 2 === 0) {
    console.log(`${i} is even`);
  }
}

硬生生写成难以理解的代码:

for (let i = 0; i <= 100; i += 1) if (i % 2 === 0) console.log(`${i} is even`);

恰恰相反,我主张的是,我们应该倾向于那些能够让我们自然而然就可以写出更少代码的语言和模式。

没错,我要吹爆 Svelte

Svelte 致力减少你必须编写的代码量。

为了讲清楚这一点,我们对比一下分别用 React、Vue 和 Svelte 三者来实现的一个非常简单的组件。

先来看看 Svelte 的版本:

<script>
  let a = 1;
  let b = 2;
</script>

<input type="number" bind:value={a}>
<input type="number" bind:value={b}>

<p>{a} + {b} = {a + b}</p>

然后看看 React 版本是什么样的,它可能是这样写的:

import React, { useState } from 'react';

export default () => {
  const [a, setA] = useState(1);
  const [b, setB] = useState(2);

  function handleChangeA(event) {
    setA(+event.target.value);
  }

  function handleChangeB(event) {
    setB(+event.target.value);
  }

  return (
    <div>
      <input type="number" value={a} onChange={handleChangeA}/>
      <input type="number" value={b} onChange={handleChangeB}/>

      <p>{a} + {b} = {a + b}</p>
    </div>
  );
};

最后是使用 Vue 实现的同等功能的版本:

<template>
  <div>
    <input type="number" v-model.number="a">
    <input type="number" v-model.number="b">

    <p>{{a}} + {{b}} = {{a + b}}</p>
  </div>
</template>

<script>
  export default {
    data: function() {
      return {
        a: 1,
        b: 2
      };
    }
  };
</script>

换而言之,同等功能的组件,用 React 来实现,你需要痛击 442 个字符,Vue 是狠打 263 个字符,而 Svelte 只消轻敲 145 个字符。(我是复制源码到粘贴板,然后在终端运行 pbpaste | wc -c 得到的计数结果)。

React 版本足足胖了三倍!

差异是如此明显,反而揭示了它的不同寻常,依据我的经验来看,React 组件普遍比同等功能的 Svelte 组件大 40% 左右。

接下来我们看看 Svelte 的设计特点,它可以帮助开发者更清晰地表达想法:

顶级元素

在 Svelte 中,一个组件的顶级元素可以有任意多个,随心所欲。

而 React 或 Vue,组件必须只能有一个顶级元素。如果尝试在 React 中的组件函数想任性地一次返回两个顶级元素,就会导致语法无效。(你可以使用 <> 代替 <div>,其实道理还是一样的嘛,还会导致额外多一层缩进呢)

在 Vue 中,标记必须放在 <template> 元素中,是不是有点多此一举了。

绑定

在 React 中,要响应输入事件你需要亲力亲为:

function handleChangeA(event) {
  setA(+event.target.value);
}

这么干除了霸占了更多的显示器屏幕空间外,还为 BUG 提供充分的滋养环境。

按理论来讲,输入的值需要自动绑定到 a,反之亦然。不过可惜,这种关系你是不容易清晰地表达出来的。

我们反而有两个紧密耦合在一起 —— 但物理上却是若即若离的代码块(事件处理程序和 value={a} 属性)。

漏屋偏逢连夜雨,你还必须记得使用前置的 + 号来强制转换字符串为数字,否则 2+2 就得到 22 而 不是 4 了。

与 Svelte 一样,Vue 确实提供了一种绑定表达式:v-model 属性,我们仍然需要在使用 v-model.number 时小心谨慎,就算它专门用于数字输入。

状态

在 Svelte 中,可以直接使用赋值运算符来更新组件的状态:

let count = 0;

function increment() {
  count += 1;
}

而在 React 中,我们使用 useState 这个 Hook:

const [count, setCount] = useState(0);

function increment() {
  setCount(count + 1);
}

这看着是多么的拖沓冗长、絮絮叨叨 —— 其实它俩要表达的意思是一模一样的,字符数却相差了近 60%。

阅读源码的时候,就需要消耗你更多的脑细胞来理解作者的意图。

而 Vue 则使用 default export 的方式导出一个 data 函数,该函数返回一个对象字面量,其属性与我们局部变量是相对应的。

像 helper 函数和子组件这类东西,你不能简简单单地导入并在模板中使用,而是必须通过将它们放置到 export default 合适的位置来“注册”一下。

样板代码之殇

以上只是管中窥豹,不过是 Svelte 帮助你在构建用户界面时减少麻烦的冰山一角。

咱还有很多杀手锏的 —— 你比方说响应式的声明,实质上 Svelte 无须显山露水,就已完美替代了 React 的 useMemouseCallbackuseEffect 这些样板代码(或者那些在每次状态变化时创建的内联函数和数组的垃圾被回收的开销)。

这是咋弄的?

其实不过是一些与众不同的约束的抉择罢了。

因为 Svelte 实际上是一个编译器,因此不必拘泥于 JavaScript 的特性:我们可以为如何方便编写一个组件量身定制一个创作体验的过程,而无需百般逢迎 JavaScript 的语义。

这更符合编码的习惯,例如自然而然会使用变量而不是那些代理或者 Hooks,同时还能生成更高性能的程序。

鱼与熊掌,孰能兼得?


< The End >

- 窗明几净,静候时日变迁 -

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

推荐阅读更多精彩内容