0.Intro

This article will introduce you to the front ring “new” pet Svelte, and the responsive principle behind it. You haven’t used Svelte yet, but chances are you’ll hear the name in some tech weekly, community, or front-end annual report. If you use Digg to write articles, you are already using Svelte, because the new digg editor [Bytemd][2] is written using Svelte 👀.

(: Svelte may not be a new thing for those who have a wide range of information sources, because its construction started as early as 2016, and I am behind.

1. What is Svelte?

It’s a front end frame by Rich Harris, the guy in wheels, and you’re probably not familiar with the word, but Rollup is, of course, the same author.

The new frame means learning a new grammar, like a new “language” every few months, which made me want to post that old picture.

Ridicule return ridicule, should learn or want to learn, otherwise will be eliminated 👻. The main features of Svelte are:

  1. Use basic HTML,CSS,Javascript to write code
  2. Directly compiled into native JS, no middleman (Virtual DOM) to earn price difference
  3. No sophisticated state management mechanism

2. Frame comparison

The React, Vue, Angular, and Svelte frameworks need to be based on a number of facts, including Star count, download trends, code volume, performance, and user satisfaction.

React Vue @angular/core Svelte
Number of Star 🌟 168661 183540 73315 47111
Code volume 🏋️♀️ 42k 22k 89.5 k. 1.6 k.

In terms of Star number, Svelte is only one quarter of Vue (YYDS) (Svelte (2016) started three years slower than Vue (2013)). But 4.7W Star number is not low.

Svelte is only 1.6K on minizipped!! Don’t forget that Another rollup is good at package optimization. However, as the project code increases and more functionality is used, the volume of Svelte compiled code increases faster than other frameworks, which will be discussed later.

NPM download trends

Npm Trendings link direct

The download gap was significant, with Svelte(231,262) only 2% of React(10,965,933). It’s not enough to just look at the surface data. Let’s run it.

Benchmark run points

The higher the green, the higher the score. As you can see from the figure above, Svelte performs quite well in terms of performance, size, and memory footprint. Look at customer satisfaction.

Customer satisfaction

Again, Svelte is number one!! (Interest is also an Interest).

Preliminary conclusions

Through the comparison of the above data, we can roughly come to the conclusion that the Svelte code is small in size, with excellent performance, promising in the future and worth further study.

3.Svelte basic grammar

Class Vue writing

<script>
  let count = 0;

  function handleClick() {
    count += 1;
  }
</script>

<button on:click={handleClick}>
  Clicked {count} {count === 1 ? 'time' : 'times'}
</button>

<style>
  button {
    color: black;
  }
</style>
Copy the code

A.svelte file contains JS, HTML, CSS, and uses similar template syntax, instruction binding.

The difference may be that Style is scoped by default, HTML does not need to be wrapped with
, and initialization steps without new Vue, data. Just define a variable, just use it. (What happens behind this will be covered in the Reactivity section.)

Written as Vue:

var vm = new Vue({
  data: {
    count: 0}})Copy the code

The magic of$:grammar

You need to trigger computations when you rely on data changes, which in Vue is typically done using computed.

var vm = new Vue({
  data: {
     count: 0
  },
  computed: {
    double: function () {
      // 'this' points to the VM instance
      return this.count * 2}}})Copy the code

We have a similar implementation in Svelte, where we use the $: keyword to declare computed variables.

<script>
  let count = 0;

  function handleClick() {
    count += 1;
  }
	
  $: double = count * 2
</script>

<button on:click={handleClick}>
  Clicked {double} times
</button>
Copy the code

In the example above, every time the button is clicked, the double is recalculated and updated to the DOM Tree. What kind of dark technology is this? Is it native JS code?

The Statements and declarations syntax is used, which can be preceded by any legal variable character, and defines a goto statement. But the semantics are different. Here Svelte just uses the deprecated syntax to declare computational properties (still native JS syntax, 👻 does not introduce black art.

We have everything we need

As a front-end framework, Svelte should also have many functions, such as template syntax, conditional rendering, event binding, animation, component lifecycle, Context, and even other frameworks do not have it, such as Store, Motion, etc. There are many, because these apis are not expensive to learn, Just look at the code when you use it.

Next comes the core of this article, how Svelte implements Reactivity, or data-driven views, compared to Vue and React.

4.Reactivity

Reactivity is what?

In high school chemistry class, we have been exposed to many experiments, such as the use of purple litmus test to identify acids and bases. An acid can turn a purple litmus solution red and an alkali can turn a purple litmus solution blue. The principle of the experiment is related to the molecular structure. The molecular structure is the link, the addition of acid/base is the action, and the result of the change of the molecular structure is the reaction Reactivity.

Taking advantage of Reactivity is often a great way to do things, such as function calculations in Excel/Number.

SUM(D10, E10) =SUM(D10, E10) =SUM(D10, E10) =SUM(D10, E10)

How did you write code before Reactivity?

To get a better sense of Reactvity’s impact on coding, imagine creating a Todo app that adds tasks, displays a list of tasks, deletes tasks, switches the DONE state of a task, and so on.

First, you need to maintain a list of tasks data.

const tasks = [
    {
        id: 'id1'.name: 'task1'.done: false}]Copy the code

Use DOM manipulation to traverse the list and render it.

function renderTasks() {
  const frag = document.createDocumentFragment();
  tasks.forEach(task= > {
    // Omit the rendering details for each task
    const item = buildTodoItemEl(task.id, task.name);
    frag.appendChild(item);
  });

  while (todoListEl.firstChild) {
    todoListEl.removeChild(todoListEl.firstChild);
  }
  todoListEl.appendChild(frag);
}
Copy the code

Then every time a new/deleted/modified task is added/deleted/modified, in addition to modifying the Tasks data, the task needs to be manually triggered to re-render tasks (of course this implementation is not good, as deleting/inserting too many DOM nodes can cause performance problems).

function addTask (newTask) {
    tasks.push(newTask)
    renderTaks()
}

function updateTask (payload) {
    tasks = / /...
    renderTaks()
}

function deleteTask () {
    tasks = / /...
    renderTaks()
}
Copy the code

Notice that every time we modify the data, we need to manually update the DOM to synchronize the UI data. (This is exactly what we did back in the jQuery era, it was expensive to develop and got out of control with too many dependencies.)

With Reactvity, the developer only needs to modify the data, leaving UI synchronization to the Framework, freeing developers from DOM manipulation.

// vue
this.tasks.push(newTask)
Copy the code

Before explaining how Svelte implements Reactivity, let’s briefly explain how React and Vue do it.

The realization of the React

React developers write their code using JSX syntax. JSX is compiled into ReactElement, which generates an abstract Virtual DOM at runtime.

Then, on each render, React re-compares the Virtual DOM before and after and does nothing if no update is required. If only the HTML attribute change is reflected in the DOM node, the setAttribute method of the node is called. If the DOM type changes, the key changes, or is not found in the new Virtual DOM, the corresponding DOM deletion/addition operation is performed.

In addition, the benefits of abstracting the Virtual DOM include cross-platform rendering and testing, such as react-native and React-Art.

Using Chrome Dev Tool’s Performance panel, let’s take a look at what React does behind a simple click-counting DEMO.

import React from "react";

const Counter = () = > {
  const [count, setCount] = React.useState(0);

  return <button onClick={()= > setCount((val) => val + 1)}>{count}</button>;
};


function App() {
  return <Counter />;
}

export default App;
Copy the code

The whole process can be roughly divided into three parts. The first is the scheduler, which handles priority (user click events are of high priority) and composable events.

The second part is the Render phase, which mainly iterates through nodes, finds the Fiber Node to be updated, performs the Diff algorithm to calculate the type of operation to be performed, marks the effectTag, and generates a linked list of Fiber nodes with effectTag. What is often called asynchronous interruptibility also occurs at this stage.

The third step is Commit, which iterates through the linked list generated in step 2 and performs the corresponding operations (add, delete, modify…).

So for our simple example React also has a lot of front-end work to do. The actual DOM modification is in the red box.

Firstchild.nodevalue = text; firstchild.nodeValue = text;

The React version used for the demo is 17.0.2, with Concurrent Mode enabled.

Use shouldComponentUpdate, memo, and all descendant nodes of the React node. UseCallback, useMemo and other methods for optimization.

The realization of the Vue

After writing for a long time, I found that I haven’t written the key point yet. In order to control the length of the Demo will not write (introduction of Vue responsive principle of the article is very many).

The general process is to collect dependencies during compilation, based on the getters for Proxy (3.x), defineProperty(2.x), and the setter implementation notifies Watcher when data changes. The implementation of Vue is cool, and every time you modify data on data, it’s like magic.

5. Svelte dimension reduction

Both React and Vue do a little more to achieve their purpose (a smaller data-driven UI) (Vue also uses the Virtual DOM). How does Svelte reduce runtime code?

The secret is in Compiler, where most of the work is done at compile time.

Core Compiler

Svelte source code is divided into compiler and Runtime.

How does Compiler collect dependencies? Dependencies in code can be analyzed at compile time, such as rendering a {name} field in a template. If name changes at some point (such as after a button is clicked), try to trigger an update view every time name is assigned. If the name will not be modified, do nothing.

This article will not explain how the Compiler is implemented, but let’s see what the code looks like with the Compiler.

<script>
  let name = 'world';
</script>

<h1>Hello {name}!</h1>
Copy the code

It will be compiled into the following code, but I have temporarily deleted the irrelevant code for easier understanding.

/* App.svelte generated by Svelte v3.38.2 */
import {
  SvelteComponent,
  append,
  detach,
  element,
  init,
  insert,
  listen,
  noop,
  safe_not_equal,
  set_data,
  text
} from "svelte/internal";

function create_fragment(ctx) {
  let h1;

  return {
    c() {
      h1 = element("h1");
      h1.textContent = `Hello ${name}! `;
    },
    m(target, anchor){ insert(target, h1, anchor); }}let name = "world";
Copy the code

The create_fragment method is a method associated with the DOM results of each component and provides some DOM hook methods, described in the next section.

Compare the code if the variable would be modified

<script>
    let name = 'world';
    function setName () {
        name = 'fesky'
    }
</script>

<h1 on:click={setName}>Hello {name}!</h1>
Copy the code

The compiled

import {
 SvelteComponent,
 append,
 detach,
 element,
 init,
 insert,
 listen,
 noop,
 safe_not_equal,
 set_data,
 text
} from "svelte/internal";

function create_fragment(ctx) {
  let h1;
  let t0;
  let t1;
  let t2;
  let dispose;

  return {
    c() {
      h1 = element("h1");
      t0 = text("Hello ");
      t1 = text(/*name*/ ctx[0]);
      t2 = text("!");
    },
    m(target, anchor) {
      insert(target, h1, anchor);
      append(h1, t0);
      append(h1, t1);
      append(h1, t2);

      if(! mounted) {// Added binding events
        dispose = listen(h1, "click"./*handleClick*/ ctx[1]); }},// Add a p (update) method
    p(ctx, [dirty]) {
      if (dirty & /*name*/ 1) set_data(t1, /*name*/ ctx[0]); }}; }// add the instance method
function instance($$self, $$props, $$invalidate) {
  let name = "world";

  function setName() {
    $$invalidate(0, name = "fesky");
  }

  return [name, setName];
}
Copy the code

In this case, there is more code to compile. The first is that the original m method in the fragment added the click event. There is a p method that calls set_data; New instance method that returns the properties that exist in each component instance and the methods that modify those properties (name, setName), if any other properties and methods are returned in the same array (not to be confused with Hooks).

It doesn’t matter if you don’t know the details, they will be introduced later. $$invalidate(0, name = “fesky”);

Remember earlier when we implemented the Todo List in native code? We manually re-render the DOM after each data modification! We don’t recommend it because it’s hard to maintain.

function addTask (newTask) {
    tasks.push(newTask)
    renderTaks()
}
Copy the code

Svelte Compile actually does this for us at the Compile stage! Analyze all the things that need to be done after the data changes to generate native JS code, so that the runtime does not need the runtime code like Vue Proxy.

Selve provides an online real-time compiler that you can try with your hands. Svelte. Dev/repl/hello -…

The next section will look at how Svelte as a whole runs from a source code perspective.

Fragment — both are pure DOM operations

Each Svelte component compiles with a create_fragment method that returns the declaration cycle hook method for some DOM nodes. Are a single letter is difficult to understand, from the source can see the meaning of each abbreviation.

interface Fragment {
  key: string|null;
  first: null;
  /* create */ c: () = > void;
  /* claim */ l: (nodes: any) = > void;
  /* hydrate */ h: () = > void;
  /* mount */ m: (target: HTMLElement, anchor: any) = > void;
  /* update */ p: (ctx: any, dirty: any) = > void;
  /* measure */ r: () = > void;
  /* fix */ f: () = > void;
  /* animate */ a: () = > void;
  /* intro */ i: (local: any) = > void;
  /* outro */ o: (local: any) = > void;
  /* destroy */ d: (detaching: 0|1) = > void;
}
Copy the code

C (create) : Creates a DOM node inside the fragment and stores it inside the closure of each fragment. M (mount) : Mount the DOM node to the top of the board at target, where events take place. P (update) : Triggered when component data has changed, in this method to check for updates. D (destroy) : removes mount and event binding.

Compile the results will be introduced from the svelte/internal text, element, append, detach, listen, and so on. Source code can be seen, are some very pure DOM operations.

export function element<K extends keyof HTMLElementTagNameMap> (name: K) {
  return document.createElement<K>(name);
}

export function text(data: string) {
  return document.createTextNode(data);
}

export function append(target: Node, node: Node) {
  if (node.parentNode !== target) {
    target.appendChild(node);
  }
}

export function detach(node: Node) {	
  node.parentNode.removeChild(node);
}

export function listen(node: EventTarget, event: string, handler: EventListenerOrEventListenerObject, options? : boolean | AddEventListenerOptions | EventListenerOptions) {
  node.addEventListener(event, handler, options);
  return () = > node.removeEventListener(event, handler, options);
}
Copy the code

We can be sure that Svelte has no Virtual DOM

$$invalidate — start of Schedule Update

As mentioned, Compiler wraps all the assigned code in $$invalidate. For example count ++, count += 1, name = ‘fesky’, etc.

What does this method do? Look at the source code, (cut some unimportant code)

(i, ret, ... rest) => {const value = rest.length ? rest[0] : ret;
  if (not_equal($$.ctx[i], $$.ctx[i] = value)) {
    make_dirty(component, i);
  }
  return ret;
}
Copy the code

What is the first parameter I? $$. CTX [I] = value $$. CTX [I] = value The result of compiling is passed a 0.

$$invalidate(0, name = "fesky");
Copy the code

In fact, the instance method returns an array containing the properties and methods of the component instance. Svelte assigns the value returned by the instance method to CTX for saving. So I is the index of the array returned by instance.

$$.ctx = instance
    ? instance(component, options.props || {}, (i, ret, ... rest) = > {
       / /...
    })
    : [];
Copy the code

At compile time, Svelte generates the corresponding number based on the position of the property in the array. For example, now I have two variables,

<script>
  let firsName = ' ';
  let lastName = ' ';

  function handleClick () {
    firsName = 'evan'
    lastName = 'zhou';
  }
</script>

<h1 on:click={handleClick}>Hello {firsName}{lastName}!</h1>
Copy the code

The invalidate section of the code will compile as follows:

function handleClick() {
  // Corresponds to the array subscript 0
  $$invalidate(0, firsName = "evan");
  // Array subscript 1
  $$invalidate(1, lastName = "zhou");
}

return [firsName, lastName, handleClick];
Copy the code

Ok, moving on, $$invalidate calls make_dirty when an assignment is not equal.

Dirty Check

function make_dirty(component, i) {
  if (component.$$.dirty[0] = = = -1) {
    dirty_components.push(component);
    schedule_update();
    component.$$.dirty.fill(0);
  }
  component.$$.dirty[(i / 31) | 0] | = (1 << (i % 31));
}
Copy the code

The main process in this method is to add components that call make_dirty to dirty_components and then call the schedule_update method. (Dirty field details delayed)

export function schedule_update() {
  if(! update_scheduled) { update_scheduled =true; resolved_promise.then(flush); }}Copy the code

Schedule_update is as simple as calling flush in promise.resolve (microTask).

(Reading the source code is a bit boring, hang in there, it’s over)

export function flush() {
  for (let i = 0; i < dirty_components.length; i += 1) {
    constcomponent = dirty_components[i]; set_current_component(component); update(component.$$); }}Copy the code

The Flush method simply consumes the preceding dirty_components and calls the update method for each component that needs to be updated.

function update($$) {
  if($$.fragment ! = =null) {
    $$.update();
    const dirty = $$.dirty;
    $$.dirty = [-1]; $$.fragment && $$.fragment.p($$.ctx, dirty); }}Copy the code

The Update method, on the other hand, returns to the P (Update) method for each fragment. So the whole link is clear. Then sort out the following ideas:

  1. Modify data, call$$invalidatemethods
  2. Determine equality, flag dirty data,make_dirty
  3. Trigger an update in microTask, traversing alldirty_componentTo update the DOM node
  4. Reset the Dirty

The magic of Bitmask

One important detail left unexplained in the previous summary was how dirty was marked.

component.$$.dirty[(i / 31) | 0] | = (1 << (i % 31));
Copy the code

If you see 31, if you see the << right shift symbol, it must be that the bit operation is not running. First of all, we need to know that all numbers in JS are 64-bit double-precision floating-point types that comply with the IEEE-754 standard. All bitwise operations will retain only integers of 32 results.

To break in this sentence: (I / 31) | 0: here is to use an array subscript I belong to the 31, then take down the whole (any integer Numbers and | 0 the result is in itself, an operation has the effect of the whole down). (1 << (I % 31)) : modulo 31 with I, then move to the left.

Dirty is an array type that holds multiple 32-bit integers. Each bit of the integer indicates whether the variables converted to the subscript of the instance array have changed.

To make it easier to understand, let’s use four-digit integers.

[1000] = > [8] indicates that the first variable in instance is dirty. [1001] = > [9] indicates that the first and fourth variables in instance are dirty. [1000.0100] = > [9.4] indicates that the first and sixth variables in instance are dirty.Copy the code

Those of you who are not familiar with these basics can turn to the other two articles I have written in the past: core Basics binary (I) 0.1 + 0.2! = 0.3 and ieEE-754 standard core base binary (two) bit operations

Looking back at the p method, each call determines whether the dependent data has changed, and only then does the DOM update.

p(ctx, [dirty]) {
  if (dirty & /*firsName*/ 1) set_data(t1, /*firsName*/ ctx[0]);
  if (dirty & /*lastName*/ 2) set_data(t2, /*lastName*/ ctx[1]);
}
Copy the code

By the way, the convention is that if the dirty first digit is stored as -1, the current component is clean.

$$.dirty = [-1];
Copy the code

A discussion of this can be found at Github Issue, and the benefit of this implementation is that the compiled code is smaller and the binary calculations are a little faster.

summary

Finally, I’ll write a DEMO that uses the Performance panel to record code execution and compare it to React.

<script>
  let count = 1;
  function handleClick () {
    count += 1
  }
</script>

<button on:click={handleClick}>{count}</button>
Copy the code

(It’s so efficient that I had to make a larger version of it.) 💰 Money well spent.

Hopefully, you’ve thoroughly mastered all the logic behind Svelte responsiveness. I made a sketch of the whole process for your reference. As a whole, the code for Svelte runtime is very concise and easy to understand. If you have time, I recommend looking at the source code.

6. Ecological

Another factor in deciding whether to use a framework is the ecology of the framework, which I have collected on the Internet and listed for your reference.

  • SSR Sapper ⭐ (7.1 k)
  • Component Library Svelte-Material – UI (1.7K ⭐)
  • Svelte – VScode (12K Usage)
  • Router svelte-Routing (1.4k ⭐)
  • Native svelte – Native ⭐ (800)
  • Unit Testing svelte-Testing – Library (390 ⭐)
  • Chrome Dev-tools Svelte Devtools (500 ⭐)

Overall, the whole ecology is not strong enough, there is a lot of room. Developing an administrative back end using Svelte may not be as smooth as using Antd, and developing H5 active pages with a highly customized UI is not a problem at all.

7. Conclusion

The most popular reason for choosing Vue over React is that Vue is small and fast. Svelte is now smaller (for small projects), faster and better for active pages. Will you use it?

Anyways, no matter how Arsenal and enriched 💐 💐 💐, next time do technology selection, a kind of choice, understand the need not and have never heard of so don’t still have very big distinction.

For me, Svelte’s implementation of Reactivity is truly unique, and I have learned a lot from the implementation principle. This article took me three days (looking for materials, looking at the source code, writing DEMO, making outline, writing article), if you feel you have harvest, welcome to like ❤️ + favorites + comments + attention, so I will be more motivated to produce a good article.

Time is short, the level is limited, inevitably there will be mistakes, welcome to correct.

8. Related links

  • This section describes the MDN front-end framework
  • When does React re-render components?
  • Rich Harris – Rethinking reactivity
  • Compile Svelte in your head
  • What do you think of Svelte as a front-end framework