在开发过程中,你可能会遇到水合问题。不要忽视这些警告。
水合不匹配不仅仅是警告——它们是严重问题的指示器,可能导致你的应用程序崩溃。
在开发过程中,Vue 会在浏览器控制台中记录水合不匹配警告。

问题:在服务器端渲染期间使用浏览器特有的 API。
<template>
<div>User preference: {{ userTheme }}</div>
</template>
<script setup>
// This will cause hydration mismatch!
// localStorage doesn't exist on the server!
const userTheme = localStorage.getItem('theme') || 'light'
</script>
解决方案:你可以使用 useCookie
<template>
<div>User preference: {{ userTheme }}</div>
</template>
<script setup>
// This works on both server and client
const userTheme = useCookie('theme', { default: () => 'light' })
</script>
问题:服务器和客户端之间数据不同。
<template>
<div>{{ Math.random() }}</div>
</template>
解决方案:使用 SSR 友好的状态
<template>
<div>{{ state }}</div>
</template>
<script setup>
const state = useState('random', () => Math.random())
</script>
问题:在 SSR 期间使用仅客户端条件。
<template>
<div v-if="window?.innerWidth > 768">
Desktop content
</div>
</template>
解决方案:使用媒体查询或在客户端处理。
<template>
<div class="responsive-content">
<div class="hidden md:block">Desktop content</div>
<div class="md:hidden">Mobile content</div>
</div>
</template>
问题:修改 DOM 或具有浏览器依赖项的库(这在标签管理器中经常发生)。
<script setup>
if (import.meta.client) {
const { default: SomeBrowserLibrary } = await import('browser-only-lib')
SomeBrowserLibrary.init()
}
</script>
解决方案:在水合完成后初始化库。
<script setup>
onMounted(async () => {
const { default: SomeBrowserLibrary } = await import('browser-only-lib')
SomeBrowserLibrary.init()
})
</script>
问题:根据当前时间变化的内容。
<template>
<div>{{ greeting }}</div>
</template>
<script setup>
const hour = new Date().getHours()
const greeting = hour < 12 ? 'Good morning' : 'Good afternoon'
</script>
解决方案:使用 NuxtTime 组件或在客户端处理。
<template>
<div>
<NuxtTime :date="new Date()" format="HH:mm" />
</div>
</template>
<template>
<div>
<ClientOnly>
{{ greeting }}
<template #fallback>
Hello!
</template>
</ClientOnly>
</div>
</template>
<script setup>
const greeting = ref('Hello!')
onMounted(() => {
const hour = new Date().getHours()
greeting.value = hour < 12 ? 'Good morning' : 'Good afternoon'
})
</script>
useFetch、useAsyncData、useStateClientOnly 组件。onMounted 中。