uni-app填坑指南——解决处理处理静态资源的问题
说实在话,这个标题其实有点夸大了。 uni-app
并没有不解决,而是解决的不够充分不够彻底。这里我们来讨论一下uni-app
在处理静态资源的问题上的一些不足之处。
1. 问题描述
在uni-app
中,我们可以将静态资源放在static
目录下,然后通过相对路径的方式引用。比如我们有一个图片资源logo.png
,我们可以通过<img src="@/static/logo.png" />
的方式引用。
目前引用包括三个方式:
-
@/static/
:引用static
目录下的资源; -
通过
~@/static/
引用static
目录下的资源。 -
如果需要引入其他目录的静态资源,那就只能通过变量的形式,如:
<template>
<view>
<image src="/static/logo.png"></image>
<image :src="imgUrl"></image>
</view>
</template>
<script>
import imgUrl from './a.png'
export default {
data() {
return {
imgUrl
}
}
}
</script>
我们接下来看,它最终的编译产物是什么呢?
编译之后的形成的文件夹结构如下:
mp-weixin
|── assets
│ ├── a.xxxxxxx.png
├── static
│ ├── logo.png
可以看到,static
目录下的资源是直接拷贝到了mp-weixin
目录下,而其他目录下的资源则是被编译成了assets
目录下的文件。
最终生成的WXML文件中,引用的路径是:
<image src="/static/logo.png"></image>
<image src="{{imgUrl}}"></image>
JS文件中引用的路径是:
const imgUrl = __webpack_require__('./a.png')
同时在runtime.js中,有./a.png
路径的定义
'./a.png': function(module, exports, __webpack_require__) {
module.exports = '/assets/a.xxxxxxx.png';
}
以上代码均为伪代码,实际情况可能有所不同。
看起来,对于static
目录以及其他目录的静态资源都处理了, 最终编译产物也是符合预期的。但是,这里有一个问题,static
和assets
目录的资源都位于mp-weixin
目录下,这将导致小程序整体包大小的增大。
熟悉小程序开发的同学都知道,小程序的包大小是有限制的,根据官方介绍,目前限制如下:
- 整个小程序所有分包大小不超过 30M(服务商代开发的小程序不超过 20M)
- 单个分包/主包大小不能超过 2M
很明显,我们不能过多的将资源放在编译后的mp-weixin
目录下。因此如何对该部分资源进行优化变成了一个问题。
2. 解决方案
描述到上述问题之后,我们如何解决这个问题呢?我们有必要描述一下我们的目标:
-
开发环境,能够将
assets
目录移出mp-weixin
文件夹,并提供文件服务器,以便于开发时能够访问到资源; -
生产环境,能够将
assets
目录移出mp-weixin
文件夹,删除assets
文件,并将资源放在CDN上,以便于小程序能够访问到资源。
那么具体怎么做呢 ?
仔细研究webpack
的构建过程,我们发现webpack
的complier的hooks
包含了几个特殊的阶段:
-
emit:资源输出阶段,这个阶段是在资源输出之前的一个阶段,我们可以在这个阶段对资源进行处理。
-
afterEmit:资源输出后阶段,这个阶段是在资源输出之后的一个阶段,我们可以在这个阶段对资源进行处理。
-
done:构建完成阶段,这个阶段是在构建完成之后的一个阶段,我们可以在这个阶段对资源进行处理。
不涉及的hooks我们就不在赘述了,有兴趣的同学可以查看webpack官方文档
从上面的描述我们可以看出,我们可以在afterEmit
和done
阶段对资源进行处理,将assets
目录移出mp-weixin
文件夹,同时开启一个本地的服务器,以便于开发时能够访问到资源。
代码解释
我们新建一个build
目录,然后在build
目录下新建一个webpack-plugin-file.js
文件,用于定义一个webpack plugin:
module.exports = class WebpackPluginFile {
constructor(options) {
this.options = options;
}
apply(compiler) {
this.logger = compiler.getInfrastructureLogger('WebpackPluginFile');
compiler.hooks.done.tap('WebpackPluginFile', () => {
this.copyAssets(compiler).then(() => {
this.startServer(compiler);
});
});
}
}
接下来我们要实现copyAssets
。
const fs = require('fs');
const fsPromise = require('fs/promises');
const path = require('path');
module.exports = class WebpackPluginFile {
async copyAssets(compiler) {
const outputPath = compiler.options.output.path;
const assetsPath = path.resolve(outputPath, 'assets');
const targetPath = path.resolve(outputPath, '..', 'assets');
return this.copyDir(assetsPath, targetPath);
}
async copyDir(src, dist) {
const stats = await fsPromise.stat(src).catch(() => null);
if (!stats) {
// 判断是否存在,如果不存在则返回
return null;
}
if (stats.isFile()) {
// 如果是文件,则直接拷贝
return this.copyFile(src, dist);
}
// 如果是目录,则遍历目录
if (stats.isDirectory()) {
const paths = await fsPromise.readdir(src);
for (let i = 0; i < paths.length; i++) {
const path = paths[i];
await this.copyDir(
path.resolve(src, path),
path.resolve(dist, path)
);
}
}
}
async copyFile(src, dist) {
// 确保目标目录存在
await this.guaranteeDir(path.dirname(dist));
// 拷贝文件
await fsPromise.copyFile(src, dist);
}
}
看起来完美解决了问题。但是,这种拷贝是全量拷贝,如果资源很多,那么拷贝的时间将会很长。因此我们可以在copyDir
方法中加入一些判断,只拷贝有变化的文件。
module.exports = class WebpackPluginFile {
cache = {};
isWatch = false;
async copyDir(src, dist) {
const stats = await fsPromise.stat(src).catch(() => null);
if (!stats) {
// 判断是否存在,如果不存在则返回
return null;
}
if (stats.isFile()) {
// 如果是开发环境,则记录文件的复制属性
if (isWatch) {
const cache = this.cache[src];
if (cache) {
if (cache.mtime >= stats.mtime && cache.size === stats.size) {
return;
}
this.cache[src] = {
mtime: stats.mtime,
size: stats.size
}
this.copyFile(src, dist);
}
} else {
this.copyFile(src, dist);
}
// 省略代码
}
}
apply(compiler) {
const { watch } = compiler.options;
this.isWatch = !!watch;
}
}
对于 watch 模式下,复制完成后更新cache缓存,包含文件的修改时间和文件大小,下次复制时,如果文件的修改时间和文件大小没有变化,则不再复制。这样能提高复制的效率。
但是,对于生产环境,我们并不需要进行缓存,对应watch = false。即直接进行拷贝。 同时,我们需要在done
阶段启动一个本地服务器,以便于开发时能够访问到资源。
module.exports = class WebpackPluginFile {
port = 8888;
app = null;
constructor(options) {
this.options = options;
if (options && options.port) {
this.port = options.port;
}
}
async startServer(compiler) {
if (this.isWatch) {
return;
}
const outputPath = compiler.options.output.path;
const targetPath = path.resolve(outputPath, '..', 'assets');
const express = require('express');
const app = express();
this.server = app;
app.use('/assets', express.static(targetPath));
app.listen(this.port, () => {
console.log(`Server is running at http://localhost:${this.port}`);
});
}
}
按照上述代码,首先初始化时从参数中获取port
,然后在done
阶段调用startServer
,其内部使用express
框架启动一个server,并将assets
目录作为静态资源目录。
需要指出的是,对于生产环境,我们通过isWatch
变量判断,如果是生产环境,则直接返回,不启动server。
最后,我们增加stopServer
方法,用于在done
阶段关闭server。
module.exports = class WebpackPluginFile {
// 省略代码
async stopServer() {
if (this.server) {
this.server.close();
}
}
apply(compiler) {
// 省略代码
compiler.hooks.done.tap('WebpackPluginFile', () => {
this.copyAssets(compiler).then(() => {
this.startServer(compiler);
});
});
compiler.hooks.shutdown.tap('WebpackPluginFile', () => {
this.stopServer();
});
}
}
小结
通过上述代码,我们实现了一个webpack
插件,用于在done
阶段将assets
目录移出mp-weixin
文件夹,并启动一个本地服务器,以便于开发时能够访问到资源。同时,我们区分了生产环境和开发环境,对于开发环境,我们通过watch
变量判断,对资源进行缓存,提高资源拷贝的效率,并启动一个本地服务器,以便于访问资源。而对于生产环境,我们只需要将资源移出mp-weixin
文件夹即可。
如何使用这个插件
由于uni-app有两个版本基于vue-cli和基于vite,其内核一样。我们采用的vue-cli的版本,因此我们可以在vue.config.js
中引入这个插件。
const WebpackPluginFile = require('./build/webpack-plugin-file');
module.exports = {
publicPath: isWatch ? 'http://127.0.0.1:8888/' : 'https://cdn.xxx.com/',
configureWebpack: {
plugins: [
new WebpackPluginFile({
port: 8888
})
]
}
}
WebpackPluginFile需要与publicPath配合使用。对于开发环境,我们将publicPath设置为http://127.0.0.1:8888/
,以便于访问到资源。对于生产环境,我们将publicPath设置为https://cdn.xxx.com/
,以便于小程序能够访问到资源。
更多思考
上述基本解决了我们一开始提出的问题,但是也衍生出一些新的问题,有兴趣的同学可以去深挖一下:
-
多个环境如何配置,这里面讲到了开发环境、生产环境,没有提到测试环境。引入测试环境后,
publicPath
如何配置? -
我们知道webpack 5对于
resource
资源默认情况有两种处理方式,asset
和inline
,某些场景会触发inline
模式,这种情况将导致图片不被复制,该情况下是否符合预期。 -
图片引入的改进。当前图片是需要手动引入的,并且通过变量传递给image元素,能否写成诸如
<image src="@/assets/logo.png" />
的形式呢?
参考资料
- uni-app官方文档
- 小程序分包加载