nuxt3服务端与客户端渲染不一致-案例

环境

makefile 复制代码
node: v20.12.2,
"nuxt": "^3.11.2",
"vue": "^3.4.27",

所有案例均基于 nuxt3

随机数

html 复制代码
<template>
  <div>
    <h3>nuxt3 渲染上的不一致</h3>

    <div>
      <div v-for="item in colors">{{ item }}</div>
    </div>
  </div>
</template>

<script setup>
const colors = ref([]);

function generateRandomColor() {
  const letters = '0123456789ABCDEF';
  let color = '#';
  for (let i = 0; i < 6; i++) {
    color += letters[Math.floor(Math.random() * 16)];
  }
  return color;
}

for (let i = 0; i < 5; i++) {
  colors.value.push(generateRandomColor());
}
</script>

看代码上没问题,放 SPA里面 也不会有任何问题,但是放nuxt3/SSR框架里面,就会出现 服务端渲染和客户端渲染不一致问题

scss 复制代码
Hydration text content mismatch on  (水合不匹配)

... 某一个组件

- rendered on server: #047FC3   服务端渲染,得到的第一项的值是 #047FC3
- expected on client: #1E87BA   客户端的时候,重新运行,变成了 #1E87BA

造成了,服务端和客户端渲染内容不一致, nuxt3 会以 客户端渲染为主, 所以我们看到的结果是: #1E87BA

解决办法:

  1. 将数据定死,就N个, 不管页面如何刷新 colors 始终是那些数据
js 复制代码
const colors = ref([
    "xxxx",
    "1111"
])
  1. server中取数据
html 复制代码
<template>
  <div>
    <h3>nuxt3 渲染上的不一致</h3>

    <div>
      <div v-for="item in colors">{{ item }}</div>
    </div>
  </div>
</template>

<script setup>
const colors = ref([]);

const res = await useFetch('/api/getColors')

colors.value = res.data.value
</script>

server/api/getColors.ts

js 复制代码
function generateRandomColor() {
  const letters = '0123456789ABCDEF';
  let color = '#';
  for (let i = 0; i < 6; i++) {
    color += letters[Math.floor(Math.random() * 16)];
  }
  return color;
}

export default defineEventHandler(() => {
  let list = []

  for (let i = 0; i < 5; i++) {
    list.push(generateRandomColor());
  }

  return list
})

network 中看这个页面的字符串内容,从 api 接口中取值,是都拼接好了结果的。

  1. 将取数的逻辑放在 onMounted 里面 从 客户端去取数据,填充数据,这样就不会造成服务端与客户端生成的结果不一致的警告
html 复制代码
<template>
  <div>
    <h3>nuxt3 渲染上的不一致</h3>

    <div>
      <div v-for="item in colors">{{ item }}</div>
    </div>
  </div>
</template>

<script setup>
const colors = ref([]);

function generateRandomColor() {
  const letters = '0123456789ABCDEF';
  let color = '#';
  for (let i = 0; i < 6; i++) {
    color += letters[Math.floor(Math.random() * 16)];
  }
  return color;
}

onMounted(() => {
  for (let i = 0; i < 5; i++) {
    colors.value.push(generateRandomColor());
  }
});
</script>
相关推荐
战南诚9 小时前
VUE中,keep-alive组件与钩子函数的生命周期
前端·vue.js
霍理迪9 小时前
Vue的响应式和生命周期
前端·javascript·vue.js
SuperEugene15 小时前
Vue3 模板语法规范实战:v-if/v-for 不混用 + 表达式精简,避坑指南|Vue 组件与模板规范篇
开发语言·前端·javascript·vue.js·前端框架
Luna-player15 小时前
Vue 3 + Vue Router 的路由配置,简单示例
前端·javascript·vue.js
angerdream15 小时前
最新版vue3+TypeScript开发入门到实战教程之Vue3详解props
javascript·vue.js
~欲买桂花同载酒~16 小时前
项目优化-vite打包优化
前端·javascript·vue.js
踩着两条虫17 小时前
AI 驱动的 Vue3 应用开发平台 深入探究(二十):CLI与工具链之构建配置与Vite集成
前端·vue.js·ai编程
踩着两条虫17 小时前
AI 驱动的 Vue3 应用开发平台 深入探究(二十):CLI与工具链之自定义构建插件
前端·vue.js·ai编程
极梦网络无忧18 小时前
基于 Vite + Vue3 的组件自动注册功能
前端·javascript·vue.js
雪碧聊技术19 小时前
前端vue代码架子搭建
前端·javascript·vue.js·前端项目代码框架搭建