基于Vue SEO的四种方案(小结)


Posted in Javascript onJuly 01, 2019

前言:众所周知,Vue SPA单页面应用对SEO不友好,当然也有相应的解决方案,下面列出几种最近研究和使用过的SEO方案,SRR和静态化基于Nuxt来说。

1.SSR服务器渲染;
2.静态化;
3.预渲染prerender-spa-plugin;
4.使用Phantomjs针对爬虫做处理。

1.SSR服务器渲染

关于服务器渲染:Vue官网介绍 ,对Vue版本有要求,对服务器也有一定要求,需要支持nodejs环境。

使用SSR权衡之处:

  1. 开发条件所限,浏览器特定的代码,只能在某些生命周期钩子函数 (lifecycle hook) 中使用;一些外部扩展库 (external library) 可能需要特殊处理,才能在服务器渲染应用程序中运行;
  2. 环境和部署要求更高,需要Node.js server 运行环境;
  3. 高流量的情况下,请准备相应的服务器负载,并明智地采用缓存策略。

优势:

  1. 更好的 SEO,由于搜索引擎爬虫抓取工具可以直接查看完全渲染的页面;
  2. 更快的内容到达时间 (time-to-content),特别是对于缓慢的网络情况或运行缓慢的设备。

不足:(开发中遇到的坑)

1.一套代码两套执行环境,会引起各种问题,比如服务端没有window、document对象,处理方式是增加判断,如果是客户端才执行:

if(process.browser){
 console.log(window);
}

引用npm包,带有dom操作的,例如: wowjs ,不能用 import 的方式,改用:

if (process.browser) {
  var { WOW } = require('wowjs');
  require('wowjs/css/libs/animate.css');
 }

2.Nuxt asyncData方法,初始化页面前先得到数据,但仅限于 页面组件 调用:

// 并发加载多个接口:
 async asyncData ({ app, query }) {
 let [resA, resB, resC] = await Promise.all([
  app.$axios.get('/api/a'),
  app.$axios.get('/api/b'),
  app.$axios.get('/api/c'),
  ])
  
  return {
  dataA: resA.data,
  dataB: resB.data,
  dataC: resC.data,
  }
 }

在asyncData中获取参数:

1.获取动态路由参数,如:

/list/:id' ==>  '/list/123

接收:

async asyncData ({ app, query }) {
 console.log(app.context.params.id) //123
}

2.获取url?获取参数,如:

/list?id=123

接收:

async asyncData ({ app, query }) {
 console.log(query.id) //123
}

3.如果你使用 v-if 语法,部署到线上大概也会遇到这个错误:

Error while initializing app DOMException: Failed to execute 'appendChild' on 'Node': This node type does not support this method.
    at Object.We [as appendChild]

根据github nuxt上的 issue第1552条 提示,要将 v-if 改为 v-show 语法。

4.坑太多,留坑,晚点更。

2.静态化

在 Nuxt.js 执行 generate 静态化打包时,动态路由会被忽略。

-| pages/
---| index.vue
---| users/
-----| _id.vue

需要动态路由先生成静态页面,你需要指定动态路由参数的值,并配置到 routes 数组中去。

// nuxt.config.js
module.exports = {
 generate: {
 routes: [
  '/users/1',
  '/users/2',
  '/users/3'
 ]
 }
}

运行打包,即可看见打包出来的页面。

但是如果路由动态参数的值是动态的而不是固定的,应该怎么做呢?

使用一个返回 Promise 对象类型 的 函数;
使用一个回调是 callback(err, params) 的 函数。

// nuxt.config.js
import axios from 'axios'

export default {
 generate: {
 routes: function () {
  return axios.get('https://my-api/users')
  .then((res) => {
  return res.data.map((user) => {
   return {
   route: '/users/' + user.id,
   payload: user
   }
  })
  })
 }
 }
}

现在我们可以从 /users/_id.vue 访问的 payload ,如下所示:

async asyncData ({ params, error, payload }) {
 if (payload) return { user: payload }
 else return { user: await backend.fetchUser(params.id) }
}

如果你的动态路由的参数很多,例如商品详情,可能高达几千几万个。需要一个接口返回所有id,然后打包时遍历id,打包到本地,如果某个商品修改了或者下架了,又要重新打包,数量多的情况下打包也是非常慢的,非常不现实。

优势:

  • 纯静态文件,访问速度超快;
  • 对比SSR,不涉及到服务器负载方面问题;
  • 静态网页不宜遭到黑客攻击,安全性更高。

不足:

  • 如果动态路由参数多的话不适用。

3.预渲染prerender-spa-plugin

如果你只是用来改善少数营销页面(例如 /, /about, /contact 等)的 SEO,那么你可能需要预渲染。无需使用 web 服务器实时动态编译 HTML,而是使用预渲染方式,在构建时 (build time) 简单地生成针对特定路由的静态 HTML 文件。优点是设置预渲染更简单,并可以将你的前端作为一个完全静态的站点。

$ cnpm install prerender-spa-plugin --save

vue cli 3 vue.config.js 配置:

const PrerenderSPAPlugin = require('prerender-spa-plugin');
const Renderer = PrerenderSPAPlugin.PuppeteerRenderer;
const path = require('path');
module.exports = {
 configureWebpack: config => {
  if (process.env.NODE_ENV !== 'production') return;
  return {
   plugins: [
    new PrerenderSPAPlugin({
     // 生成文件的路径,也可以与webpakc打包的一致。
     // 下面这句话非常重要!!!
     // 这个目录只能有一级,如果目录层次大于一级,在生成的时候不会有任何错误提示,在预渲染的时候只会卡着不动。
     staticDir: path.join(__dirname,'dist'),
     // 对应自己的路由文件,比如a有参数,就需要写成 /a/param1。
     routes: ['/', '/product','/about'],
     // 这个很重要,如果没有配置这段,也不会进行预编译
     renderer: new Renderer({
      inject: {
       foo: 'bar'
      },
      headless: false,
      // 在 main.js 中 document.dispatchEvent(new Event('render-event')),两者的事件名称要对应上。
      renderAfterDocumentEvent: 'render-event'
     })
    }),
   ],
  };
 }
}

在main.js中添加:

new Vue({
 router,
 render: h => h(App),
 mounted () {
 document.dispatchEvent(new Event('render-event'))
 }
}).$mount('#app')

注意:router中必须设置 mode: “history” 。

打包出来可以看见文件,打包出文件夹 /index.html ,例如: about => about/index.html ,里面有html内容。

优势:

  • 改动小,引入个插件就完事;

不足:

  • 无法使用动态路由;
  • 只适用少量页面的项目,页面多达几百个的情况下,打包会很很很慢;

4.使用Phantomjs针对爬虫做处理

Phantomjs是一个基于webkit内核的无头浏览器,即没有UI界面,即它就是一个浏览器,只是其内的点击、翻页等人为相关操作需要程序设计实现。

虽然“PhantomJS宣布终止开发”,但是已经满足对Vue的SEO处理。

这种解决方案其实是一种旁路机制,原理就是通过Nginx配置, 判断访问的来源UA是否是爬虫访问,如果是则将搜索引擎的爬虫请求转发到一个node server,再通过PhantomJS来解析完整的HTML,返回给爬虫。

 基于Vue SEO的四种方案(小结)

具体代码戳这里: vue-seo-phantomjs 。

要安装全局 phantomjs ,局部 express ,测试:

$ phantomjs spider.js 'https://www.baidu.com'

如果见到在命令行里出现了一推html,那恭喜你,你已经征服PhantomJS啦。

启动之后或者用postman在请求头增加 User-Agent 值为 Baiduspider ,效果一样的。

部署上线

线上要安装 node pm2 phantomjs ,nginx相关配置:

upstream spider_server {
 server localhost:3000;
}

server {
 listen  80;
 server_name example.com;
 
 location / {
  proxy_set_header Host   $host:$proxy_port;
  proxy_set_header X-Real-IP  $remote_addr;
  proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;

  if ($http_user_agent ~* "Baiduspider|twitterbot|facebookexternalhit|rogerbot|linkedinbot|embedly|quora link preview|showyoubot|outbrain|pinterest|slackbot|vkShare|W3C_Validator|bingbot|Sosospider|Sogou Pic Spider|Googlebot|360Spider") {
  proxy_pass http://spider_server;
  }
 }
}

优势:

  • 完全不用改动项目代码,按原本的SPA开发即可,对比开发SSR成本小不要太多;
  • 对已用SPA开发完成的项目,这是不二之选。

不足:

  • 部署需要node服务器支持;
  • 爬虫访问比网页访问要慢一些,因为定时要定时资源加载完成才返回给爬虫;
  • 如果被恶意模拟百度爬虫大量循环爬取,会造成服务器负载方面问题,解决方法是判断访问的IP,是否是百度官方爬虫的IP。

总结

如果构建大型网站,如商城类,别犹豫,直接上SSR服务器渲染,当然也有相应的坑等你,社区较成熟,英文好点,一切问题都迎刃而解。

如果只是个人博客、公司官网这类,其余三种都可以。

如果对已用SPA开发完成的项目进行SEO优化,而且支持node服务器,请使用 Phantomjs 。

很少写文章,这是我这个月对Vue SEO方案的探索,写的不对的地方请指出,谢谢理解~

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

Javascript 相关文章推荐
YUI 读码日记之 YAHOO.lang.is*
Mar 22 Javascript
jQuery 插件 将this下的div轮番显示
Apr 09 Javascript
JavaScript表达式:URL 协议介绍
Mar 10 Javascript
js模仿java的Map集合详解
Jan 06 Javascript
全面解析Bootstrap布局组件应用
Feb 22 Javascript
JavaScript蒙板(model)功能的简单实现代码
Aug 04 Javascript
详解JS几种变量交换方式以及性能分析对比
Nov 25 Javascript
jQuery UI仿淘宝搜索下拉列表功能
Jan 10 Javascript
Angular4学习教程之HTML属性绑定的方法
Jan 04 Javascript
微信小程序 如何引入外部字体库iconfont的图标
Jan 31 Javascript
解决vue项目中type=”file“ change事件只执行一次的问题
May 16 Javascript
js+html实现周岁年龄计算器
Jun 25 Javascript
JavaScript一元正号运算符示例代码
Jun 30 #Javascript
重学JS之显示强制类型转换详解
Jun 30 #Javascript
JavaScript判断浏览器运行环境的详细方法
Jun 30 #Javascript
微信小程序如何自定义table组件
Jun 29 #Javascript
微信小程序如何调用图片接口API并居中显示
Jun 29 #Javascript
微信小程序如何调用json数据接口并解析
Jun 29 #Javascript
pm2启动ssr失败的解决方法
Jun 29 #Javascript
You might like
PHP制作图型计数器的例子
2006/10/09 PHP
php批量上传的实现代码
2013/06/09 PHP
PHP文件锁定写入实例解析
2014/07/14 PHP
PHP中使用SimpleXML检查XML文件结构实例
2015/01/07 PHP
php检查日期函数checkdate用法实例
2015/03/19 PHP
php实现的操作excel类详解
2016/01/15 PHP
Laravel框架搜索分页功能示例
2019/02/01 PHP
在laravel-admin中列表中禁止某行编辑、删除的方法
2019/10/03 PHP
ymPrompt的doHandler方法来实现获取子窗口返回值的方法
2010/06/25 Javascript
JavaScript 和 Java 的区别浅析
2013/07/31 Javascript
jQuery读取和设定KindEditor值的方法
2013/11/22 Javascript
jquery easyui combox一些实用的小方法
2013/12/25 Javascript
js加入收藏以及使用Jquery更改透明度
2014/01/26 Javascript
JS操作iframe里的dom(实例讲解)
2014/01/29 Javascript
兼容主流浏览器的JS复制内容到剪贴板
2014/12/12 Javascript
JavaScript 学习笔记之基础中的基础
2015/01/13 Javascript
JQuery zClip插件实现复制页面内容到剪贴板
2015/11/02 Javascript
Js获取当前日期时间及格式化代码
2016/09/17 Javascript
BootStrap Table 后台数据绑定、特殊列处理、排序功能
2017/05/27 Javascript
Js实现京东无延迟菜单效果实例(demo)
2017/06/02 Javascript
JavaScript实现像雪花一样的Hexaflake分形
2020/07/07 Javascript
Python Mysql数据库操作 Perl操作Mysql数据库
2009/01/12 Python
解决python2.7用pip安装包时出现错误的问题
2017/01/23 Python
Python拼接字符串的7种方法总结
2018/11/01 Python
解决python 未发现数据源名称并且未指定默认驱动程序的问题
2018/12/07 Python
解析浏览器的一些“滚动”行为鉴赏
2019/09/16 HTML / CSS
白兰氏健康Mall:BRAND’S
2017/11/13 全球购物
马来西亚综合购物网站:Lazada马来西亚
2018/06/05 全球购物
乌克兰巴士票购买网站:inBus
2021/03/12 全球购物
工厂车间标语
2014/06/19 职场文书
十八大标语口号
2014/10/09 职场文书
教师群众路线教育实践活动个人对照检查材料
2014/11/04 职场文书
辩论赛开场白大全(主持人+辩手)
2015/05/29 职场文书
我的暑假生活作文(五年级)范文
2019/08/07 职场文书
Mysql中调试存储过程最简单的方法
2021/06/30 MySQL
win10如何快速切换窗口 win10切换窗口快捷键分享
2022/07/23 数码科技