大家好,我是辉夜真是太可爱啦。本文主要记录了自己的博客在
禁用缓存
的情况下,从八九秒加载时间到最终985ms
的优化实践,开启缓存的情况下能达到138ms
的访问速度,预览地址为:www.cooldream.fun
本人的个人博客采用Vue-cli4.1.2 + typescript
构建
项目目录结构如下
├─ src //主文件
│ ├─ api //接口文件夹
| | |- config.js //后端接口地址的配置,将测试、开发、生产环境分开
| | └─ user.js //接口文件,配置了token请求头,具体接口根据需求修改
│ ├─ assets //资源文件
│ ├─ components //公用组件
│ ├─ directive //vue自定义指令
| ├─ filters //存放过滤器文件,自带了手机号加密,手机号格式化,时间日期处理
| ├─ interceptors //存放axios拦截器配置,写入了接口调用的loading加载以及http状态码报错拦截
| ├─ interceptors //放置公用的接口,对数据进行类型限制
| ├─ layout //布局文件,通过子路由渲染方式实现,具体HTML布局根据需求修改
| ├─ mixins //混入文件,配置了一个平滑滚动的方法
| ├─ plugins //外部插件文件夹,配置了按需引入的element-ui
| ├─ reg //存放正则以及校验的文件夹
| | |- reg.ts //存放正则表达式,自带了传真,邮箱,qq,手机号,银行卡号,固定电话,密码强度校验正则
| | └─ validator.ts //存放element-ui自定义校验,自带了传真,邮箱,qq,手机号,银行卡号,固定电话,密码强度自定义校验
| ├─ router //路由文件
| ├─ store //vuex全局变量文件
| | |- index.ts //store主文件
| | └─ module //store模块文件夹
| | | └─ user.ts //存放user相关的全局变量
| ├─ stylus //css预处理器文件夹
| | |- reset.styl //样式初始化文件,自带了非标准盒子,a标签清除下划线,清除内外边距,禁止图片拖拽等效果
| | └─ color.styl //颜色变量文件
| ├─ utils //公用方法文件夹
| | |- area.ts //存放省市区三级地区的数据
| | |- array.ts //存放数组相关的公用方法,自带了两个元素交换位置,元素前进后退一格,元素置顶或末尾,去重,删除指定元素操作
| | └─ object.ts //存放对象相关的公用方法,自带了对象清空所有值的方法
| ├─ views //页面文件夹
| ├─ main.ts //主配置文件
| ├─ babel.config.js //babel配置文件,写入了element-ui按需加载的配置
| ├─ package.json //npm的包管理文件
| ├─ tsconfig.json //ts配置文件
| ├─ vue.config.js //vue配置文件
1.关闭productionSourceMap
首先,由于最新版的脚手架不自带配置文件
了,先在根目录新建vue.config.js
文件,关闭productionSourceMap
,在vue.config.js
中写入如下内容
module.exports = {
productionSourceMap: false
}
2.开启Gzip压缩
安装插件compression-webpack-plugin
,打开代码压缩,npm install --save-dev compression-webpack-plugin
,现在的vue.config.js
代码如下,但是,需要注意的是,服务器上nginx也必须开启gzip才能生效
// 是否为生产环境
const isProduction = process.env.NODE_ENV !== 'development';
// gzip压缩
const CompressionWebpackPlugin = require('compression-webpack-plugin')
module.exports = {
productionSourceMap: false,
configureWebpack: config => {
// 生产环境相关配置
if (isProduction) {
//gzip压缩
const productionGzipExtensions = ['html', 'js', 'css']
config.plugins.push(
new CompressionWebpackPlugin({
filename: '[path].gz[query]',
algorithm: 'gzip',
test: new RegExp(
'\\.(' + productionGzipExtensions.join('|') + ')$'
),
threshold: 10240, // 只有大小大于该值的资源会被处理 10240
minRatio: 0.8, // 只有压缩率小于这个值的资源才会被处理
deleteOriginalAssets: false // 删除原文件
})
)
}
}
}
打开nginx
文件夹下的nginx.conf
文件,在http
模块中写入以下内容
# 开启gzip
gzip on;
# 启用gzip压缩的最小文件,小于设置值的文件将不会压缩
gzip_min_length 1k;
# gzip 压缩级别,1-9,数字越大压缩的越好,也越占用CPU时间,后面会有详细说明
gzip_comp_level 2;
# 进行压缩的文件类型。javascript有多种形式,后面的图片压缩不需要的可以自行删除
gzip_types text/plain application/javascript application/x-javascript text/css application/xml text/javascript application/x-httpd-php image/jpeg image/gif image/png;
# 是否在http header中添加Vary: Accept-Encoding,建议开启
gzip_vary on;
# 设置压缩所需要的缓冲区大小
gzip_buffers 4 16k;
然后输入命令nginx -s reload
重启nginx服务
如果后端nginx开启了gzip,可以从network
中的Content-Encoding
中看到gzip
3.开启CDN加速(建议选配,CDN虽然速度快,但没有本地打包稳定)
将使用的插件使用cdn链接,并且配置webpack
将使用CDN
的插件不进行打包,别忘记还要再index.html
中引入js
以及css
// 是否为生产环境
const isProduction = process.env.NODE_ENV !== 'development';
// 本地环境是否需要使用cdn
const devNeedCdn = false
// cdn链接
const cdn = {
// cdn:模块名称和模块作用域命名(对应window里面挂载的变量名称)
externals: {
vue: 'Vue',
vuex: 'Vuex',
'vue-router': 'VueRouter',
'marked': 'marked',
'highlight.js': 'hljs',
'nprogress': 'NProgress',
'axios': 'axios'
},
// cdn的css链接
css: [
'https://cdn.bootcss.com/nprogress/0.2.0/nprogress.min.css'
],
// cdn的js链接
js: [
'https://cdn.bootcss.com/vue/2.6.10/vue.min.js',
'https://cdn.bootcss.com/vuex/3.1.2/vuex.min.js',
'https://cdn.bootcss.com/vue-router/3.1.3/vue-router.min.js',
'https://cdn.bootcss.com/marked/0.8.0/marked.min.js',
'https://cdn.bootcss.com/highlight.js/9.18.1/highlight.min.js',
'https://cdn.bootcss.com/nprogress/0.2.0/nprogress.min.js',
'https://cdn.bootcss.com/axios/0.19.2/axios.min.js'
]
}
module.exports = {
chainWebpack: config => {
// ============注入cdn start============
config.plugin('html').tap(args => {
// 生产环境或本地需要cdn时,才注入cdn
if (isProduction || devNeedCdn) args[0].cdn = cdn
return args
})
// ============注入cdn start============
},
configureWebpack: config => {
// 用cdn方式引入,则构建时要忽略相关资源
if (isProduction || devNeedCdn) config.externals = cdn.externals
}
}
接下来,在index.html
中引入使用了CDN
的链接
<!DOCTYPE html>
<html lang="en" style="width: 100%;height: 100%;">
<head>
<meta charset="utf-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width,initial-scale=1.0">
<link rel="icon" href="<%= BASE_URL %>favicon.ico">
<!-- 使用CDN的CSS文件 -->
<% for (var i in htmlWebpackPlugin.options.cdn &&
htmlWebpackPlugin.options.cdn.css) { %>
<link href="<%= htmlWebpackPlugin.options.cdn.css[i] %>" rel="stylesheet" />
<% } %>
<!-- 使用CDN的CSS文件 -->
<title>CoolDream</title>
</head>
<body style="width: 100%;height: 100%;">
<noscript>
<strong>We're sorry but blog doesn't work properly without JavaScript enabled. Please enable it to continue.</strong>
</noscript>
<div id="app"></div>
<!-- built files will be auto injected -->
<!-- 使用CDN的JS文件 -->
<% for (var i in htmlWebpackPlugin.options.cdn &&
htmlWebpackPlugin.options.cdn.js) { %>
<script src="<%= htmlWebpackPlugin.options.cdn.js[i] %>"></script>
<% } %>
<!-- 使用CDN的JS文件 -->
</body>
</html>
打包后抛到服务器上,打开开发者工具的network,如果看到http
请求cdn
,那么就代表配置成功了,如图所示
4.代码压缩
先安装插件npm i -D uglifyjs-webpack-plugin
然后在最上方引入依赖
// 代码压缩
const UglifyJsPlugin = require('uglifyjs-webpack-plugin')
在configureWebpack
模块中引入代码压缩
// 代码压缩
config.plugins.push(
new UglifyJsPlugin({
uglifyOptions: {
//生产环境自动删除console
compress: {
drop_debugger: true,
drop_console: true,
pure_funcs: ['console.log']
}
},
sourceMap: false,
parallel: true
})
)
5.公共代码抽离,写在configureWebpack
模块中
// 公共代码抽离
config.optimization = {
splitChunks: {
cacheGroups: {
vendor: {
chunks: 'all',
test: /node_modules/,
name: 'vendor',
minChunks: 1,
maxInitialRequests: 5,
minSize: 0,
priority: 100
},
common: {
chunks: 'all',
test: /[\\/]src[\\/]js[\\/]/,
name: 'common',
minChunks: 2,
maxInitialRequests: 5,
minSize: 0,
priority: 60
},
styles: {
name: 'styles',
test: /\.(sa|sc|c)ss$/,
chunks: 'all',
enforce: true
},
runtimeChunk: {
name: 'manifest'
}
}
}
}
6.图片压缩()
1.使用图片压缩插件
-
先安装插件
npm install image-webpack-loader --save-dev
-
在
chainWebpack
中新增以下代码
// ============压缩图片 start============
config.module
.rule('images')
.use('image-webpack-loader')
.loader('image-webpack-loader')
.options({ bypassOnDebug: true })
.end()
// ============压缩图片 end============
2.将静态资源存储在云端,个人用的七牛云,对象存储用于存储文件,使用cdn
加速让存储的静态资源访问速度更快。(推荐,速度能快挺多)
-
我个人申请了七牛云,实名认证就有10G空间可用,每个月有10G的免费流量,不过不绑定域名的话只有30天体验机会,我是绑定了我的域名进行DNS解析中转,具体的操作可以参考这一篇博客,www.cnblogs.com/mazhichu/p/…
7.首屏骨架屏优化(选配,看自己实际需要吧)
1.安装插件
npm install vue-skeleton-webpack-plugin --save
2.新建骨架屏文件
在src
下新建Skeleton
文件夹,其中新建index.js
以及index.vue
,在其中写入以下内容,其中,骨架屏的index.vue
页面样式请自行编辑
//index.js
import Vue from 'vue'
// 创建的骨架屏 Vue 实例
import skeleton from './index.vue';
export default new Vue({
components: {
skeleton
},
template: '<skeleton />'
});
//index.vue
<template>
<div class="skeleton-box">
loading
</div>
</template>
<script lang="ts"> import {Vue,Component} from "vue-property-decorator"; @Component({ name:'Skeleton' }) export default class Skeleton extends Vue{ } </script>
<style lang="stylus" scoped> .skeleton-box{ font-size 24px display flex align-items center justify-content center width 100vh height 100vh } </style>
3.在vue.config.js
中配置骨架屏
在vue.config.js
中写入以下内容
//骨架屏渲染
const SkeletonWebpackPlugin = require('vue-skeleton-webpack-plugin')
//path引入
const path = require('path')
//configureWebpack模块中写入内容
// 骨架屏渲染
config.plugins.push(
new SkeletonWebpackPlugin({
webpackConfig: {
entry: {
app: path.join(__dirname,'./src/components/Skeleton/index.js')
}
}
})
)
8.关闭预先加载模块
vue-cli
默认开启 prefetch
(预先加载模块),提前获取用户未来可能会访问的内容
如果你只想首屏尽快地渲染,可以先不管后面页面的快慢,那么可以将这个模块删除
config.plugins.delete('prefetch')
所以,最终的配置文件如下(关闭预先加载模块已注释,如需要可以解开封印)
// 是否为生产环境
const isProduction = process.env.NODE_ENV !== 'development';
// 代码压缩
const UglifyJsPlugin = require('uglifyjs-webpack-plugin')
// gzip压缩
const CompressionWebpackPlugin = require('compression-webpack-plugin')
//骨架屏渲染
const SkeletonWebpackPlugin = require('vue-skeleton-webpack-plugin')
//path引入
const path = require('path')
// 本地环境是否需要使用cdn
const devNeedCdn = false
// cdn链接
const cdn = {
// cdn:模块名称和模块作用域命名(对应window里面挂载的变量名称)
externals: {
vue: 'Vue',
vuex: 'Vuex',
'vue-router': 'VueRouter',
'marked': 'marked',
'highlight.js': 'hljs',
'nprogress': 'NProgress',
'axios': 'axios'
},
// cdn的css链接
css: [
'https://cdn.bootcss.com/nprogress/0.2.0/nprogress.min.css'
],
// cdn的js链接
js: [
'https://cdn.bootcss.com/vue/2.6.10/vue.min.js',
'https://cdn.bootcss.com/vuex/3.1.2/vuex.min.js',
'https://cdn.bootcss.com/vue-router/3.1.3/vue-router.min.js',
'https://cdn.bootcss.com/marked/0.8.0/marked.min.js',
'https://cdn.bootcss.com/highlight.js/9.18.1/highlight.min.js',
'https://cdn.bootcss.com/nprogress/0.2.0/nprogress.min.js',
'https://cdn.bootcss.com/axios/0.19.2/axios.min.js'
]
}
module.exports = {
productionSourceMap: false,
chainWebpack: config => {
// ============预先加载模块 关闭============
// config.plugins.delete('prefetch')
// ============预先加载模块 关闭============
// ============注入cdn start============
config.plugin('html').tap(args => {
// 生产环境或本地需要cdn时,才注入cdn
if (isProduction || devNeedCdn) args[0].cdn = cdn
return args
})
// ============注入cdn start============
// ============压缩图片 start============
config.module
.rule('images')
.use('image-webpack-loader')
.loader('image-webpack-loader')
.options({ bypassOnDebug: true })
.end()
// ============压缩图片 end============
},
configureWebpack: config => {
// 用cdn方式引入,则构建时要忽略相关资源
if (isProduction || devNeedCdn) config.externals = cdn.externals
// 生产环境相关配置
if (isProduction) {
//gzip压缩
const productionGzipExtensions = ['html', 'js', 'css']
config.plugins.push(
new CompressionWebpackPlugin({
filename: '[path].gz[query]',
algorithm: 'gzip',
test: new RegExp(
'\\.(' + productionGzipExtensions.join('|') + ')$'
),
threshold: 10240, // 只有大小大于该值的资源会被处理 10240
minRatio: 0.8, // 只有压缩率小于这个值的资源才会被处理
deleteOriginalAssets: false // 删除原文件
})
)
// 代码压缩
config.plugins.push(
new UglifyJsPlugin({
uglifyOptions: {
//生产环境自动删除console
compress: {
drop_debugger: true,
drop_console: true,
pure_funcs: ['console.log']
}
},
sourceMap: false,
parallel: true
})
)
}
// 骨架屏渲染
config.plugins.push(
new SkeletonWebpackPlugin({
webpackConfig: {
entry: {
app: path.join(__dirname,'./src/components/Skeleton/index.js')
}
}
})
)
// 公共代码抽离
config.optimization = {
splitChunks: {
cacheGroups: {
vendor: {
chunks: 'all',
test: /node_modules/,
name: 'vendor',
minChunks: 1,
maxInitialRequests: 5,
minSize: 0,
priority: 100
},
common: {
chunks: 'all',
test: /[\\/]src[\\/]js[\\/]/,
name: 'common',
minChunks: 2,
maxInitialRequests: 5,
minSize: 0,
priority: 60
},
styles: {
name: 'styles',
test: /\.(sa|sc|c)ss$/,
chunks: 'all',
enforce: true
},
runtimeChunk: {
name: 'manifest'
}
}
}
}
}
}
8.nginx配置缓存
同样也可以提高网站的访问速度(虽然这点有点偏离前端打包主题了,但是对于自己独立开发个人博客网站的前端来说还是非常有用的!)在nginx.conf的http模块中写入一下内容
# 设置缓存路径并且使用一块最大100M的共享内存,用于硬盘上的文件索引,包括文件名和请求次数,每个文件在1天内若不活跃(无请求)则从硬盘上淘汰,硬盘缓存最大10G,满了则根据LRU算法自动清除缓存。
proxy_cache_path /var/cache/nginx/cache levels=1:2 keys_zone=imgcache:100m inactive=1d max_size=10g;
然后在nginx.conf的serve模块中写入一下内容,保存配置,nginx -s reload
重启服务即可看到效果
location ~* ^.+\.(css|js|ico|gif|jpg|jpeg|png)$ {
log_not_found off;
# 关闭日志
access_log off;
# 缓存时间7天
expires 7d;
# 源服务器
proxy_pass http://localhost:8888;
# 指定上面设置的缓存区域
proxy_cache imgcache;
# 缓存过期管理
proxy_cache_valid 200 302 1d;
proxy_cache_valid 404 10m;
proxy_cache_valid any 1h;
proxy_cache_use_stale error timeout invalid_header updating http_500 http_502 http_503 http_504;
}
本文主要参考于www.jianshu.com/p/476387c7f…
另外,再次分享一个我构建的基于Vue2.6.10+Typescript
构建的空白项目,包括css样式的初始化,以及基本常用的axios,vue-router,模块化使用vuex,element-ui已经按需引入配置好,还有axios拦截器,axios请求的全局loaindg加载,路由组件懒加载,以及对于不同环境的基本Url封装,还附带了一些常用的方法,以及包括打包优化的cdn引入,代码压缩,图片压缩,关闭map等打包优化都已配置完成,关于ts的使用,要使用修饰符,在Home.vue中,常用的使用方法我也都已经列举出来了,地址为:github.com/Jack-Star-T…
今天的文章Vue2.6.10(Vue-cli4)项目打包性能优化实践分享到此就结束了,感谢您的阅读。
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。
如需转载请保留出处:https://bianchenghao.cn/22461.html