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

注意:原文发表于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 >

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

Svelte 码半功倍的更多相关文章

  1. Qt之模式、非模式、半模式对话框

    简述 关于"模式"和"非模式"对话框,相信大家都比较熟悉,但其中有一个可能很多人都比较陌生,介于两者之间的状态,我们称之为"半模式". 简述 ...

  2. 矩阵(matrix)

    我们定义一个矩阵的权值为这个矩阵四个角上的数值的最小值.现在小M有一个矩阵,他想在这个矩阵中寻找到一个权值最大的子矩阵,请你告诉他这个最大权值.(距形规模最大为2000*2000) 比赛 看到第二题那 ...

  3. 越狱Season 1-Episode 21: Go

    Season 1, Episode 21: Go -Michael: I need you to let me get us out of here. 我需要你帮我出去 -Patoshik: If y ...

  4. ElasticSearch中文分词(IK)

    ElasticSearch常用的很受欢迎的是IK,这里稍微介绍下安装过程及测试过程.   1.ElasticSearch官方分词 自带的中文分词器很弱,可以体检下: [zsz@VS-zsz ~]$ c ...

  5. bugku misc writeup(一个普通的压缩包)

    这个题做了好几个小时,因为没有writeup,一点一点摸索,做题思路写出来给大家交流 首先这是一个zip.rar压缩包,下载下来第一步就是拖进hexeditor中观察,检查下文件的头尾结构是否有问题, ...

  6. mina statemachine解读(一)

      statemachine(状态机)在维护多状态数据时有非常好的作用,现在github上star排名最前的是squirrel-foundation以及spring-statemachine,而min ...

  7. 【题解】Luogu P4054 [JSOI2009]计数问题

    原题传送门 我自闭了qaq 这道题非常简单,因为1<=c<=100,所以直接对每个c开二维树状数组,操作就跟模板一样 写码5分钟,调码半小时,这道题的输入顺序是x1,x2,y1,y2,我真 ...

  8. 到底什么是dp思想(内含大量经典例题,附带详细解析)

    期末了,通过写博客的方式复习一下dp,把自己理解的dp思想通过样例全部说出来 说说我所理解的dp思想 dp一般用于解决多阶段决策问题,即每个阶段都要做一个决策,全部的决策是一个决策序列,要你求一个 最 ...

  9. 抽丝剥茧:理解Android权限机制

    前一段时间面试官问我Android在Linux的基础上,权限做了哪些改变.霹雳呱啦说了一堆,但是说着说着,始终感觉自己说的缺了点东西,自己理解还是不够到位,而且网上的很多文章在原理上基本都是大同小异, ...

随机推荐

  1. WPF 之 依赖属性与附加属性(五)

    一.CLR 属性 ​ 程序的本质是"数据+算法",或者说用算法来处理数据以期得到输出结果.在程序中,数据表现为各种各样的变量,算法则表现为各种各样的函数(操作符是函数的简记法). ...

  2. 初窥 Python 的 import 机制

    本文适合有 Python 基础的小伙伴进阶学习 作者:pwwang 一.前言 本文基于开源项目: https://github.com/pwwang/python-import-system 补充扩展 ...

  3. [The Preliminary Contest for ICPC Asia Shanghai 2019] B-Light bulbs(差分+思维)

    前言 最近有很多算不上事的事,搞得有点心烦,补题难免就很水,没怎么搞,自我检讨一番~~ 说实话网络赛题目的质量还是挺高的,题目都设计的挺好的,很值得学习.这场比赛那会只有我们大二的在做,其他人去参加$ ...

  4. 2019牛客暑期多校训练营(第六场)D-Move

    >传送门< 题意: 你有n件行李,有k个箱子体积相同的箱子,遵循下面的规则将行李放进箱子里面 每次都取当前最大的可以放进箱子的行李放进箱子,如果该箱子放不进任何行李那么就换一个新的箱子再按 ...

  5. Luogu T9376 区间GCD

    题目背景 无 题目描述 给定一长度为n的动态序列,请编写一种数据结构,要求支持m次操作,包括查询序列中一闭区间中所有数的GCD,与对一闭区间中所有数加上或减去一个值. 输入输出格式 输入格式: 第1行 ...

  6. AcWing 247. 亚特兰蒂斯 (线段树,扫描线,离散化)

    题意:给你\(n\)个矩形,求矩形并的面积. 题解:我们建立坐标轴,然后可以对矩形的横坐标进行排序,之后可以遍历这些横坐标,这个过程可以想像成是一条线从左往右扫过x坐标轴,假如这条线是第一次扫过矩形的 ...

  7. Codeforces Round #683 (Div. 2, by Meet IT) D. Catching Cheaters (DP)

    题意:给你两个字符串,每次取它们的子串C和D,然后求LCS,得到的贡献为\(4*LCS(C,D)-|C|-|D|\),求最大贡献. 题解:首先应该了解\(O(n^2)\)的LCS的dp写法,然后在此基 ...

  8. JavaScript预编译过程理解

    1-JavaScript运行三部曲 语法分析 预编译 解释执行 语法分析很简单,就是引擎检查你的代码有没有什么低级的语法错误: 解释执行顾名思义便是执行代码了: 预编译简单理解就是在内存中开辟一些空间 ...

  9. 使用nodejs爬取图片

    在运行代码前,请确保本机是否有nodejs环境 1 D:\ > node -v 2 v12.1.0 //版本号 需要用到的包 axios //请求页面 cheerio // 把get请求的页面 ...

  10. cs实时系统之网关设计

    今天给大家讲一下client-server系统(cs)设计,基本结构 1.client 客户端,插件式开发,负责对应ui的展示 2.gateway 网关层,管理客户端通信连接,负载后端集群服务 3.s ...