nuxt3+ts+vue3的ssr项目总结

目录

一、什么是SSR、SEO、SPA,它们之间的关系又是怎样的。

二、VUE做SSR的几种方法

1、插件prerender-spa-plugin

2、VUE开启SSR渲染模式

3、使用NUXT框架 

三、NUXT3+VUE3+TS

(一)基本配置

1、文件夹介绍

assets

components

pages

api

plugins

utils 

2、布局使用

3、useFetch请求封装

(二)引入element-pluse

1、安装

2、引入

3、使用

(三)引入md文档

1、下载

2、引入

3、使用

用于输入数据

用于回显数据

(四)v-show与v-if踩坑

1、区别

2、问题

(五)el- select与el-tooltip踩坑

1、问题


一、什么是SSR、SEO、SPA,它们之间的关系又是怎样的。

SSR (Server-Side Rendering)服务器端渲染,是一种网页渲染方法,其中服务器在向客户端发送 HTML 之前将页面渲染为完全呈现的 HTML。在 SSR 中,服务器会处理页面的渲染,然后将呈现的 HTML 发送到客户端浏览器,浏览器接收到的是已经包含内容的页面。这有助于搜索引擎更好地索引页面内容,提高首次加载速度,以及有助于 SEO(搜索引擎优化)。

SEO (Search Engine Optimization)搜索引擎优化,是一系列技术和策略,旨在改进网站在搜索引擎结果页面上的排名,以增加网站的曝光度和访问量。在网站的 SEO 中,内容的可索引性、关键词的使用、页面速度和用户体验等都是重要因素。

SPA (Single Page Application)单页面应用,是一种基于 Web 的应用程序,它在加载初始页面后,通过 AJAX 或其他技术在同一个页面上动态加载内容,而不需要完整的页面刷新。SPA 通常在客户端使用 JavaScript 处理页面路由和渲染。SPA 的优点包括更流畅的用户体验,但它可能对搜索引擎索引和首次加载速度产生挑战。

SSR 和 SEO:SSR 通过在服务器端将完全呈现的 HTML 发送到客户端,有助于提高页面在搜索引擎中的可索引性。搜索引擎可以更容易地读取和理解页面内容,从而提高 SEO。相比之下,SPA 在初次加载时可能只有一个空的 HTML 骨架,内容是通过 JavaScript 动态加载的,这可能对 SEO 造成挑战。

SSR 和 SPA:SSR 和 SPA 是两种不同的页面渲染方法。SSR 在服务器端完成渲染,可以提供更好的 SEO 和首次加载性能,但也可能增加服务器负担。SPA 在客户端动态加载内容,提供更流畅的用户体验,但可能面临搜索引擎索引和初始加载速度的问题。一些项目结合两者,使用 SSR 渲染首次加载的内容,然后在后续页面切换时转为 SPA。

想了解更多关于ssr,seo,spa的知识可以看下面这篇文章:

浅谈SPA、SEO、SSR - 简书前后端分离算是最近Web开发的大趋势了,目前已经有大量的公司使用了前后端分离的开发方式。那我们就来大概谈谈前后端分离开发中必须要了解和接触的几个概念:SPA、SEO和SSR。...icon-default.png?t=N7T8https://www.jianshu.com/p/fcb98533bc18

二、VUE做SSR的几种方法

1、插件prerender-spa-plugin

prerender-spa-plugin 是一个用于预渲染单页面应用(SPA)的插件,它可以帮助你在构建时生成静态 HTML 文件,以优化搜索引擎索引和首次加载性能。它需要搭配webpack进行使用,并且改插件有局限性,只能用于较少页面的渲染,一旦渲染的较多负载会非常大,且不适用动态平衡路由的渲染。如商品详情页,文章详情页等等。且该项目已经停止更新了,请谨慎使用。

prerender-spa-plugin官网:GitHub - chrisvfritz/prerender-spa-plugin: Prerenders static HTML in a single-page application.Prerenders static HTML in a single-page application. - GitHub - chrisvfritz/prerender-spa-plugin: Prerenders static HTML in a single-page application.icon-default.png?t=N7T8https://github.com/chrisvfritz/prerender-spa-plugin

这里推荐一篇prerender-spa-plugin详细使用文章:

https://www.cnblogs.com/chuaWeb/p/prerender-plugin.htmlicon-default.png?t=N7T8https://www.cnblogs.com/chuaWeb/p/prerender-plugin.html写的非常详细。


2、VUE开启SSR渲染模式

Vue.js 的服务器端渲染(SSR)通过创建一个服务器入口文件来处理客户端请求,使用 createApp 函数创建 Vue 应用实例并配置路由,根据请求的 URL 匹配路由并预取数据,使用 renderToString 方法将应用渲染为完全呈现的 HTML 字符串,然后将渲染后的 HTML 和状态数据作为响应返回给客户端,客户端接管渲染过的 HTML 并激活应用。从而实现更好的搜索引擎优化、更快的首次加载速度以及服务器和客户端之间更一致的行为。但这个方案也被pass掉了,因为项目过于老,且代码很乱。各种冲突版本兼容问题层出不穷因此也放弃了。但这应该是在节省成本的方法中最适合的一个方案了。

Vue.js 的服务器端渲染(SSR)官网:

Vue.js 服务器端渲染指南 | Vue SSR 指南Vue.js 服务端渲染指南icon-default.png?t=N7T8https://v2.ssr.vuejs.org/zh/这里附带一篇我认为写的较好的博主的文章,大家可以借鉴一下:

如何在vue中实现SSR服务端渲染? - 掘金SSR意为服务端渲染,指由服务侧完成页面的 HTML 结构拼接的页面处理技术,发送到浏览器,然后为其绑定状态与事件,成为完全可icon-default.png?t=N7T8https://juejin.cn/post/7063795725296992270

3、使用NUXT框架 

官网:介绍 · 开始使用Nuxt3 Nuxt中文站Nuxt的目标是使web开发直观和性能与伟大的DX铭记在心。icon-default.png?t=N7T8https://nuxt.com.cn/docs/getting-started/introduction

接下里来我将对NUXT的一些基本使用与踩坑展开描述。

三、NUXT3+VUE3+TS

首先这里默认你已经安装了NUXT3,如果不会的话,请移步至官网文档,就一个命令的事。(注意:node版本要大于16.11.0)

(一)基本配置

1、文件夹介绍

assets

在根目录文件下创建一个名为assets的文件夹用于存储静态文件。如图所示,可以将css文件,图片文件,svg文件包括iconfont字体等文件都存储在这里。

⚠️:这是官方约定俗成的东西。

components

一般用于存储封装好的通用组件。

pages

存放页面的文件夹

 

api

存放网络请求

 

plugins

存放各类插件。

utils 

存放各类工具。

2、布局使用

首先新建一个 layouts 文件夹,写入 default.vue 文件。

 找到 app.vue 文件添加 NuxtLayout 标签。

运行效果:

 

3、useFetch请求封装

首先在根目录下新建一个文件夹 service ,并新建一个文件 index.ts

添加代码如下:

import { UseFetchOptions } from "nuxt/app";type Methods = "GET" | "POST" | "DELETE" | "PUT";const BASE_URL = "https://xxx.com/";export interface IResultData<T> {code: number;data: T;msg: string;
}class HttpRequest {request<T = any>(url: string,method: Methods,data: any,options?: UseFetchOptions<T>,) {return new Promise((resolve, reject) => {const newOptions: UseFetchOptions<T> = {baseURL: BASE_URL,method: method,...options,};if (method === "GET" || method === "DELETE") {newOptions.params = data;}if (method === "POST" || method === "PUT") {newOptions.body = data;}useFetch(url, newOptions).then((res) => {resolve(res.data);}).catch((error) => {reject(error);});});}// 封装常用方法get<T = any>(url: string, params?: any, options?: UseFetchOptions<T>) {return this.request(url, "GET", params, options);}post<T = any>(url: string, data: any, options?: UseFetchOptions<T>) {return this.request(url, "POST", data, options);}Put<T = any>(url: string, data: any, options?: UseFetchOptions<T>) {return this.request(url, "PUT", data, options);}Delete<T = any>(url: string, params: any, options?: UseFetchOptions<T>) {return this.request(url, "DELETE", params, options);}
}const httpRequest = new HttpRequest();export default httpRequest;

再在根目录下新建一个 api 文件夹,新建文件index。在这里引入并定义接口,以便于接口的统一处理。

import httpRequest from "~/service";const getAppdetail = (params: any) => {const URL = `/app-server/marketui/${params.marketID}/apps/${params.appID}/detail`return httpRequest.get(URL, params);
};export {getAppdetail,
};

使用如下:

<template>这是主页面的内容
</template><script setup lang="ts">
import { ref } from 'vue';
import { getAppdetail } from '../api/index';
let marketID = ref<string>('859a51f9bb3b48b5bfd222e3bef56425');
let appID = ref<string>('10d2295fc98d4163acc4b2ec9d2917b9');
let getAppdetailData = async () => {let Appdetail: any = await getAppdetail({marketID: marketID.value,appID: appID.value,});console.log(JSON.parse(JSON.stringify(Appdetail.value)));
};
getAppdetailData()
</script>
<style  scoped>
</style>

请求成功: 

        

(二)引入element-pluse

1、安装

安装element-pluse

npm install element-plus --save

官网文档:

Button 按钮 | Element Plusa Vue 3 based component library for designers and developersicon-default.png?t=N7T8https://element-plus.org/zh-CN/component/button.html 

2、引入

在 根目录创建 plugins 文件夹 ,新建一个名为 element-plus.js 的文件,并添加如下文件内容。

import { defineNuxtPlugin } from '#app'
import ElementPlus from 'element-plus'
import { ID_INJECTION_KEY } from 'element-plus'
import zhCn from 'element-plus/es/locale/lang/zh-cn'import 'element-plus/dist/index.css'
export default defineNuxtPlugin(nuxtApp => {nuxtApp.vueApp.use(ElementPlus, {locale: zhCn,})nuxtApp.vueApp.provide(ID_INJECTION_KEY, {prefix: 100,current: 0,})
})

 大功告成就可以使用了。

3、使用

我们更改一下about页面的代码。

运行效果: 

(三)引入md文档

⚠️:此插件使用时需要搭配 <ClientOnly></ClientOnly> 或者 <no-ssr></no-ssr> 使用,否则会报错,因为在服务端获取不到顶级的window对象。 

1、下载

命令

npm install mavon-editor --save

GitHub:

GitHub - hinesboy/mavonEditor: mavonEditor - A markdown editor based on Vue that supports a variety of personalized featuresmavonEditor - A markdown editor based on Vue that supports a variety of personalized features - GitHub - hinesboy/mavonEditor: mavonEditor - A markdown editor based on Vue that supports a variety of personalized featuresicon-default.png?t=N7T8https://github.com/hinesboy/mavonEditor 

2、引入

在 根目录创建 plugins 文件夹 ,新建一个名为 vue-mavon-editor.js 的文件,并添加如下文件内容。

import { defineNuxtPlugin } from '#app'
import mavonEditor from 'mavon-editor'
import "mavon-editor/dist/css/index.css";export default defineNuxtPlugin( nuxtApp => {nuxtApp.vueApp.use(mavonEditor)
})

nuxt.config.ts 中添加如下代码。 

export default defineNuxtConfig({//其他配置plugins: [{ src: '@/plugins/vue-mavon-editor.js', ssr: false, mode: 'client' },],
})

3、使用

用于输入数据

这次我们更改 home 文件夹下的 index.vue

<template><h1>这是home页面</h1><ClientOnly><mavon-editorref="md"placeholder="请输入文档内容...":boxShadow="false"style="z-index:1;border: 1px solid #d9d9d9;height:50vh"v-model="value":toolbars="toolbars"/></ClientOnly></template><script setup lang="ts">
import { ref } from 'vue';
let value = ref<string>('**这是加粗的文字**');
let toolbars = ref<any>({bold: true, // 粗体italic: true, // 斜体header: true, // 标题underline: true, // 下划线strikethrough: true, // 中划线mark: true, // 标记superscript: true, // 上角标subscript: true, // 下角标quote: true, // 引用ol: true, // 有序列表ul: true, // 无序列表link: true, // 链接imagelink: true, // 图片链接code: true, // codetable: true, // 表格fullscreen: true, // 全屏编辑readmodel: true, // 沉浸式阅读htmlcode: true, // 展示html源码help: true, // 帮助/* 1.3.5 */undo: true, // 上一步redo: true, // 下一步trash: true, // 清空save: false, // 保存(触发events中的save事件)/* 1.4.2 */navigation: true, // 导航目录/* 2.1.8 */alignleft: true, // 左对齐aligncenter: true, // 居中alignright: true, // 右对齐/* 2.2.1 */subfield: true, // 单双栏模式preview: true // 预览})
</script>
<style  scoped>
</style>

 效果如图:

用于回显数据

<template><h1>这是home页面</h1><ClientOnly><mavon-editorref="md"v-highlight class="md" v-model="value" //引入要转换的内容:subfield="false"//开启单栏模式       :defaultOpen="'preview'" //开启单栏模式:toolbarsFlag="false" //开启单栏模式:editable="false"//开启单栏模式 :scrollStyle="true"//开启滚动条样式/></ClientOnly></template><script setup lang="ts">
import { ref } from 'vue';let value = ref<string>('**这是加粗的文字**');
</script>
<style  scoped>
</style>

(四)v-show与v-if踩坑

1、区别

  1. 渲染方式的区别

    • v-if:当条件为真时,元素会被渲染到DOM中;当条件为假时,元素会被完全从DOM中移除。
    • v-show:无论条件是真还是假,元素都会被渲染到DOM中,但是通过设置元素的display属性来控制是否显示在页面上。
  2. 初始渲染性能

    • v-if:在初始渲染时,如果条件为假,元素不会被添加到DOM中,因此可以提高初始加载性能。
    • v-show:无论条件如何,元素都会被添加到DOM中,只是通过CSS的display属性来控制显示与隐藏,因此初始加载性能上稍逊于v-if
  3. 切换开销

    • v-if:当条件从假切换到真时,会进行DOM插入操作,从真切换到假时,会进行DOM移除操作,可能涉及更多的DOM操作开销。
    • v-show:无论切换条件如何,只需操作CSS属性,因此切换时的性能开销较小。
  4. 适用场景

    • v-if:适合在条件不经常改变的情况下使用,因为它在切换时有更高的开销。
    • v-show:适合在条件需要经常改变的情况下使用,因为它的切换开销较小。

2、问题

使用v-show可能会导致文件加载错误,或者模块渲染失败。比如在这段代码中:

<div class="tag_box" v-if="parentNode"><div class="tag_icon"><img class="helm_icon" src="../../../assets/svg/tag-type.svg" alt="" /></div><div class="tag_name">{{ parentNode }}</div>
</div>

我的img文件中指向的是静态文件中的一个icon图标。

当我使用v-if时,正常展示。

 当我使用v-show时,展示就出现了问题。

目前不清楚是为何导致的但需要注意这一点。 

(五)el- select与el-tooltip踩坑

1、问题

可以详见这个issues。https://github.com/element-plus/element-plus/issues/9414icon-default.png?t=N7T8https://github.com/element-plus/element-plus/issues/9414

总的来说就是使用 el- select 与 el-tooltip 的地方会导致渲染出现问题,比如这里。

    <!-- 导航栏 --><!-- <no-ssr> --><nav class="navbar"><div class="navbar-box"><div style="font-size: 24px">云原生应用市场</div><div><ul><li><a :href="`/docs`" target="_blank"> 文档 </a></li><li><el-popover :width="160" placement="bottom" trigger="hover"><template #default><div><imgclass="wechart":src="marketInfo &&marketInfo.banners &&marketInfo.banners.length > 0 &&marketInfo.banners[0].imageURL"alt=""/><pstyle="font-size: 12px;margin-bottom: 0px;margin-top: 6px;text-align: center;">扫一扫,邀你进群交流吧</p></div></template><template #reference>加入群聊</template></el-popover></li><li><a :href="`/enterprise/login`"> 登录 </a></li></ul></div></div></nav><!-- </no-ssr> -->

运行效果:

 

可以看到,我这里点击跳转以后地址栏发生了变化,但是页面并没有跳过去,而且控制台也报错。Uncaught (in promise) TypeError: Cannot read properties of null (reading 'parentNode') 

 解决方法

使用 <no-ssr></no-ssr> 把对应的 el- select与el-tooltip 包裹住,让这一小段代码不使用ssr渲染,从上面的issues也可以看到这是element plus的一个bug,但是截止到目前仍未修复。

总结:坑还是挺多的,而且很多问题虽然解决了但不知道是为什么(有知道为什么的大佬如果您不嫌弃的话麻烦给我讲解一下),因为这个玩意已经加了两周多的班了,终于捋出来一些头绪了。终于也是快接近尾声了。seo优化这个词也终于从我的面试八股文中走了出来真真切切的体验了一把,虽然过程很艰难,但好在最难的时刻已经过去啦。加油加油各位!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/90348.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

2023年数字孪生行业研究报告

第一章 行业概况 1.1 定义 数字孪生&#xff08;Digital Twin&#xff09;是一种先进的建模技术&#xff0c;它通过创建一个物理实体的虚拟复制品&#xff0c;以实时模拟、预测和优化实体的行为和性能。这个虚拟模型会同步收集和分析来自其物理对应物的数据&#xff0c;从而提…

Matlab图像处理-图像缩放

基本概念 图像缩放是指将给定的图像在x轴方向按比例缩放a倍&#xff0c;在y轴方向按比例缩放b倍&#xff0c;从而获得一幅新的图像。 如果ab&#xff0c;即在x轴方向和y轴方向缩放的比率相同&#xff0c;则称这样的比例缩放为图像的全比例缩放。 如果a≠b&#xff0c;图像比…

全画面塑料激光透光率测试仪

随着光学塑料成型技术的不断发展&#xff0c;光学塑料透镜在各类光学系统中得到了广泛的应用。光学塑料具有便于大批量生产&#xff0c;设计灵活性高&#xff0c;重量轻和耐冲击等特点。最近几年来&#xff0c;国内外有许多专家学者对热塑性半结晶塑料激光焊接工艺上的应用进行…

【微服务部署】02-配置管理

文章目录 1.ConfigMap1.1 创建ConfigMap方式1.2 使用ConfigMap的方式1.3 ConfigMap使用要点建议 2 分布式配置中心解决方案2.1 什么时候选择配置中心2.2 Apollo配置中心系统的能力2.2.1 Apollo创建配置项目2.2.2 项目使用2.2.3 K8s中使用Apollo 1.ConfigMap ConfigMap是K8s提供…

零撸大肉,赛博尔Seppol游戏,无限制闯关打碎片,装备,直接变现项目。

2023年7月10日&#xff0c;在上海外滩酒店—— 由来自硅谷、华尔街的技术先锋&#xff0c;与中国科技翘楚阿里、腾讯的骨干团队联手呈现&#xff0c;区块链元宇宙游戏塞波尔 Seppol于上海精彩亮相路演。 1&#xff0c;栖息之地&#xff0c;宠物可放入栖息之地进行挖矿&#xf…

自动驾驶和辅助驾驶系统的概念性架构(二)

摘要&#xff1a; 本篇为第二部分主要介绍底层计算单元、示例工作负载 前言 本文档参考自动驾驶计算联盟(Autonomous Vehicle Computing Consortium)关于自动驾驶和辅助驾驶计算系统的概念系统架构。该架构旨在与SAE L1-L5级别的自动驾驶保持一致。本文主要介绍包括功能模块图…

CSA研讨会|聚焦云原生安全,探讨技术与应用策略

为产业数字化保驾护航&#xff0c; 云原生安全体系如何有效抵御网络威胁&#xff1f; 网络安全的下一个十年&#xff0c; 云原生安全是网络安全创新之路吗&#xff1f; CNAPP部署现状&#xff0c;你了解多少&#xff1f; 9月6日&#xff08;周三&#xff09;下午14&#xff1a…

QTday1(实现图形化界面、QT工程项目各文件初始程序的介绍)

1.实现图形化界面 #include "widget.h" #include "ui_widget.h" #include <QDebug> //输出函数对应的头文件 #include <QIcon>Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget) {ui->setupUi(…

VictoriaLogs:一款超低占用的 ElasticSearch 替代方案

image.png 背景 前段时间我们想实现 Pulsar 消息的追踪流程&#xff0c;追踪实现的效果图如下&#xff1a; 实现其实比较简单&#xff0c;其中最重要的就是如何存储消息。 消息的读取我们是通过 Pulsar 自带的 BrokerInterceptor 实现的&#xff0c;对这个感兴趣的朋友后面会单…

指针与空间按钮的交互

文章目录 原理案例&#xff1a;“直线指针”和“点击按钮”的交互1、效果2、步骤 原理 指针不能直接和空间按钮交互&#xff0c;得借助一个中间层——分发器——它分发指针的进入、退出、选择事件&#xff0c;空间按钮自动监听这些事件 案例&#xff1a;“直线指针”和“点击…

【java中的Set集合】HashSet、LinkedHashSet、TreeSet(最通俗易懂版!!)

目录 一、HashSet集合 1.HashSet集合的特点 2.HashSet常用方法 二、LinkedHashSet集合 LinkedHashSet集合的特点 三、TreeSet集合 1.TreeSet集合的特点 2.TreeSet的基本使用 四、HashSet、LinkedHashSet、TreeSet的使用场景 五、list和set集合的区别 一、HashSet集合 …

12、监测数据采集物联网应用开发步骤(9.1)

监测数据采集物联网应用开发步骤(8.2) TCP/IP Server开发 在com.zxy.common.Com_Para.py中添加如下内容 #锁机制 lock threading.Lock() #本机服务端端口已被连接客户端socket list dServThreadList {} #作为服务端接收数据拦截器 ServerREFLECT_IN_CLASS "com.plug…