使用 Zip 加速 CocosWeb 加载

01

1 引言

使用 Zip 加速 CocosWeb 加载

前段时间使用 Cocos3.8 做一个云展厅项目,要求在 Web 平台上线(微信 H5&浏览器)。

这个云展厅项目使用 gltf 模型,gltf 模型中有拆分很多 Mesh 和材质。

而在 Cocos 中 gltf 会被拆分解析为 Cocos 资产,发布 Web 后加载一个这种大 gltf 就可能有几百个 request,明明网速飞起,但是加载还是很慢,因为此时项目中加载速度的瓶颈已经不是网速,而是 request 的数量太多了。

2 产生的原因&如何解决

为什么有这么多 request

  • 新建一个项目,从ThreeJS的demo中拿来一个gltf,将这个gltf直接放在resources文件夹下,方便在demo进行预加载,这个gltf中共28材质,32个Mesh和一些骨骼&贴图

02

  • 创建Start场景用于预加载资源,和Game场景用于展示模型

03

  • 创建Start脚本,对资源做一个简单的预加载,加载完成后进入Game场景
import {_decorator, Component, director, Label, settings, ProgressBar, resources, assetManager, Settings} from 'cc';

const {ccclass, property} = _decorator;

@ccclass('Start')
export class Start extends Component {

    @property(ProgressBar)
    progressBar: ProgressBar;

    @property(Label)
    barLab: Label = null;

    async start() {
    
        // 直接加载resources根目录
        await this.preload([
            {
                path: "/",
                type: "dir",
            },
        ]);

        director.loadScene("Game");

    }

    /**
     * 预加载资源
     */
    preload = (pkg) => {
        return new Promise<void>((resolve, reject) => {
            const pathArr = [];

            pkg.forEach((asset) => {
                if (typeof asset == "string") {
                    return pathArr.push(asset);
                }
                switch (asset.type) {
                    case "dir":
                        resources.getDirWithPath(asset.path).forEach((v) => pathArr.push(v.path));
                        break;
                        
                    default:
                        pathArr.push(asset.path);
                }
            });

            resources.load(
                pathArr,
                (finish: number, total: number, _) => {
                    const pro = finish / total;
                    if (pro < this.progressBar.progress) {
                        return;
                    }
                    this.progressBar.progress = pro;
                    this.barLab.string = `正在加载中   ${(pro * 100).toFixed(0)}%`;
                },
                async (error) => {
                    if (error) console.error(error);
                    resolve();
                }
            );
        });
    }
}
  • 运行游戏,在本地web环境查看NetWork,request数量为379,和这个gltf相关的就有216个,打包发布至Web环境,选择合并所有Json,加载该gltf总共用了35次request。

本地Web

04

打包合并所有Json

05

  • 明明只有一个gltf却用了35次request来加载。

  • 原因在于Cocos将gltf资源转换成了Cocos资产,将Mesh,材质等拆解了出来,每个资源除了资源本身外还会有一个记录属性依赖的Json文件

06

如何解决

  • 将整个bundle打包,比如打包成zip文件,进入游戏先加载需要的bundle的zip文件一次下载并且解压,之后需要资源直接从解压完的文件里取。

3 Zip 和 JsZip 的使用

Zip

不必多言,想必大家都知道

JsZip 的使用

不必重复了,直接上npm平台参考文档吧

jszip
https://www.npmjs.com/package/jszip

文档看起来肯定很抽象,不如直接跟着下面的步骤实操。

4 探索 Cocos 加载资源的奥秘

  • 查看NetWork,可以发现Cocos下载资源都会通过一个download-file.ts文件,移动鼠标到download-file.ts上就可以看到他的调用栈,其中主要是download-file.tsdownloader.ts也就是资源下载管线的一部分。那么我们直接打开源码进入到这里

07

08

  • 在代码中我们可以看到,大部分文件的下载都是通过downloadFile方法进行下载的,这个方法就是刚才的download-file.ts中的方法,该方法使用XMLHttpRequest下载文件

09

10

  • 既然我们已经知道在Cocos中,大部分资源的下载都依赖于XMLHttpRequest,那么我们可以想办法拦截它,重定向到我们解包的zip包就可以避免发起它真实的网络请求从而消耗大量时间了。

5 如何加载自己的 Zip 包

装载自己的 zip

  • 浅写一个 ZipLoader 并作为单例使用
  • 偷个懒,这里直接使用 Cocos 内置 API 加载远程文件吧,注意这个 API 已经弃用,未来可能删除
  • 我们直接不管容错,把 demo 跑通再说
  • 并使用 Promise 配合外部 async/await 来简化控制流。
import {assetManager} from "cc";
import JSZIP from "jszip";

export default class ZipLoader {

    static _ins: ZipLoader;
    static get ins() {
        if (!this._ins) {
            this._ins = new ZipLoader();
        }
        return this._ins;
    }

    /**
     * 下载单个zip文件为buffer
     * 为什么这里带上后缀名后面会讲到,是为了方面自动化
     * @param path 文件路径
     * @returns zip的buffer
     */
    downloadZip(path: string) {
        return new Promise((resolve) => {
            assetManager.downloader.downloadFile(
                path + '.zip',
                {xhrResponseType: "arraybuffer"},
                null,
                (err, data) => {
                    resolve(data);
                }
            );
        });
    }

    /**
     * 解析加载Zip文件
     * @param path 文件路径
     */
    async loadZip(path: string) {
        const jsZip = JSZIP();

        // 下载
        const zipBuffer = await this.downloadZip(path);

        // 解压
        const zipFile = await jsZip.loadAsync(zipBuffer);
    }
}
  • 在之前的Start.ts中添加代码
  • 注意以下几点
  • 作者这里有自动化压缩上传插件,会自动修改server字段,server就是项目发布的根目录带协议和域名,例如https://xxx.com/cc_project/version/
  • 作者这里会将需要zip加载的包注入到window上
  • 注入的js类似window["zipBundle"] = ["internal", "main", "resources"];
  • 作者这里所有bundle全都在远程所以只加载remote中的文件就行了且zip文件和bundle文件夹在同一目录下
/* ... */

@ccclass('Start')
export class Start extends Component {

    /* ... */
    
    async start() {

        // 作者这里有自动化压缩上传插件,会自动修改server字段
        // 并且会将需要zip加载的包注入到window上
        // 注入的js类似与下面这行
        // window["zipBundle"] = ["internal", "main", "resources"];
        const remoteUrl = settings.querySettings(Settings.Category.ASSETS, "server");
        const zipBundle = window["zipBundle"] || [];

        // 作者这里所有bundle全都是远程bundle所以只加载remote中的文件就行了
        // 且zip文件和bundle文件夹在同一目录下
        const loadZipPs = zipBundle.map((name: string) => {
            return ZipLoader.ins.loadZip(`${remoteUrl}remote/${name}`);
        });
        
        // 先等zip加载完
        await Promise.all(loadZipPs);

        // 直接加载resources根目录
        await this.preload([
            {
                path: "/",
                type: "dir",
            },
        ]);

        director.loadScene("Game");

    }
    
    /* ... */
    
}

不自定义引擎,拦截 Cocos 加载

查阅了 Cocos 的文档没有很好的批量实现这个需求的方式,又因为 Cocos 引擎更新比较频繁,我个人又喜欢多用新引擎新功能,所以我选择不自定义引擎,直接采用拦截 Cocos 加载的方法实现将加载资源替换到自己的zip包。

通过阅读源码我们已经知道除了图片资源,其他资源都是通过 XMLHttpRequest 来加载的,那么很简单,我们直接拦截 XMLHttpRequest 就行了。

那么你问我怎么才能拦截一个浏览器 Native 对象,这可是 Js,Js 无所不能!

拦截open和send

  • 不必多说,按下面这种方法就可拦截一个XMLHttpRequest来做一些操作
// 拦截open
const oldOpen = XMLHttpRequest.prototype.open;
// @ts-ignore
XMLHttpRequest.prototype.open = function (method, url, async, user, password) {
  return oldOpen.apply(this, arguments);
}

// 拦截send
const oldSend = XMLHttpRequest.prototype.send;
XMLHttpRequest.prototype.send = async function (data) {
  return oldSend.apply(this, arguments);
}
  • 添加解析zip的代码,将zip中的代码解析到完整的路径上
/* ... */;

const ZipCache = new Map<string, any>();

export default class ZipLoader {

    /* ... */

    constructor() {
        this.init();
    }

    /* ... */

    /**
     * 解析加载Zip文件
     * @param path 文件路径
     */
    async loadZip(path: string) {

        const jsZip = JSZIP();

        const zipBuffer = await this.downloadZip(path);

        const zipFile = await jsZip.loadAsync(zipBuffer);
        // 解析zip文件,将路径,bundle名,文件名拼起来,直接存在一个map里吧
        zipFile.forEach((v, t) => {
            if (t.dir) return;
            ZipCache.set(path + "/" + v, t);
        });
    }
    
    init() {
        // 拦截open
        const oldOpen = XMLHttpRequest.prototype.open;
        // @ts-ignore
        XMLHttpRequest.prototype.open = function (method, url, async, user, password) {
            return oldOpen.apply(this, arguments);
        }

        // 拦截send
        const oldSend = XMLHttpRequest.prototype.send;
        XMLHttpRequest.prototype.send = async function (data) {
            return oldSend.apply(this, arguments);
        }
    }
  • 在拦截的open和send中取消网络请求,直接定向到我们缓存在zip资源,由于我们不能直接修改xhr的response,因为他是只读属性,所以我们要借助Object.getOwnPropertyDescriptorObject.defineProperty,话不多说,直接看代码把
/* ... */

const ZipCache = new Map<string, any>();
const ResCache = new Map<string, any>();

export default class ZipLoader {

    /* ... */

    constructor() {
        this.init();
    }

    /* ... */

    init() {

        const accessor = Object.getOwnPropertyDescriptor(XMLHttpRequest.prototype, 'response');
        Object.defineProperty(XMLHttpRequest.prototype, 'response', {
            get: function () {
                if (this.zipCacheUrl) {
                    const res = ResCache.get(this.zipCacheUrl);
                    return this.responseType === "json"
                        ? JSON.parse(res)
                        : res;
                }
                return accessor.get.call(this);
            },
            set: function (str) {
                // console.log('set responseText: %s', str);
                // return accessor.set.call(this, str);
            },
            configurable: true
        });

        // 拦截open
        const oldOpen = XMLHttpRequest.prototype.open;
        // @ts-ignore
        XMLHttpRequest.prototype.open = function (method, url, async, user, password) {
            // 有这个资源就记录下来
            if (ZipCache.has(url as string)) {
                this.zipCacheUrl = url;
            }
            return oldOpen.apply(this, arguments);
        }

        // 拦截send
        const oldSend = XMLHttpRequest.prototype.send;
        XMLHttpRequest.prototype.send = async function (data) {
            if (this.zipCacheUrl) {
                // 有缓存就不解析了
                if (!ResCache.has(this.zipCacheUrl)) {

                    const cache = ZipCache.get(this.zipCacheUrl);

                    if (this.responseType === "json") {
                        const text = await cache.async("text");
                        ResCache.set(this.zipCacheUrl, text);
                    } else {
                        // 直接拿cocos设置的responseType给zip解析
                        const res = await cache.async(this.responseType);
                        ResCache.set(this.zipCacheUrl, res);
                    }
                }
                
                // 解析完了直接调用onload,并且不再发起真实的网络请求
                this.onload();
                return;
            }

            return oldSend.apply(this, arguments);
        }
    }
}
  • 打包项目手动压缩bundle文件夹上传进行测试,可以看到我们下载了三个zip资源,大量的json和bin文件夹都没有下载,和我们测试gltf相关的文件,仅有两张贴图而已,而且能正常进入Game场景,说明我们刚才写的代码是有效的,加载该gltf文件的request次数从35次降到了3次

11

12

6 发布自动化

  • 编写 Cocos 插件打包自动压缩 bundle 为 zip
  • 这个就比较简单了,新建一个构建插件
  • 编写一个zip.ts,文件内容如下
import * as fs from "fs";
import JSZIP from "jszip";

//读取目录及文件
function readDir(zip, nowPath) {
    const files = fs.readdirSync(nowPath);
    files.forEach(function (fileName, index) {
        console.log(fileName, index);
        const fillPath = nowPath + "/" + fileName;
        const file = fs.statSync(fillPath);
        if (file.isDirectory()) {
            const dirlist = zip.folder(fileName);
            readDir(dirlist, fillPath);
        } else {
            // 排除图片文件,下面会讲到
            if (fileName.endsWith(".png") || fileName.endsWith(".jpg")) {
                return;
            }
            zip.file(fileName, fs.readFileSync(fillPath));//压缩目录添加文件
        }
    });
}

//开始压缩文件
export function zipDir(name, dir, dist) {
    return new Promise<void>((resolve, reject) => {
        const zip = new JSZIP();
        readDir(zip, dir);
        zip.generateAsync({
            type: "nodebuffer",
            compression: "DEFLATE",
            compressionOptions: {
                level: 9
            }
        }).then(function (content) {
            fs.writeFileSync(`${dist}/${name}.zip`, content, "utf-8");
            resolve();
        });
    });
}
  • 在hooks中onAfterBuild中编写压缩脚本的内容,压缩脚本的内容在其他操作(如压缩图片,混淆代码,修改js等)都做完之后,且在上传资源前,且只针对web模板大致内容如下
export const onAfterBuild: BuildHook.onAfterBuild = async function (options: ITaskOptions, result: IBuildResult) {

    // 非需要的模板不进行这个操作
    if (options.platform !== "web-mobile") return;

    // 修改脚本,混淆代码,压缩资源等
    / ... /
    if (fs.existsSync(result.dest + "/remote")) {
        await Promise.all(
            fs.readdirSync(result.dest + "/remote")
                .map((dirName) => {
                    return zipDir(dirName, result.dest + "/remote/" + dirName, result.dest + "/remote");
                })
        )
    }
    / ... /

    // 上传

};

7 做一个简单的优化

通过前面阅读源码和 NetWork 中看到,Cocos 加载图片的方式不是通过 XMLHttpRequest,而是通过创建 Image 对象的方式。

此片文章的内容暂时不研究如何将加载图片也替换到使用自己的 Zip,因为我自己也还没做。

所以我选择直接在打包 zip 的时候过滤 png/jpg 文件来降低 zip 包的大小,仅在 zip 中打包需要的文件即可。

公众号原文地址: Cocos发布Web几千个request?使用zip压缩bundle加速加载。

21赞

公众号原文地址: Cocos发布Web几千个request?使用zip压缩bundle加速加载。

1赞

可以上传个demo吗

有没可能页游就是按需加载的压缩有点多余

目前我这个,就是进场景直接显示整个场馆的内容,所以在loading的时候堆了大量request,还是存在很大瓶颈的

等我整理一个,不过上面的文章里就已经包含了demo全部代码了

把模型资源放倒于一个分包中,把分包打包成zip可以不 :rofl:

我觉得可以

好像有道理,不过我不清楚楼主是否是zip在本地直接加载的

换言之就是预先下载zip包并且解析zip包,拦截引擎发送的请求,如果有相同即返回zip解析后的某个文件,是这样理解么

是的,没错

其实request默认有限制的,不会数量多少就发多少。ZIP涉及到版本管理,另一个问题

就是有限制才导致慢啊

之前就有这样的想法,
想不到就看到了相关文章

瞬间内存占用多少.在垃圾设备上怕是要crash吧.

web下bundle好像没有提供zip选项

所以官方干嘛偷懒,这种功能官方不是分分钟就能做吗,哈哈哈哈哈

这个可以交钱定制 :smiley:

1赞

哈哈哈,我算是被你整破防了