SingleFile 是一个浏览器插件兼容 Chrome、Firefox(桌面端和移动端)、Microsoft Edge、Vivaldi、Brave、Waterfox、Yandex 和 Opera 浏览器。它可以帮助你将一个完整的网页保存为一个单一的 HTML 文件。另一个版本SingleFileZ 是一款可以把一个网页包括图片、样式完整打包压缩后保存为 HTML 的浏览器扩展,并且能够让浏览器实现自解压。SingleFileZ很有意思,SingleFileZ 与 SingleFile 功能完全相同,只不过增加了压缩功能,使用 SingleFile 下载后的单一 HTML 文件为 627KB,而使用 SingleFileZ 下载后的单一 HTML 文件为 265 KB。而更有意思的是,可以直接使用压缩工具打开由 SingleFileZ 生成的 HTML 文件,不过要打开这个压缩 HTML 文件,需要 Chrome 启动时添加 –allow-file-access-from-files 参数。
SingleFile在主流的浏览器插件商店可用:
也可以下载插件的 zip 包 – https://github.com/gildas-lormeau/SingleFile/archive/master.zip,拖曳到浏览器插件管理界面进行安装。
SingleFile的使用非常简单,等待网页完全加载完毕,点击插件工具栏上的 SingleFile 按钮即可保存页面,下载的 HTML 文件保存在浏览器设置的本地文件夹。在处理一个页面时,你可以再次点击该按钮来取消该动作。
录了自己的博客在禁用缓存的情况下,从八九秒加载时间到最终985ms的优化实践,开启缓存的情况下能达到138ms的访问速度
├─ 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配置文件
module.exports = {
productionSourceMap: false
}
// 是否为生产环境
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 // 删除原文件
})
)
}
}
}
# 开启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;
image.png
// 是否为生产环境
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'
},
// 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'
]
}
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
}
}
<!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>
image.png
// 代码压缩
const UglifyJsPlugin = require('uglifyjs-webpack-plugin')
// 代码压缩
config.plugins.push(
new UglifyJsPlugin({
uglifyOptions: {
//生产环境自动删除console
compress: {
drop_debugger: true,
drop_console: true,
pure_funcs: ['console.log']
}
},
sourceMap: false,
parallel: true
})
)
// 公共代码抽离
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'
}
}
}
}
// ============压缩图片 start============
config.module
.rule('images')
.use('image-webpack-loader')
.loader('image-webpack-loader')
.options({ bypassOnDebug: true })
.end()
// ============压缩图片 end============
// 是否为生产环境
const isProduction = process.env.NODE_ENV !== 'development';
// 代码压缩
const UglifyJsPlugin = require('uglifyjs-webpack-plugin')
// gzip压缩
const CompressionWebpackPlugin = require('compression-webpack-plugin')
// 本地环境是否需要使用cdn
const devNeedCdn = false
// cdn链接
const cdn = {
// cdn:模块名称和模块作用域命名(对应window里面挂载的变量名称)
externals: {
vue: 'Vue',
vuex: 'Vuex',
'vue-router': 'VueRouter',
'marked': 'marked',
'highlight.js': 'hljs',
'nprogress': 'NProgress'
},
// 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'
]
}
module.exports = {
productionSourceMap: false,
chainWebpack: config => {
// ============注入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.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'
}
}
}
}
}
}
# 设置缓存路径并且使用一块最大100M的共享内存,用于硬盘上的文件索引,包括文件名和请求次数,每个文件在1天内若不活跃(无请求)则从硬盘上淘汰,硬盘缓存最大10G,满了则根据LRU算法自动清除缓存。
proxy_cache_path /var/cache/nginx/cache levels=1:2 keys_zone=imgcache:100m inactive=1d max_size=10g;
一节中,我们已经安装好了 webpack ,本节我们来学习 何使用 webpack 进行文件打包。
要打包文件,首先我们需要初始化一个项目,前面我们已经创建好了一个 xkd_webpack 的项目,并且这个项目根目录下已经有了一个 package.json 文件、package-lock.json 文件和一个 node_modules 文件夹,如下所示:
在实际项目中,为了方便管理许多文件,我们可能需要创建一个 src 文件夹存放入口文件等开发文件,然后创建一个 dist 文件夹存放最终打包的文件,还有其他类型的文件,为了方便管理我们也会放在同一个目录下。
但是因为我们这里只是举例说一下如何使用 webpack 打包文件,所以我们直接项目根目录下,创建一个静态页面 index.html 和一个 JS 的入口文件 index.js 文件,文件名称是我们自定义的,如果你想使用其他的名称也是可以的。
下面是 index.html 文件的内容:
<html>
<head>
<meta charset="utf-8">
<title>webpack入门</title>
</head>
<body>
<script type="text/javascript" src="bundle.js"></script>
</body>
</html>
在这个 HTML 文件中我们引入了一个 bundle.js 文件,这个 bundle.js 文件就是最终的打包后的文件,但是现在还没有生成哟,要等我们执行完打包命令后才会生成这个文件。
然后入口文件 index.js 的内容如下所示:
document.write('你好,侠课岛!');
最后执行下列命令,就可以成功将 index.js 文件打包到 bundle.js文件中:
webpack index.js -o bundle.js
执行命令效果如下所示:
命令执行成功后,项目根目录下会生成一个 bundle.js 文件。这个文件的作用就是用了一个立即执行函数,然后将 index.js 的内容封装成一个函数,作为参数传进内部执行,这样就完成了文件的打包:
此时我们在浏览器中打开 index.html 文件,页面将会显示 "你好,侠课岛!",这同时也能证明 index.js 文件成功打包到了 bundle.js 文件中,因为我们只在 index.html 文件中引入了 bundle.js 文件。
当然我们在项目中肯定不只有一个 .js 文件,那么如果我们除了 index.js 文件还有其他的 .js 文件,要如何做呢。
示例:
例如项目中还有一个 module.js 文件,内容如下所示:
module.exports = '侠课岛欢迎你!'
这样我们可以修改入口文件 index.js,将创建好的 module.js 模块引入到入口文件中:
document.write('你好,侠课岛!')
document.write(require('./module.js')) // 引入模块
再次执行 webpack index.js -o bundle.js ,会重新打包文件。
在页面启动时,会先执行 index.js 文件中的代码,其它文件中的代码会在执行到 require 语句的时候再执行。 此时刷新浏览器,我们可以看到浏览器中的显示内容发现了改变,显示内容变为 "你好,侠课岛!侠课岛欢迎你!"。
webpack 会分析入口文件,解析包含依赖关系的各个文件,这些文件都打包到 bundle.js 中。webpack 会给每个模块分配一个唯一的 id 并通过这个 id 索引和访问模块。
在旧版本的 webpack 中,我们是使用 webpack index.js bundle.js 命令来执行打包操作,而新的 webpack4.0+ 版本打包文件时要在命令中加一个 -o ,否则会报错。
链接:https://www.9xkd.com/
*请认真填写需求信息,我们会在24小时内与您取得联系。