业务场景
- 首先来看一个简单的 Vue 组件
test.vue
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 | <template> <div> <h2>clientHeight: {{ clientHeight }} px </h2> </div> </template> <script type= "text/babel" > export default { data(){ return { } }, computed :{ clientHeight() { return document.body.clientHeight; } }, mounted(){ } } </script> |
- 上面
test.vue
组件通过 Vue computed 属性 clientHeight 直接获取 document 的文档高度,这段代码在前端渲染是不会报错的,也能拿到正确的值。但如果把这个组件放到 SSR(Server Side Render) 模式下, 就会报如下错误:
ReferenceError: document is not defined
解决方案
- 通过 typeof 判断是否是存在 document 对象, 如果存在则执行后面代码。 这种方式虽然能解决问题, 但在 Webpack 构建压缩时, 不会执行的代码不会被剔除,也会打包到 js 文件中去, 因为这个是在运行期才知道结果的, 所以在 Webpack 构建方案中,不建议使用 typeof 方式判断。而是使用 Webpack 提供的 webpack.DefinePlugin 插件定义常量解决。
1 2 3 | clientHeight() { return typeof document === 'object' ? document.body.clientHeight : '' ; } |
- 使用 Webpack 提供的 webpack.DefinePlugin 插件定义常量解决。 这里直接使用 easywebpackhttps://github.com/hubcarl/easywebpack 内置的全局 Webpack 常量 EASY_ENV_IS_BROWSERhttp://hubcarl.github.io/easywebpack/webpack/env 进行判断。 这样在构建压缩期间, 如果是 Node 模式构建, EASY_ENV_IS_BROWSER 会被替换为 false,如果是 Browser 模式构建, EASY_ENV_IS_BROWSER 会被替换为 true,最后构建后代码也就是变成了 true 或者 false 的常量。 因为这个是构建期间执行的,压缩插件剔除永远不会被执行的代码, 也就是 dead_code
1 2 3 | clientHeight() { return EASY_ENV_IS_BROWSER ? document.body.clientHeight : '' ; } |
NPM Vue 组件 SSR 支持
针对上面这种自己写的代码,我们可以通过这种方式解决,因为可以直接修改。但如果我们引入的一个 npm Vue 插件想进行SSR渲染, 但这个插件里面使用了 window/docment 等浏览器对象, 并没有对 SSR 模式进行兼容,这个时候该如何解决呢?
一般我们通过 通过 v-if 来决定是否渲染该组件
和 Vue 只在前端挂载组件解决问题
可以解决。
通过 v-if 来决定是否渲染该组件
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 | <template> <div v- if = "isBrowser" > <Loading></Loading> </div> </template> <script type= "text/babel" > export default { componets:{ Loading: () =>import( 'vue-loading' ); } data(){ return { isBrowser: EASY_ENV_IS_BROWSER } }, mounted(){ } } </script> |
Vue 只在前端挂载组件解决问题
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 | <template> <div> <Loading></Loading> </div> </template> <script type= "text/babel" > export default { data(){ return { } }, beforeMount() { // 只会在浏览器执行 this .$options.components.Loading = () =>import( 'vue-loading' ); }, mounted(){ } } </script> |
loading
组件因为没有注册, 在 SSR 模式, <Loading></Loading>
会被原样输出到 HTML 中,不会报错且不能被浏览器识别, 在显示时不会有内容。当 SSR 直出 HTML 后,浏览器模式中执行 beforeMount
挂载组件, 从而达到解决服务端渲染报错的问题
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· AI与.NET技术实操系列:基于图像分类模型对图像进行分类
· go语言实现终端里的倒计时
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 25岁的心里话
· 闲置电脑爆改个人服务器(超详细) #公网映射 #Vmware虚拟网络编辑器
· 零经验选手,Compose 一天开发一款小游戏!
· 通过 API 将Deepseek响应流式内容输出到前端
· AI Agent开发,如何调用三方的API Function,是通过提示词来发起调用的吗
2019-12-24 浅析DES、AES、RSA、MD5加密算法及其应用场景
2019-12-24 关于JSON.parse(JSON.stringify(obj))实现深拷贝应该注意的坑