详解Vue基于 Nuxt.js 实现服务端渲染(SSR)


Posted in Javascript onApril 05, 2018

直接使用 Vue 构建前端单页面应用,页面源码时只有简单的几行 html,这并不利于网站的 SEO,这时候就需要服务端渲染

2016 年 10 月 25 日,zeit.co 背后的团队对外发布了一个 React 的服务端渲染应用框架 Next.js

几小时后,一个基于 Vue.js 的服务端渲染应用框架应运而生,与 Next.js 异曲同工,这就是Nuxt.js

一、快速模板

在已经安装了 vue-cli 的前提下,可以快速创建一个 nuxt 的项目模板

vue init nuxt-community/starter-template MyProject

其中 MyProject 是项目文件夹名称,可自定义

通过 npm install (似乎用 yarn install 更顺利) 安装依赖之后,可以直接 npm run dev 在 开发环境 启动项目

默认启动的地址为 http://localhost:3000/,可以在 package.json 中添加以下配置来修改主机端口号

"config": {
  "nuxt": {
   "host": "0.0.0.0",
   "port": "7788"
  }
 },

开发完成后执行 npm run build 打包代码,最后 npm start 启动服务

二、重要目录

生成的 项目目录 如下

详解Vue基于 Nuxt.js 实现服务端渲染(SSR)

大部分文件夹名称 都是 nuxt 默认保留的,不可修改

其中比价比较关键的目录有三个:

1. components 组件目录

详解Vue基于 Nuxt.js 实现服务端渲染(SSR)

一般用来存放 非页面级别 的组件,如 header、footer 等公共组件

该目录下的组件具有常规 vue 组件的方法和特性,不会被 nuxt.js 扩展特性

2. layouts 布局目录

详解Vue基于 Nuxt.js 实现服务端渲染(SSR)

可以修改该目录下的 default.vue 来修改默认布局

<template>
 <div>
  <my-header></my-header>
  <nuxt/>
  <my-footer></my-footer>
 </div>
</template>

其中 <nuxt/> 是必需的,页面的主体内容会显示在这里 (类似于根节点的 <router-view/>)

此外还可以在目录下新增 error.vue 作为错误页面,具体的写法可以参考官方文档

3. pages 页面目录

详解Vue基于 Nuxt.js 实现服务端渲染(SSR)

用于存放页面级别的组件,nuxt 会根据该目录下的页面结构生成路由

比如上图中的页面结构,会生成这样的路由配置:

router: {
 routes: [
  {
   name: 'index',
   path: '/',
   component: 'pages/index.vue'
  },
  {
   name: 'about',
   path: '/about',
   component: 'pages/about.vue'
  },
  {
   name: 'classroom',
   path: '/classroom',
   component: 'pages/classroom.vue',
   children: [
    {
     path: 'student',
     component: 'pages/classroom/student.vue',
     name: 'student'
    },
    { //定义带参数的动态路由,需要创建对应的以下划线作为前缀的 Vue 文件或目录
     path: ':id',
     component: 'pages/classroom/_id.vue',
     name: 'classroom-id'
    }
   ]
  }
 ]
}

此外,该目录下的 vue 组件还具备一些 Nuxt.js 提供的特殊功能特性

其中 asyncData 方法比较常用,支持异步数据处理

这个方法会在 页面组件 每次加载之前被调用,然后获取数据并返回给当前组件

asyncData ({ params, error }) {
  return axios.get(`api/posts/${params.id}`)
  .then((res) => {
   return { name: res.data.name}
  })
  .catch((e) => {
   error({ statusCode: 404, message: 'not found' })
  })
 }

asyncData 方法的第一个参数为上下文对象 context,具体属性可以 查看这里

由于asyncData方法是在组件初始化前被调用的,所以在方法内是没有办法通过 this 来引用组件的实例对象

三、使用插件

如果项目中还需要引入其他的第三方插件,可以直接在页面中引入,这样在打包的时候,会将插件打包到页面对应的 js 里面

但要是别的页面也引入了同样的插件,就会重复打包。如果没有需要分页打包的需求,这时候可以配置 plugins

以 element-ui 为例,在安装了 element-ui 之后,在 plugins 目录下创建 elementUI.js

详解Vue基于 Nuxt.js 实现服务端渲染(SSR)

然后在根目录的 nuxt.config.js 中添加配置项 build.vendor 和 plugins

build: {
  vendor: ['~/plugins/elementUI.js']
 },
 plugins: [
  {src: '~/plugins/elementUI.js'},
 ]

这里的 plugins 属性用来配置 vue.js 插件,也就是 可以用 Vue.user() 方法 的插件

默认只需要 src 属性,另外还可以配置 ssr: false,让该文件只在客户端被打包引入

如果是像 axios 这种第三方 (不能 use) 插件,只需要在 plugins 目录下创建 axios.js

// axios.js

import Vue from 'vue'
import axios from 'axios'

const service = axios.create({
 baseURL: '/api'
})

Vue.prototype.$ajax = axios
export default service

然后在 build.vendor  中添加配置 (不需要配置 plugins)

build: {
 vendor: ['~/plugins/axios.js']
}

这样在打包的时候,就会把 axios 打包到 vendor.js 中

四、Vuex 状态树

如果在 store 目录下创建了 index.js,nuxt.js 会根据该目录下的文件创建 Vuex 状态树

// store/index.js

import Vue from 'vue'
import Vuex from 'vuex'
import Axios from '~/plugins/axios.js';

Vue.use(Vuex)

const store = () => new Vuex.Store({
 state: {
  author: 'WiseWrong',
  info: ''
 },
 mutations: {
  setInfo (state, val) {
   state.info = val
  }
 },
 actions: {
  loadAboutMeInfo ({commit, state}) {
   return Axios.get(`/about`)
    .then(res => {
     console.log('ajax is success')
     console.log(res.data.info)
     commit('setInfo', res.data.info)
    })
    .catch(err => {
     console.log('error')
    })
  }
 }
})

export default store

Nuxt.js 内置引用了 vuex 模块,不需要额外安装

上面的代码中,我在 actions 中写了一个 loadAboutMeInfo() 方法,用来请求 /api/about 接口

然后在 about.vue 页面中调用

// about.vue 

<template>
 <section class="container">
  <div>
   <img src="~/assets/about.png" alt="">
  </div>
  <h1>{{$store.state.info}}</h1>
 </section>
</template>

<script>
export default {
 fetch({ store }) {
  return store.dispatch('loadAboutMeInfo')
 },
 name: 'about',
 data () {
  return {}
 }
}
</script>

成果演示:

详解Vue基于 Nuxt.js 实现服务端渲染(SSR) 

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
对JavaScript的eval()中使用函数的进一步讨论
Jul 26 Javascript
关于跨站脚本攻击问题
Dec 22 Javascript
使用JS或jQuery模拟鼠标点击a标签事件代码
Mar 10 Javascript
javascript复制粘贴与clipboardData的使用
Oct 16 Javascript
JS组件Bootstrap Select2使用方法解析
May 30 Javascript
Js得到radiobuttonlist选中值的两种方法(推荐)
Aug 25 Javascript
微信小程序 图片等比例缩放(图片自适应屏幕)
Nov 16 Javascript
vue2中使用less简易教程
Mar 27 Javascript
深入浅析angular和vue还有jquery的区别
Aug 13 jQuery
Vue实现远程获取路由与页面刷新导致404错误的解决
Jan 31 Javascript
ES10 特性的完整指南小结
Mar 04 Javascript
微信小程序登录对接Django后端实现JWT方式验证登录详解
Jul 29 Javascript
express默认日志组件morgan的方法
Apr 05 #Javascript
React Native悬浮按钮组件的示例代码
Apr 05 #Javascript
关于Google发布的JavaScript代码规范你要知道哪些
Apr 04 #Javascript
Angular HMR(热模块替换)功能实现方法
Apr 04 #Javascript
解决vue中使用swiper插件问题及swiper在vue中的用法
Apr 04 #Javascript
vue基于mint-ui实现城市选择三级联动
Jun 30 #Javascript
vue中mint-ui的使用方法
Apr 04 #Javascript
You might like
elgg 获取文件图标地址的方法
2010/03/20 PHP
PHP下打开URL地址的几种方法小结
2010/05/16 PHP
PHP操作数组的一些函数整理介绍
2011/07/17 PHP
ThinkPHP3.1新特性之内容解析输出详解
2014/06/19 PHP
摘自织梦CMS的HTTP文件下载类
2015/08/08 PHP
php使用curl下载指定大小的文件实例代码
2017/09/30 PHP
js的alert弹出框出现乱码解决方案
2013/09/02 Javascript
js arguments,jcallee caller用法总结
2013/11/30 Javascript
JavaScript验证电子邮箱的函数
2014/08/22 Javascript
浅析Javascript中bind()方法的使用与实现
2016/05/30 Javascript
jQuery实现的自动加载页面功能示例
2016/09/04 Javascript
jQuery 局部div刷新和全局刷新方法总结
2016/10/05 Javascript
js实现密码强度检验
2017/01/15 Javascript
浅谈Angular2 ng-content 指令在组件中嵌入内容
2017/08/18 Javascript
解决Vue 浏览器后退无法触发beforeRouteLeave的问题
2017/12/24 Javascript
详解swipe使用及竖屏页面滚动方法
2018/06/28 Javascript
JS实现的视频弹幕效果示例
2018/08/17 Javascript
JavaScript实时更新当前的时间的示例代码
2020/07/15 Javascript
[01:34]DAC2018主赛事第四日五佳镜头 Gh巨牙海民助Miracle-死里逃生
2018/04/07 DOTA
Python sys.path详细介绍
2013/10/17 Python
详解opencv Python特征检测及K-最近邻匹配
2019/01/21 Python
django框架模型层功能、组成与用法分析
2019/07/30 Python
pycharm 对代码做静态检查操作
2020/06/09 Python
python实现发送QQ邮件(可加附件)
2020/12/23 Python
英国豪华针织品牌John Smedley的在线销售商:The Outlet by John Smedley
2018/04/08 全球购物
普通PHP程序员笔试题
2016/01/01 面试题
现在输入n个数字,以逗号,分开;然后可选择升或者降序排序;按提交键就在另一页面显示按什么排序,结果为,提供reset
2012/11/09 面试题
农救科工作职责
2013/11/27 职场文书
自我管理的活动方案
2014/08/25 职场文书
初中数学教学反思范文
2016/02/17 职场文书
记一次Mysql不走日期字段索引的原因小结
2021/10/24 MySQL
Nginx下SSL证书安装部署步骤介绍
2021/12/06 Servers
mybatis源码解读之executor包语句处理功能
2022/02/15 Java/Android
Java实现学生管理系统(IO版)
2022/02/24 Java/Android
Z-Order加速Hudi大规模数据集方案分析
2022/03/31 Servers
苹果发布了MagSafe固件更新,可以不外接电源实现最高7.5W充电
2022/04/21 数码科技