Learn how to optimize the performance of your Vue applications. I give you detailed best practices to reduce load times and enhance user satisfaction.
7. September 2024
Vue is designed to be performant for most common use cases without significant need for manual optimizations. However, there are scenarios where additional fine-tuning is required.
Before we start optimizing, we need to measure performance. Here are some useful tools:
Vue-specific performance markers can be enabled with app.config.performance
.
For pages that need to load very quickly, consider Server-Side Rendering (SSR) or Static Site Generation (SSG). Pure client-side SPAs (Single Page Applications) often have slow load times. Marketing pages (e.g., landing pages) can be separately served as static HTML.
One of the most effective ways to optimize load speed is reducing the JavaScript bundle size:
// Example of tree-shaking with dynamic import
import { defineAsyncComponent } from 'vue';
const Foo = defineAsyncComponent(() => import('./Foo.vue'));
Code splitting involves dividing the application bundle into smaller chunks that can be loaded on demand. This allows for faster initial page load speeds.
function loadLazy() {
return import('./lazy.js');
}
A child component only updates when at least one of its props has changed. It is important to keep props stable to avoid unnecessary updates.
<template>
<div :class="{ active: isActive }">...</div>
</template>
<script lang="ts" setup>
defineProps<{ id: number, active: boolean }>();
</script>
v-once
and v-memo
v-once
: Can be used to render content that relies on runtime data but never needs to update.v-memo
: Can conditionally skip updates for large sub-trees or v-for
lists.<template>
<div v-once>
{{ neverChangingContent }}
</div>
</template>
Starting from Vue 3.4, a computed property will only trigger effects when its value changes.
const count = ref(0);
const isEven = computed(() => count.value % 2 === 0);
Large lists should be virtualized to improve performance. Only the items close to the viewport are rendered.
Available libraries:
vue-virtual-scroller
vue-virtual-scroll-grid
vueuc/VVirtualList
Vue's reactivity system is deep by default. For large, deeply nested arrays, this can cause performance overhead. Use shallowRef()
and shallowReactive()
to maintain reactivity only at the root level.
const shallowArray = shallowRef([/* large list of deep objects */]);
Sometimes you create renderless components or higher-order components for better abstraction or code organization. Renderless components are those that do not render their own DOM elements but provide functions and behavior to their child components. Higher-order components are components that render other components with additional props.
While these patterns are useful and often necessary, keep in mind that component instances are more expensive than plain DOM nodes. Each component instance carries overhead in terms of memory usage and performance.
Overusing these patterns, especially in performance-critical areas like large lists, can result in noticeable performance costs in terms of longer render times and increased memory consumption.
Imagine a list with 100 items, where each item has its own component instance and multiple child components.
<template>
<div v-for="item in items" :key="item.id">
<ItemComponent :item="item" />
</div>
</template>
<script lang="ts" setup>
const items = ref([...]);
</script>
In this situation, hundreds of component instances are created, each with its own internal state and lifecycle hooks.
By removing unnecessary component abstractions, you can significantly reduce the number of these instances:
<template>
<div v-for="item in items" :key="item.id">
<div :class="{ active: item.isActive }">{{ item.name }}</div>
</div>
</template>
<script lang="ts" setup>
const items = ref([...]);
</script>
It's important to note that removing only a few component instances won't have a noticeable impact. If a component is rendered only a few times in the entire application, the performance costs are negligible.
The best scenario to consider this optimization is in large lists or heavily used areas of the application. For instance, in a list of 100 items where each item contains multiple child components, removing a single unnecessary abstraction component could result in the reduction of hundreds of component instances.
The goal is to strike the right balance between code organization and performance, recognizing when abstraction is costing more than it's worth in terms of performance.