前言

本篇主要针对nuxtjs中的一些重要概念整理和代码实现!

在学习vue服务端渲染之前,先搞清楚几个概念:

  • 什么是客户端渲染(CSR)
  • 什么是服务端渲染(SSR)
  • CSR和SSR有什么异同

客户端渲染(CSR):当用户在浏览器中输入网址,打开网页,此时的页面只有样式和一些html代码构成的空壳页面,并没有数据。这就需要我们通过执行js代码,请求相关数据,请求到数据之后,通过模板(vue),将这些数据渲染到页面,最终呈现给用户完整的页面。

服务端渲染(SSR):当用户在浏览器中输入网址,打开网页,此时的后端会根据请求的网址,拿到相关页面并将数据填入到页面,完成页面的渲染,最后将完整的页面返回给客户端(浏览器),呈现到用户面前。

利弊:客户端渲染,就是现在非常流行的前后分离的开发模式,极大程度的减轻后端压力,但是对SEO不友好;服务端渲染对SEO友好,但是对后端以及服务器的性能要求较高。

正文:

先用node实现简易的服务端渲染:

  1. const Vue = require('vue');
  2. const server = require('express')();
  3.  
  4. server.get('/',(req,res)=>{
  5.  
  6. //创建 Vue 实例
  7. const app = new Vue({
  8. template:`<div>hello vue SSR!</div>`
  9. })
  10.  
  11. // 创建 renderer 没下载的需要手动下载 npm i vue-server-renderer --save改包用于vue的服务端渲染
  12. const renderer = require('vue-server-renderer').createRenderer();
  13.  
  14. renderer.renderToString(app).then(html=>{
  15. res.send(`
  16. <!DOCTYPE html>
  17. <html lang="en">
  18. <head>
  19. <meta charset="UTF-8">
  20. <meta name="viewport" content="width=device-width, initial-scale=1.0">
  21. <title>Document</title>
  22. </head>
  23. <body>
  24. ${html}
  25. </body>
  26. </html>
  27. `)
  28. }).catch(err=>{
  29. console.log(err)
  30. })
  31. })
  32.  
  33. server.listen(8001)

检查源代码:

下面使用nuxjs来构建项目!

首先,使用nuxt提供的脚手架( create-nuxt-app)构建项目,根据提示一步一步操作。

主要注意俩点:1.Choose custom server framework,选择服务端渲染的框架,我们这里选择的是express

        2.Choose rendering mode 这个选项可以选择SSR和Single Page APP,因为我们需要构建服务端渲染多页面项目,所以这里选择SSR

下面就是项目创建完成后的目录结构:

目录结构:

  • assets::资源目录,用于组织未编译的静态资源(css文件、图标等)
  • components: 组件目录,用于组织应用的vue组件,这些组件不会像页面组件(pages中的组件)那样有 asyncData 方法的特性
  • layouts:布局目录,用于组织应用的布局组件,若无额外配置,该目录不能被重命名
  • middleware:中间件目录,用于存放应用的中间件
  • pages: 页面目录,用于组织应用的路由及视图,Nuxt.js 框架读取该目录下所有的 .vue文件并自动生成对应的路由配置,若无额外配置,该目录不能被重命名
  • plugins:插件目录,用于组织那些需要在 根vue.js应用实例化之前需要运行的JavaScript插件
  • server:服务器目录,用于服务端渲染
  • static:静态文件目录,用于存放应用的静态文件,此类文件不会被 Nuxt.js 调用 Webpack 进行构建编译处理。服务器启动的时候,该目录下的文件会映射至应用的根路径 / 下(/static/robots.txt 映射至 /robots.txt),若无额外配置,该目录不能被重命名
  • store:store目录,用于组织应用的 Vuex状态树 文件,若无额外配置,该目录不能被重命名
  • nuxt.config.js:用于组织 Nuxt.js 应用的个性化配置,以便覆盖默认配置,若无额外配置,该目录不能被重命名
  • package.json:用于描述应用的依赖关系和对外暴露的脚本接口,该文件不能被重命名

项目跑起来:

检查源码,页面的所有数据都能够看到。

接下来正式学习nuxt,主要分为3部分:

  • 生命周期
  • 路由
  • vuex状态树

一、生命周期(钩子函数)

  • nuxtServerInit => 服务器初始化
  • middleware => 中间件
  • validate() => 参数校验
  • asyncData()和fetch() => 异步数据处理
  • render => 客户端渲染
  • beforeCreate和created => 这俩个钩子服务端和客户端都存在,可以拿到服务端的上下文以及组件本身

下面具体使用这些生命周期:

1.nuxtServerInit 

在store目录中新建index.js:

  1. export const actions = {
  2. nuxtServerInit(store, context){
  3. console.log('我是nuxtServerInit生命周期!', store, context)
  4. }
  5. }

项目跑起来后,在终端会打印相关信息:

2.middleware 

middleware中间件可以在配置文件中、布局页面、页面组件中使用。

在nuxt.config.js配置文件中,将middleware写进router配置中,可以进行全局的导航守卫:

  1. module.exports = {
  2. mode: 'universal',
  3. router: {
  4. middleware: 'isLogin'
  5. },
  6. ...

接下来在middleware目录中新建isLogin.js文件:

  1. export default (context) => {
  2. console.log('我是全局守卫')
  3. }

打印:

在布局页面中使用中间件,在layout目录下的default.vue中:

  1. <script>
  2. export default {
  3. // middleware: 'isLogin',
  4. middleware(){
  5. console.log('我是布局页面的中间件!')
  6. },
  7. }
  8. </script>

在页面组件内中间件可以直接写外部之前定义的中间件,也可以用中间件钩子!

同样,在页面组件中,pages目录下的index.vue中:

  1. <script>
  2. export default {
  3. // middleware: 'isLogin',
  4. middleware(){
  5. console.log('我是页面组件的中间件!')
  6. },
  7. }
  8. </script>

通过打印的顺序可以看出,中间件的执行顺序是:nuxt.config.js文件 => 布局文件 => 页面文件,所有中间件函数的参数中,都能拿到服务端的上下文信息context。

3.validate()

validate()需要定义在页面组件内,即pages目录下的页面,可以用来进行参数校验,拦截是否可以进入该页面。具体实现,在pages/index.vue中:

  1. //js 代码
  2. export default {
  3. // middleware: 'isLogin',
  4. middleware(){
  5. console.log('我是页面组件的中间件!')
  6. },
  7. validate(context){
  8. const {params, query} = context;
  9. //对传过来的params和query进行校验
  10. //...
  11. console.log('我是页面validate参数校验')
  12. return true; // 校验通过,校验失败返回false,进入到404页面
  13. },
  14. components: {
  15. Logo
  16. }
  17. }

 4.asyncData()和fetch()

这俩个钩子也是在页面组件中使用,pages/index.vue中:

  1. asyncData(context){
  2. // 异步读取数据,处理相关业务逻辑
  3. console.log('我是asyncData')
  4. //最后返回数据
  5. return {
  6. a: 1
  7. };
  8. },
  9.  
  10. fetch(context){
  11. // 异步读取数据,处理相关业务逻辑,将数据提交给vuex
  12. console.log('我是fetch')
  13. },

5.beforeCreate()和created()

在pages/index.vue中:

  1. ...
  2. fetch(context){
  3. // 异步读取数据,处理相关业务逻辑,将数据提交给vuex
  4. console.log('我是fetch')
  5. },
  6. beforeCreate() {
  7. console.log('我是beforeCreate')
  8. },
  9. created() {
  10. console.log('我是create')
  11. },
  12. ...

客户端:

服务端:

二、路由(约定式、自定义)

1.约定式路由:依据pages目录结构自动生成 vue-router模块的路由配置。

1>基础路由:

在layouts目录中的default.vue:

  1. <template>
  2. <div>
  3. <!-- 声明式跳转:nuxt-link跟vue中的router-link一样,负责路由跳转 -->
  4. <nuxt-link to="/">首页</nuxt-link>
  5. <nuxt-link to="/tabA">tabA</nuxt-link>
  6. <nuxt-link to="/tabB">tabB</nuxt-link>
  7.  
  8. <!-- 展示区,跟vue中的router-view一样 -->
  9. <nuxt/>
  10. </div>
  11. </template>

效果:

nuxt自动生成的路由为:

  1. router: {
  2. routes: [
  3. {
  4. name: 'index',
  5. path: '/',
  6. component: 'pages/index.vue'
  7. },
  8. {
  9. name: 'tabA',
  10. path: '/tabA',
  11. component: 'pages/tabA.vue'
  12. },
  13. {
  14. name: 'tabB',
  15. path: '/tabB',
  16. component: 'pages/tabB.vue'
  17. }
  18. ]
  19. }

2>动态路由

在 Nuxt.js 里面定义带参数的动态路由,需要创建对应的以下划线“_”作为前缀的 Vue 文件 或 目录。

pages/tabA.vue:

  1. <template>
  2. <div>
  3. <h3>
  4. tabA
  5. </h3>
  6. <!-- 俩种路由传参方式 -->
  7. <nuxt-link to="/tabA/1?a=tabA1">tabA动态路由1</nuxt-link>
  8. <!-- name: 'tabA-id',这里-id必须是tabA目录下的_id.vue的名称,且将 _ 改为 - -->
  9. <nuxt-link :to="{name: 'tabA-id', params:{id: 2}, query:{a: 'tabA2'}}">tabA动态路由2</nuxt-link>
  10.  
  11. <!-- 子路由的展示区 -->
  12. <nuxt/>
  13. </div>
  14. </template>

pages/tabA/_id.vue:

  1. <template>
  2. <div>
  3. <h3>我是tabA的动态路由页</h3>
  4. </div>
  5. </template>

效果:

这里会发现一个问题,路由为/tabA的时候,也打开了动态路由页,这是因为当路由为tabA时,会发现它下面还有子路由,由于没有指定默认的页面,因此会找到一个子路由进行渲染。为了解决这个问题,可以在tabA目录下新建一个index.vue。

tabA/index.vue:

  1. <template>
  2. <h3>我是动态子路由默认页</h3>
  3. </template>

此时nuxt自动生成的路由信息为:

  1. router: {
  2. routes: [
  3. {
  4. name: 'index',
  5. path: '/',
  6. component: 'pages/index.vue'
  7. },
  8. {
  9. name: 'tabA-id',
  10. path: '/tabA/:id',
  11. component: 'pages/tabA/_id.vue'
  12. },
  13. ...
  14. ]
  15. }

3>嵌套路由

创建内嵌子路由,你需要添加一个 Vue 文件,同时添加一个与该文件同名的目录用来存放子视图组件。

在tabA目录下新建tabC目录和tabC.vue

tabA/tabC.vue:

  1. <template>
  2. <div>
  3. <!-- 子路由展示区 -->
  4. <nuxt></nuxt>
  5. </div>
  6. </template>

tabA/tabC/index.vue:

  1. <template>
  2. <div>
  3. <h3>我是tabC</h3>
  4. </div>
  5. </template>

从上图可以看到俩种颜色的区域,是俩级展示区,通常我们一个页面只需要一个展示区,因此需要改一下项目的目录结构:

在pages目录下的页面为一级展示区,其中有tabA、tabB,接下来将tabA目录下的页面也设置成一级展示区。将tabA.vue内容拷贝至tabA目录下的index.vue中,并删除tabA.vue。

2、扩展路由

nuxt中支持声明式路由,同样也支持自定义路由,根据路由渲染指定的视图,需要在nuxt.config.js中配置:

  1. router: {
  2. extendRoutes(routes, resolve){
  3. routes.push({
  4. name: 'tabD',
  5. path: '/tabD',
  6. component: resolve(__dirname, 'pages/extendTabD.vue')
  7. })
  8. }
  9. },

在pages中新建extendTabD.vue:

pages/extendTabD.vue

  1. <template>
  2. <div>
  3. <h3>我是扩展路由tabD</h3>
  4. </div>
  5. </template>

在layouts/default.vue中增加链接: <nuxt-link to="/tabD">tabD</nuxt-link>

3、路由过渡动画

1>全局过渡动画:

transition.css:

  1. /* 路由统一动效 */
  2.  
  3. /* 动画形式 */
  4. .page-enter-active, .page-leave-active{
  5. transition: opacity .5s;
  6. }
  7.  
  8. /* 入 退 */
  9. .page-enter, .page-leave-active{
  10. opacity: 0;
  11. }

nuxt.config.js:

  1. /*
  2. ** Global CSS
  3. */
  4. css: [
  5. 'assets/css/transition.css'
  6. ],

2>局部过渡动画

extendTabD.vue:

  1. <template>
  2. <div>
  3. <h3>我是扩展路由tabD</h3>
  4. </div>
  5. </template>
  6.  
  7. <script>
  8. export default {
  9. name: "extendTabD",
  10. transition: 'tabd'
  11. }
  12. </script>
  13.  
  14. <style scoped>
  15. .tabd-enter-active, .tabd-leave-active{
  16. transition: .5s ease all;
  17. }
  18. .tabd-enter, .tabd-leave-active{
  19. margin-left: -1000px;
  20. }
  21. </style>

 4、路由守卫

  • 前置
  • 后置
  • 组件内部

1>前置:

全局守卫:

  • 在nuxt.config.js的router内引入middleware
  • 在layouts文件中定义middleware
  • 在插件中定义前置全局守卫,beforeEach()

组件独享守卫:

  • 在页面中引入middleware或middleware()
  • 跟vue-router一样的组件内部钩子函数,比如:beforeRouteEnter

2>后置:

在插件中定义后置全局守卫,afterEach()

具体实现:

全局守卫:

a.借助middleware实现前置的路由守卫,在生命周期的middleware中已经基本实现。

middleware/isLogin.js:

  1. export default (context) => {
  2. console.log('我是全局守卫')
  3. const {redirect} = context
  4. redirect('/tabD') // 利用redirect()页面重定向
  5. }

b.借助插件实现前置、后置全局守卫:

plugins/router.js

  1. export default ({app, redirect})=>{
  2. //app == vue实例
  3. //全局前置守卫,进入的页面不是tabA的页面,都会重定向到tabA
  4. app.router.beforeEach((to, from, next)=>{
  5. if(to.name == 'tabA'){
  6. next()
  7. }else{
  8. redirect({name: 'tabA'})
  9. }
  10. });
  11. //全局后置守卫
  12. app.router.afterEach((to, from)=>{
  13. console.log('我是后置守卫')
  14. })
  15. }

nuxt.config.js中配置plugins:

  1. plugins: [
  2. '~/plugins/router.js'
  3. ],

组件内守卫:

extendTabD.vue:

  1. <template>
  2. <div>
  3. <h3>我是扩展路由tabD</h3>
  4. </div>
  5. </template>
  6.  
  7. <script>
  8. export default {
  9. name: "extendTabD",
  10. transition: 'tabd',
  11. beforeRouteEnter(to, from, next){
  12. alert('haha');
  13. }
  14. }
  15. </script>

三、vuex状态树

nuxt中集成了vuex,因此我们在项目中不需要下载引入,而是按照约定使用:

nuxt会尝试找到 src 目录(默认是应用根目录)下的 store 目录,如果该目录存在,它将做以下的事情:

  1. 引入vuex模块
  2. 将vuex模块加到vendors构建配置中
  3. 设置vue根实例的store配置项

nuxt中的vuex使用方式跟vue中大同小异,除了不需要手动引入之外,在模块化中,store目录下的每个子模块(除了根模块index.js),都会被转换成为状态树指定命名的子模块,其它使用方式跟vue项目中的vuex基本相同。

store/index.js:

  1. const state = ()=>({
  2.  
  3. })
  4. const mutations = {
  5.  
  6. }
  7.  
  8. const actions = {
  9.  
  10. }
  11. export default {
  12. state,
  13. mutations,
  14. actions
  15. }

store/user.js:

  1. const state = () => ({
  2. isLogin: false
  3. })
  4.  
  5. const mutations = {
  6. change_Login(state, isLogin) {
  7. state.isLogin = isLogin
  8. }
  9. }
  10.  
  11. const actions = {
  12. changeLogin({commit},isLogin) {
  13. commit('change_Login', isLogin)
  14. }
  15. }
  16. export default {
  17. namespaced: true,
  18. state,
  19. mutations,
  20. actions
  21. }

pages/tabB.vue:

  1. <template>
  2. <div>
  3. <h3>tabB</h3>
  4. <h4>登录状态:{{isLogin ? '已登录' : '未登录'}}</h4>
  5. </div>
  6. </template>
  7.  
  8. <script>
  9. import { mapState } from 'vuex';
  10. export default {
  11. name: "TabB",
  12. computed:{
  13. ...mapState({
  14. isLogin: state => state.user.isLogin
  15. })
  16. },
  17. created() {
  18. setTimeout(()=>{
  19. this.$store.dispatch('user/changeLogin', true)
  20. }, 1000)
  21. }
  22. }
  23. </script>

效果:一秒后,登录状态从“未登录”变为“已登录”。

以上就是我对nuxt学习过程中一些重要概念的理解和实践,有什么不足欢迎评论指正!

脚踏实地行,海阔天空飞

vue服务端渲染之nuxtjs的更多相关文章

  1. [vue] vue服务端渲染nuxt.js

    初始化 使用脚手架工具 create-nuxt-app 快速创建 npx create-nuxt-app <项目名> npx create-nuxt-app 执行一些选择 在集成的服务器端 ...

  2. Egg + Vue 服务端渲染工程化实现

    在实现 egg + vue 服务端渲染工程化实现之前,我们先来看看前面两篇关于Webpack构建和Egg的文章: 在 Webpack工程化解决方案easywebpack 文章中我们提到了基于 Vue ...

  3. vue服务端渲染axios预取数据

    首先是要参考vue服务端渲染教程:https://ssr.vuejs.org/zh/data.html. 本文主要代码均参考教程得来.基本原理如下,拷贝的原文教程. 为了解决这个问题,获取的数据需要位 ...

  4. vue服务端渲染简单入门实例

    想到要学习vue-ssr的同学,自不必多说,一定是熟悉了vue,并且多多少少做过几个项目.然后学习vue服务端渲染无非解决首屏渲染的白屏问题以及SEO友好. 话不多说,笔者也是研究多日才搞明白这个服务 ...

  5. vue服务端渲染提取css

    vue服务端渲染,提取css单独打包的好处就不说了,在这里主要说的是抽取css的方法 要从 *.vue 文件中提取 CSS,可以使用 vue-loader 的 extractCSS 选项(需要 vue ...

  6. 解析Nuxt.js Vue服务端渲染摸索

    本篇文章主要介绍了详解Nuxt.js Vue服务端渲染摸索,写的十分的全面细致,具有一定的参考价值,对此有需要的朋友可以参考学习下.如有不足之处,欢迎批评指正. Nuxt.js 十分简单易用.一个简单 ...

  7. Nuxt.js vue服务端渲染

    一.为什么要用Nuxt.js 原因其实不用多说,就是利用Nuxt.js的服务端渲染能力来解决Vue项目的SEO问题. 二.Nuxt.js和纯Vue项目的简单对比 1. build后目标产物不同 vue ...

  8. 实例PK(Vue服务端渲染 VS Vue浏览器端渲染)

    Vue 2.0 开始支持服务端渲染的功能,所以本文章也是基于vue 2.0以上版本.网上对于服务端渲染的资料还是比较少,最经典的莫过于Vue作者尤雨溪大神的 vue-hacker-news.本人在公司 ...

  9. Vue服务端渲染和Vue浏览器端渲染的性能对比

    Vue 2.0 开始支持服务端渲染的功能,所以本文章也是基于vue 2.0以上版本.网上对于服务端渲染的资料还是比较少,最经典的莫过于Vue作者尤雨溪大神的 vue-hacker-news.本人在公司 ...

随机推荐

  1. 1、mysql基础入门(1)

    1.mysql基础入门: 1.1.数据库介绍:

  2. POJ 3761:Bubble Sort——组合数学

    题目大意:众所周知冒泡排序算法多数情况下不能只扫描一遍就结束排序,而是要扫描好几遍.现在你的任务是求1~N的排列中,需要扫描K遍才能排好序的数列的个数模20100713.注意,不同于真正的冒泡排序算法 ...

  3. 1.5Java、万维网以及其他

    要点提示:Java是一种功能强大和多用途的编程语言,可用于开发运行在移动设备.台式计算机以及服务器端的软件.

  4. django 使用jpype 报错:raise+OSError('JVM+cannot+be+restarted')

    #调用jar包 def getJar(arg1,arg2): jarpath = os.path.join(os.path.abspath('.'), 'tools/GetTest-1.0-SNAPS ...

  5. 【重学Java】多线程基础(三种创建方式,线程安全,生产者消费者)

    实现多线程 简单了解多线程[理解] 是指从软件或者硬件上实现多个线程并发执行的技术. 具有多线程能力的计算机因有硬件支持而能够在同一时间执行多个线程,提升性能. 并发和并行[理解] 并行:在同一时刻, ...

  6. [转载]API网关

    1. 使用API网关统一应用入口 API网关的核心设计理念是使用一个轻量级的消息网关作为所有客户端的应用入口,并且在 API 网关层面上实现通用的非功能性需求.如下图所示:所有的服务通过 API 网关 ...

  7. python使用笔记26--多线程、多进程

    1.概念 线程.进程 进程是资源的集合,也就是一个程序 线程是一个程序运行的最小单位 线程是在进程里面的 默认,一个进程就只有一个线程 一个电脑有几核CPU就只能同时运行几个任务,比如4核CPU只能同 ...

  8. 基于Ryu的流量采集代码实现

    1 from __future__ import division 2 import time 3 import math 4 import xlwt 5 from ryu.controller im ...

  9. webdriver xpath

    aa=wd.find_elements_by_xpath('//a') for a in aa: print(a.text) #显示所有A标签中文本 aa=wd.find_elements_by_xp ...

  10. HTML5-CSS(一)

    一.创建 CSS 样式表有三种方式 1. 元素内嵌样式<p style="color:red;font-size:50px;">这是一段文本</p>解释:即 ...