当我们将游戏构建发布到web平台时,勾选Md5 Cache选项可以开启MD5 Pipe,它的作用是给构建后的资源加上md5后缀,避免浏览器的缓存导致部分资源不是最新,因为使用了md5后缀后,当资源内容发生变化时,资源的名字就不一样了,缓存就会失效。

比如6e056173-d285-473c-b206-40a7fff5386e.json,在开启了MD5 Cache选项之后再构建会生成6e056173-d285-473c-b206-40a7fff5386e.a548e.json,在文件后缀.json前加上了一个简化的md5——a548e。

开启 MD5 PIPE

勾选了Md5 Cache选项后进行构建,生成的web-mobile/src/settings.xxx.js文件中会设置一个全局变量_CCSettings,该变量中有一个md5AssetsMap字段记录所有资源文件的简化md5。

    md5AssetsMap: {
"08/08ddbd1c9.json": "ade93",
"6e/6e056173-d285-473c-b206-40a7fff5386e.json": "a548e",
"assets/6e/6e056173-d285-473c-b206-40a7fff5386e.png": "68270"
}

在Creator生成的main.xxx.js中会获取_CCSettings变量进行处理,一个关键的地方就是cc.AssetLibrary.init,将settings.md5AssetsMap传入AssetLibrary。

cc.AssetLibrary.init({
libraryPath: 'res/import',
rawAssetsBase: 'res/raw-',
rawAssets: settings.rawAssets,
packedAssets: settings.packedAssets,
md5AssetsMap: settings.md5AssetsMap
});

在cc.AssetLibrary.init中,根据md5AssetsMap变量决定是否创建MD5Pipe,将md5Pipe插入到cc.loader.assetLoader之后。

    var md5AssetsMap = options.md5AssetsMap;
if (md5AssetsMap) {
var md5Pipe = new MD5Pipe(md5AssetsMap, _libraryBase, _rawAssetsBase);
cc.loader.insertPipeAfter(cc.loader.assetLoader, md5Pipe);
cc.loader.md5Pipe = md5Pipe;
}

在一些核心模块中,我们可以看到直接使用url时,都会检测是否存在cc.loader.md5Pipe,有则调用其transformURL对url进行处理,比如CCVideoPlayer.js,这些处理属于不走Pipeline的路径处理,属于这些组件的内部逻辑。

    _updateVideoSource: function () {
var sgNode = this._sgNode;
let url = '';
if (this.resourceType === ResourceType.REMOTE) {
url = this.remoteURL;
}
else if (this._clip) {
url = this._clip.nativeUrl || '';
}
if (url && cc.loader.md5Pipe) {
url = cc.loader.md5Pipe.transformURL(url);
}
sgNode.setURL(url);
},

MD5 PIPE的实现

Md5 Pipe的实现非常简单(位于load-pipeline下的md5-pipe.js),它只对res/import或res/raw-开头的url进行处理,根据md5AssetsMap对应的精简md5值,重新组装url。

MD5Pipe.prototype.transformURL = function (url, hashPatchInFolder) {
var index = url.indexOf('?');
var key = url;
if (index !== -1) {
key = url.substr(0, index);
}
// 如果是以libraryBase开头'res/import',去掉这个前缀
if (key.startsWith(this.libraryBase)) {
key = key.slice(this.libraryBase.length);
// 如果是以rawAssetsBase开头'res/raw-',也去掉这个前缀
} else if (key.startsWith(this.rawAssetsBase)) {
key = key.slice(this.rawAssetsBase.length);
} else {
// 其他情况不处理,比如是一个完整的http链接
return url;
} // 取出该资源对应的精简md5
let hashValue = this.md5AssetsMap[key];
if (hashValue) {
// 如果hashPatchInFolder为true则将这个md5组装到目录下
// JSB下的Spine、Label都有使用hashPatchInFolder
if (hashPatchInFolder) {
var dirname = cc.path.dirname(url);
var basename = cc.path.basename(url);
url = `${dirname}.${hashValue}/${basename}`;
} else {
// 正常情况是将精简md5插入到文件扩展名之前
var matched = false;
url = url.replace(ExtnameRegex, (function(match, p1) {
matched = true;
return "." + hashValue + p1;
}));
if (!matched) {
url = url + "." + hashValue
}
}
}
return url;
};

Cocos Creator 资源加载流程剖析【四】——额外流程(MD5 PIPE)的更多相关文章

  1. Cocos Creator 资源加载流程剖析【一】——cc.loader与加载管线

    这系列文章会对Cocos Creator的资源加载和管理进行深入的剖析.主要包含以下内容: cc.loader与加载管线 Download部分 Load部分 额外流程(MD5 Pipe) 从编辑器到运 ...

  2. Cocos Creator 资源加载流程剖析【二】——Download部分

    Download流程的处理由Downloader这个pipe负责(downloader.js),Downloader提供了各种资源的"下载"方式--即如何获取文件内容,有从网络获取 ...

  3. Cocos Creator 资源加载流程剖析【三】——Load部分

    Load流程是整个资源加载管线的最后一棒,由Loader这个pipe负责(loader.js).通过Download流程拿到内容之后,需要对内容做一些"加载"处理.使得这些内容可以 ...

  4. Cocos Creator 资源加载流程剖析【六】——场景切换流程

    这里讨论场景切换的完整流程,从我们调用了loadScene开始切换场景,到场景切换完成背后发生的事情.整个流程可以分为场景加载和场景切换两部分,另外还简单讨论了场景的预加载. 加载场景的流程 load ...

  5. Cocos Creator 资源加载流程剖析【五】——从编辑器到运行时

    我们在编辑器中看到的资源,在构建之后会进行一些转化,本章将揭开Creator对资源进行的处理. 资源处理的整体规则 首先我们将Creator的开发和运行划分为以下几个场景: 编辑器 当我们将资源放到编 ...

  6. Cocos Creator 资源加载(笔记)

    cc.loader 加载资源动态加载资源要注意两点,一是所有需要通过脚本动态加载的资源,都必须放置在 resources 文件夹或它的子文件夹下.resources 需要在 assets 文件夹中手工 ...

  7. 【原】从一个bug浅谈YUI3组件的资源加载

    篇前声明:为了不涉及业务细节,篇内信息统一以某游戏,某功能代替 前不久,某游戏准备内测客户端,开发人员测试过程中发现某功能突然不灵了,之前的测试一切ok,没有发现任何异常,第一反应是,游戏内浏览器都是 ...

  8. 记一次cocos项目的加载速度优化

    半个月前,我们用cosos creator做了一个简单的小游戏,也许算不上小游戏吧..一边学cocos,一边做,几经波折后终于上线了.然鹅,功能是实现了,但是加载速度十分感人(毕竟没经验嘛,无辜脸). ...

  9. 细谈unity资源加载和卸载

    转载请标明出处:http://www.cnblogs.com/zblade/ 一.概要 在了解unity的资源管理方式之后,接下来细谈一下Unity的资源是如何从磁盘中加载到运行时的内存中,以及又是如 ...

随机推荐

  1. linux 系统自动定制运行 crontab

    在UNIX下怎样实现和Windows下“计划任务”一样的功能 $crontab -e 编辑脚本 $crontab -l 察看脚本   用$crontab -e 编辑脚本,加入下列行 :分 小时 星期 ...

  2. mybatis精讲(五)--映射器组件

    目录 前言 标签 select insert|update|delete 参数 resultMap cache 自定义缓存 # 加入战队 微信公众号 前言 映射器之前我们已经提到了,是mybatis特 ...

  3. 关于jsp中jstl报错Can not find the tag library descriptor for "http://java.sun.com/jsp/jstl/core

    有的时候在开发jsp时,需要使用jstl时,在jsp上面引用jstl却出现错误:Can not find the tag library descriptor for "http://jav ...

  4. tomcat部署项目,详细!

    一.导出war包 1.先导出项目的war包(idea为例) 点+号,选择 之后点ok,确定,关闭窗口,回到idea主页面 在弹出窗口中选择新建的war,选build 之后在war导出目录,找到这个wa ...

  5. 【原创】002 | 搭上SpringBoot事务源码分析专车

    前言 如果这是你第二次看到师长,说明你在觊觎我的美色! 点赞+关注再看,养成习惯 没别的意思,就是需要你的窥屏^_^ 专车介绍** 该趟专车是开往Spring Boot事务源码分析的专车 专车问题 为 ...

  6. JetBrains 迷你地图插件 CodeGlance

    JetBrains 本身不带迷你地图功能,但可以通过插件的形式来实现. 直接在 Settings 里边搜索 CodeGlance,安装后重启 IDE 就有了.

  7. php mysql_connect 在同一host下多数据库mysql_select_db()的bug .

    操作方法 创建两个数据库test1 test2 同一个host下面 分别在两个数据库中创建表 -- ---------------------------- -- Table structure fo ...

  8. 这个立冬,我线下面基了一位TMD高级专家,太牛逼了!

    立冬刚过,迎面而来的是一股寒气.天气如此,市场亦是如此.昨天周五,和1个认识的技术专家老刘约饭,也算是线下面基,增进感情.每年我都要向比我高阶的朋友讨教.不由自主聊到了他的职场生涯.鱼哥一直以为自己命 ...

  9. iOS 自定义TabBarController

    转自:http://blog.csdn.net/xn4545945/article/details/35994863 一.自定义的思路 iOS中的TabBarController确实已经很强大了,大部 ...

  10. CF579 - A Raisinng bacteria

    You are a lover of bacteria. You want to raise some bacteria in a box. Initially, the box is empty. ...