分析关于Vue SSR 的 Cookies 问题

本篇文章主要介绍了浅谈vue ssr 的 cookies 问题,内容挺不错的,现在分享给大家,也给大家做个参考。

一个网站一旦涉及到多用户, 就很难从 Cookies 中逃脱, Vue SSR 的 cookies 也真算是遇到的一个不小的问题, 从开始玩 SSR 开始到现在, 一共想出了3种方案, 从最早的把 Cookies 注入到 state 中, 到把 Cookies 注入到 global, 到现在的将 Cookies 注入到组件的 asyncData 方法.

随着 Vue 的升级, 第一种方案已经不再适用, 第二种也有不少的限制, 于是想到第三种方案, 下来就说说具体实现的方法:

第一种方案

第一种方案已经不再适用, 这里不再细说

立即学习“前端免费学习笔记(深入)”;

第二种方案

思路: 将 cookies 注入到 ssr 的 context里, 然后在请求 api 时读取, 再追加到 axios 的header 里

1, 首先在 server.js 里将 cookies 加到 context里

const context = { title: 'M.M.F 小屋', description: 'M.M.F 小屋', url: req.url, cookies: req.cookies}renderer.renderToString(context, (err, html) => { if (err) {  return errorHandler(err) } res.end(html)})

登录后复制

之后, Vue 会把 context 加到 global.__VUE_SSR_CONTEXT__

2, 在 api.js 里读取 cookies

import axios from 'axios'import qs from 'qs'import md5 from 'md5'import config from './config-server'const SSR = global.__VUE_SSR_CONTEXT__const cookies = SSR.cookies || {}const parseCookie = cookies => { let cookie = '' Object.keys(cookies).forEach(item => {  cookie+= item + '=' + cookies[item] + '; ' }) return cookie}export default { async post(url, data) {  const cookie = parseCookie(cookies)  const res = await axios({   method: 'post',   url: config.api + url,   data: qs.stringify(data),   timeout: config.timeout,   headers: {    'X-Requested-With': 'XMLHttpRequest',    'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8',    cookie   }  })  return res },}

登录后复制

为什么可以这么做?

默认情况下,Vue 对于每次渲染,bundle renderer 将创建一个新的 V8 上下文并重新执行整个 bundle。应用程序代码与服务器进程隔离, 所以每个访问的用户上下文都是独立的, 不会互相影响.

但是从Vue@2.3.0开始, 在createBundleRenderer方法的选项中, 添加了runInNewContext选项, 使用 runInNewContext: false,bundle 代码将与服务器进程在同一个 global 上下文中运行,所以我们不能再将 cookies 放在 global, 因为这会让所有用户共用同一个 cookies.

为什么现在不这么做?

那我们继续将runInNewContext设置成true, 不就好了吗? 当然也是可以的, 但是重新创建上下文并执行整个 bundle 还是相当昂贵的,特别是当应用很大的时候.

以我自己的博客为例, 之前只有渲染 5 个路由组件, loadtest 的 rps, 有 50 左右, 但是后来把后台的 12 个路由组件也加到 SSR 后, rps 直接降到了个位数…

所以出现了现在的第三种方案

第三种方案

思路: 将 Cookies 作为参数注入到组件的asyncData方法, 然后用传参数的方法把 cookies 传给 api, 不得不说这种方法很麻烦, 但是这是个人能想到的比较好的方法

步骤1:

还是在 server.js 里, 把 cookies 注入到 context 中

const context = { title: 'M.M.F 小屋', url: req.url, cookies: req.cookies,}renderer.renderToString(context, (err, html) => { if (err) {  return handleError(err) } res.end(html)})

登录后复制

步骤2:

在entry-server.js里, 将cookies作为参数传给 asyncData 方法

Promise.all(matchedComponents.map(({asyncData}) => asyncData && asyncData({ store, route: router.currentRoute, cookies: context.cookies, isServer: true, isClient: false}))).then(() => { context.state = store.state context.isProd = process.env.NODE_ENV === 'production' resolve(app)}).catch(reject)

登录后复制

步骤3:

在组件里, 把 cookies 做为参数给 Vuex 的 actions

export default { name: 'frontend-index', async asyncData({store, route, cookies}, config = { page: 1}) {  config.cookies = cookies  await store.dispatch('frontend/article/getArticleList', config) } // .....}

登录后复制

步骤4:

在 Vuex 里将 cookies 做为参数给 api

import api from '~api'const state = () => ({ lists: {  data: [],  hasNext: 0,  page: 1,  path: '' },})const actions = { async ['getArticleList']({commit, state}, config) {  // vuex 作为临时缓存  if (state.lists.data.length > 0 && config.path === state.lists.path && config.page === 1) {   return  }  let cookies  if (config.cookies) {   cookies = config.cookies   delete config.cookies  }  const { data: { data, code} } = await api.get('frontend/article/list', {...config, cache: true}, cookies)  if (data && code === 200) {   commit('receiveArticleList', {    ...config,    ...data,   })  } },}const mutations = { ['receiveArticleList'](state, {list, hasNext, hasPrev, page, path}) {  if (page === 1) {   list = [].concat(list)  } else {   list = state.lists.data.concat(list)  }  state.lists = {   data: list, hasNext, hasPrev, page, path  } },}const getters = {}export default { namespaced: true, state, actions, mutations, getters}

登录后复制

这里一定要注意, state 一定要用函数返回值来初始化 state, 不然会导致所有用户共用 state

步骤5:

在 api 里接收 cookies, 并加到 axios 的 headers 里

import axios from 'axios'import qs from 'qs'import config from './config-server'const parseCookie = cookies => { let cookie = '' Object.keys(cookies).forEach(item => {  cookie+= item + '=' + cookies[item] + '; ' }) return cookie}export default { get(url, data, cookies = {}) {  const cookie = parseCookie(cookies)  return axios({   method: 'get',   url: config.api + url,   data: qs.stringify(data),   timeout: config.timeout,   headers: {    'X-Requested-With': 'XMLHttpRequest',    'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8',    cookie   }  }) },}

登录后复制

第四种方案

步骤1:

还是在 server.js 里, 把 cookies 注入到 context 中

const context = {  title: 'M.M.F 小屋',  url: req.url,  cookies: req.cookies,}renderer.renderToString(context, (err, html) => {  if (err) {    return handleError(err)  }  res.end(html)})

登录后复制

步骤2:

在entry-server.js里, 将cookies作为参数传给 api.setCookies 方法, api.setCookies 是什么东西后面会有

api.setCookies(context.cookies) // 这一句Promise.all(matchedComponents.map(({asyncData}) => asyncData && asyncData({ store, route: router.currentRoute, cookies: context.cookies, isServer: true, isClient: false}))).then(() => { // ...}

登录后复制

步骤3:

重写 api.js

import axios from 'axios'import qs from 'qs'import config from './config-server'const parseCookie = cookies => {  let cookie = ''  Object.keys(cookies).forEach(item => {    cookie+= item + '=' + cookies[item] + '; '  })  return cookie}export default {  api: null,  cookies: {},  setCookies(value) {    value = value || {}    this.cookies = value    this.api = axios.create({      baseURL: config.api,      headers: {        'X-Requested-With': 'XMLHttpRequest',        cookie: parseCookie(value)      },      timeout: config.timeout,    })  },  post(url, data) {    if (!this.api) this.setCookies()    return this.api({      method: 'post',      url,      data: qs.stringify(data),      headers: {        'Content-Type': 'application/x-www-form-urlencoded; charset=UTF-8',      }    }).then(res => {      return res    })  },  get(url, params) {    if (!this.api) this.setCookies()    return this.api({      method: 'get',      url,      params,    }).then(res => {      return res    })  }}

登录后复制

步骤4:

没有步骤4了, 直接引入 api 调用即可

如果你没有将 axios 重新封装, 那么也可以把第五步省略, 直接在第四部把 cookies 给 axios 即可

方案 2 具体实例: https://github.com/lincenying/mmf-blog-vue2-ssr

方案 3 具体实例: https://github.com/lincenying/mmf-blog-vue2-pwa-ssr

方案 4 具体实例: https://github.com/lincenying/mmf-blog-vue2-pwa-ssr

综上, 如果你项目不大, 还是直接用方案 2 吧, 项目有很多页面, 并且大部分页面是每个用户都一样的, 可以考虑方案 3, 或者你有什么更好的方法, 欢迎讨论

Vue SSR 对需要 SEO, 并且每个用户看到的内容都是一致的, 配合缓存, 将是一个非常好的体验…

以上就是本文的全部内容,希望对大家的学习有所帮助,更多相关内容请关注PHP中文网!

相关推荐:

关于Vue2 SSR缓存 Api 数据的方法

关于Vue2 SSR缓存 Api 数据的方法

以上就是分析关于Vue SSR 的 Cookies 问题的详细内容,更多请关注【创想鸟】其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至253000106@qq.com举报,一经查实,本站将立刻删除。

发布者:PHP中文网,转转请注明出处:https://www.chuangxiangniao.com/p/2747727.html

(0)
上一篇 2025年3月8日 04:26:54
下一篇 2025年3月3日 19:45:47

AD推荐 黄金广告位招租... 更多推荐

相关推荐

  • vue 计算属性与方法跟侦听器区别的解析

    这篇文章主要介绍了详解vue 计算属性与方法跟侦听器区别(面试考点),内容挺不错的,现在分享给大家,也给大家做个参考。 计算属性 模板内的表达式非常便利,但是设计它们的初衷是用于简单运算的。在模板中放入太多的逻辑会让模板过重且难以维护。例如…

    2025年3月8日 编程技术
    200
  • Vue2.0 多 Tab切换组件的封装介绍

    本篇文章主要介绍了vue2.0 多 tab切换组件的封装实例,内容挺不错的,现在分享给大家,也给大家做个参考。 Vue2.0 多 Tab切换组件简单封装,满足自己简单的功能,可以直接拿去使用! 首先上效果图: 功能简单介绍: 1、支持tab…

    2025年3月8日
    200
  • Vue 去除路径中的#号的方法

    大家都知道vue-router有两种模式,hash模式和history模式,带#的则是hash模式。接下来给大家带来了vue 去除路径中的#号的解决方法,感兴趣的朋友一起看看吧 在开发过程中发现路径中带有/#/的标示,而且还去不掉,很丑陋。…

    2025年3月8日
    200
  • Vue组件通信实践的介绍

    本篇文章主要介绍了vue组件通信实践记录(推荐),内容挺不错的,现在分享给大家,也给大家做个参考。 组件通信 几乎所有的mvvm框架中都要涉及组件通信的功能(吐槽一下knockout,毕竟是鼻祖就先不说它了)。而且目前的前端形式来看,组件化…

    2025年3月8日
    200
  • VueJS组件之间通过props交互及验证的方式

    本篇文章主要介绍了vuejs组件之间通过props交互及验证的方式,具有一定的参考价值,感兴趣的小伙伴们可以参考一下。 props 是父组件用来传递数据的一个自定义属性。父组件的数据需要通过 props 把数据传给子组件,子组件需要显式地用…

    2025年3月8日
    200
  • 如何解决Vue路由History mode模式中页面无法渲染

    这篇文章主要介绍了详解vue路由history mode模式中页面无法渲染的原因及解决,非常具有实用价值,需要的朋友可以参考下 Vue下路由History mode导致页面无法渲染的原因 用 Vue.js + vue-router 创建单页…

    2025年3月8日
    200
  • vue-cli构建项目反向代理配置的解析

    本篇文章主要介绍了详解vue-cli构建项目反向代理配置,内容挺不错的,现在分享给大家,也给大家做个参考。 本文介绍了vue-cli构建项目反向代理配置,分享给大家,具体如下: proxyTable: {//配置请求代理’/dlsys’:{…

    2025年3月8日
    200
  • 使用Vue自定义数字键盘组件的方法

    这篇文章主要介绍了关于使用vue自定义数字键盘组件的方法,有着一定的参考价值,现在分享给大家,有需要的朋友可以参考一下 最近做 Vue 开发,因为有不少页面涉及到金额输入,产品老是觉得用原生的 input 进行金额输入的话 体验很不好,于是…

    2025年3月8日
    200
  • vue单个组件实现无限层级多选菜单功能的介绍

    这篇文章主要介绍了vue单个组件实现无限层级多选菜单的相关资料,非常不错,具有参考借鉴价值,需要的朋友可以参考下 wTree.vue  原理:每一个多选框都是一个节点,每个节点就是一个wTree组件,有父级(顶级level为0),有子级(底…

    编程技术 2025年3月8日
    200
  • vue 设置路由的登录权限的方法

    这篇文章主要介绍了vue 设置路由的登录权限的方法,非常不错,具有一定的参考借鉴价值,需要的朋友可以参考下 index.js 将需要登录权限的路由设置meta属性 meta:{requireAuth:true},登录后复制 main.js …

    2025年3月8日
    200

发表回复

登录后才能评论