学习nuxt的一个根本原因就是为了SEO,要了解SEO,我们要知道,浏览器爬虫的工作流程以及SPA、SSG、SSR这些概念只有这样我们才能了解到为什么学习Nuxt以及为什么学习Nuxt主要解决了什么问题
单页面程序(SPA)全称是:Single-page-application,SPA应用是在客户端(即浏览器端渲染)术语称CSR
静态站点生成(SSG)全称是:Static site Generate,即预先生成好的静态网站,一般用于官方文档或者博客网站比较多
服务器端渲染(SSR)全称是:Server Side Render,即在服务器端渲染页面,将渲染好的HTML返回给浏览器呈现
学习SSR的一大目的就是利于SEO,要搞清楚这些就要知道爬虫的基本工作流程
我们在使用浏览器搜索时,浏览器给我们呈现的内容就是爬虫爬取结果后根据一定的算法呈现出来的,简单来说浏览器整个就是爬虫爬取出来的,那么它的工作流程到底是什么样呢
爬虫会在网络中发现各种网页,将网页中的爬取的内容存放到临时库中,网页中如果遇到其他网站,就重复该过程
爬取完结果后,爬虫会对爬取的数据进行分析(例如title元素、图片、视频等),将爬取的网页进行归档分类,并且会对临时库中的信息进行筛选不符合规则的会被清理,最后会把爬虫的结果符合规则的存放到索引区供用户搜索时呈现
用户搜索时,搜索引擎会根据内容的类型,选择一组更加符合规则的呈现给用户
SEO是搜索引擎优化(Search Engine Optimization)的缩写,是一种通过优化网站的内容、结构和技术等方面,以提高网站在搜索引擎中的排名和曝光度的方法和策略。
SEO是为了提高网站在搜索引擎中的排名和曝光度,了解了爬虫的工作流程,我们就知道排序中最重要的是索引编制阶段,而该阶段又借助于我们的页面内容,SPA返回的又是空白页面,所以不利于SEO
我们采用Node和webpack来搭建vue的SSR项目,这个只是一个简单的项目,主要目的是为了了解一下搭建SSR的流程,后面我也会使用Nuxt进行搭建Vue SSR项目
安装依赖
npm i express
npm i -D nodemon
npm i -D webpack webpack-cli webpack-node-externals
npm i vue
npm i -D vue-loader
npm i -D babel-loader @babel/preset-env
npm i -D webpack-merge webpack-node-externals
npm i vue-router -D
npm i pinia
project
|
└───build 打包后的代码
│ │
│ └───client
│ └───server
│
└───config 打包配置文件
| └─── base.config.js
| └─── client.config.js
| └─── server.config.js
│
└── src 源代码
| └───client 客户端
| | └─── index.js 入口文件
| └───router 路由
| | └─── index.js 入口文件
| └───server 服务器端
| | └─── index.js 入口文件
| └───store pinia
| | └─── index.js 入口文件
| └───views 视图
| | └─── about.vue
| | └─── home.vue
| └───app.js
| └───App.vue
| └───package-lock.josn
| └───package.json
"scripts": {
"test": "echo \"Error: no test specified\" && exit 1",
"dev": "nodemon ./src/server/index",
"start": "nodemon ./build/server/server_bundle.js",
"build:server": "webpack --config ./config/server.config.js --watch",
"build:client": "webpack --config ./config/client.config.js --watch"
},
命令详情:
需要注意的是我们在运行时是要有顺序的,要先把客户端和服务端代码打包后再启动服务
配置文件是webpack打包配置,我们需要打包服务器端代码和客户端代码,由于两者打包配置有重复点所以新建一个基础的配置文件,并使用webpack-merge去合并配置
let { VueLoaderPlugin } = require("vue-loader/dist/index");
module.exports = {
mode: "development",
module: {
rules: [
{
test: /\.js$/,
loader: "babel-loader",
options: {
presets: ["@babel/preset-env"],
},
},
{
test: /\.vue$/,
loader: "vue-loader",
},
],
},
plugins: [new VueLoaderPlugin()],
resolve: {
// 添加的后缀,项目中导包就不需要编写文件后缀
extensions: [".js", ".json", ".wasm", ".jsx", ".vue"],
},
};
let path = require("path");
let { DefinePlugin } = require("webpack");
let { merge } = require("webpack-merge");
let baseConfig = require("./base.config");
module.exports = merge(baseConfig, {
target: "web", //fs path
entry: "./src/client/index.js",
output: {
filename: "client_bundle.js",
path: path.resolve(__dirname, "../build/client"),
},
plugins: [
new DefinePlugin({
__VUE_OPTIONS_API__: false,
__VUE_PROD_DEVTOOLS__: false,
}),
],
});
let path = require("path");
let nodeExternals = require("webpack-node-externals");
let { VueLoaderPlugin } = require("vue-loader/dist/index");
let { merge } = require("webpack-merge");
let baseConfig = require("./base.config");
module.exports = merge(baseConfig, {
target: "node", //fs path
entry: "./src/server/index.js",
output: {
filename: "server_bundle.js",
path: path.resolve(__dirname, "../build/server"),
},
externals: [nodeExternals()], //排除node_module中的包
});
import { createApp } from "vue";
import App from "../App.vue";
import createRouter from "../router";
import { createWebHashHistory } from "vue-router";
import { createPinia } from "pinia";
let app = createApp(App);
let router = createRouter(createWebHashHistory());
app.use(router);
let pinia = createPinia();
app.use(pinia);
router.isReady().then(() => {
//等待路由加载完成之后再挂载
app.mount("#app");
});
import { createRouter } from "vue-router";
const routes = [
{
path: "/",
component: () => import("../views/home.vue"),
},
{
path: "/about",
component: () => import("../views/about.vue"),
},
];
export default function (history) {
return new createRouter({
history,
routes,
});
}
let express = require("express");
let server = express();
import createApp from "../app";
import { renderToString } from "@vue/server-renderer";
// 部署静态资源
server.use(express.static("build"));
import createRouter from "../router";
// 内存路由=>node用
import { createMemoryHistory } from "vue-router";
import { createPinia } from "pinia";
server.get("/*", async (req, res) => {
let app = createApp();
let router = createRouter(createMemoryHistory());
/*
服务器端和客户端都注册路由的原因是为了实现路由同步
用户进入页面时将渲染好的字符串返回(服务器端返回正确的html字符串)
在页面跳转可以无刷新跳转(客户端可以继续跳转)
*/
app.use(router);
// 跳转页面(路由跳转完成之后再渲染)
await router.push(req.url || "/");
await router.isReady(); //等待(异步)路由加载完成,再渲染页面
// 创建pinia
const pinpa = createPinia();
app.use(pinpa);
// 注册路由
let appStringHtml = await renderToString(app);
res.send(`
Document
${appStringHtml}
`);
});
server.listen(3000, () => {
console.log("服务器启动成功");
});
import { defineStore } from "pinia";
export const useHomeStore = defineStore("home", {
state() {
return {
count: 1,
};
},
actions: {
increment() {
this.count++;
},
decrement() {
this.count--;
},
},
});
About
{{ count }}
Home
{{ count }}
import { createSSRApp } from "vue";
import App from "./App.vue";
// 写函数返回app实例,作用是避免跨请求状态的污染
// 通过函数来返回app实例,可以保证每个请求都会返回一个新的app实例
export default function createApp() {
return createSSRApp(App);
}
Vue App
{{ count }}
首页
关于
为了方便大家查看,我这里附上源代码的仓库连接
https://github.com/XY0987/blog_vue3_ssr
Nuxt.js是一个基于Vue.js的通用应用框架,它可以帮助开发者快速构建高性能的单页应用(SPA)和静态网站。Nuxt.js基于Vue.js的生态系统,提供了许多有用的功能和约定,使得开发过程更加简单和高效。
mpx nuxi init 项目名
pnpm dix nuxi init 项目名
npm i -g nuxi
nuxi init 项目名
由于墙的原因,我们再构建项目时大概率会报错,这是我们有两种解决方法
配置host,本地dns解析
Mac电脑host配置路径: /stc/hosts
Window电脑host配置路径: c/Windows/System32/drivers/etc/hosts
新增配置
185.199.110.133 raw.githubusercontent.com
手动克隆项目(有时候我们配置上述方法也不行,比如我的电脑就不行,挂也不行,只能开加速器手动克隆了)
git clone -b v3 https://github.com/nuxt/starter.git 文件夹名
npm i
npm run build
npm run dev
npm run generate
npm run preview
npm run postinstall
project
|
└───assets 静态资源
│
└───components 组件
│
└── composables 组合API
|
└── layout 自定义布局
|
└── pages 页面,nuxt会根据页面目录结构和文件名自动注册路由
|
└── plugins 插件
|
└── app.vue 入口文件
|
└── app.config.ts 配置文件
|
└── nuxt.config.ts nuxt配置文件
|
└── package-lock.json
|
└── package.json
|
└── tsconfig.json ts配置
在学习过程中遇到比较难受点就是下载依赖比较卡顿,还有就是有时候即使跟着网上的教程但是还是不行,chargpt给答案也是模棱两可,还是得去看文档或者去github中的issues去找答案