【cocos creator】热更新

一、介绍

试了官方的热更新功能,总结一下
主要用于安卓包热更新

参考:

Cocos Creator 2.2.2 热更新简易教程
基于cocos creator2.4.x的热更笔记

二、使用软件
1、cocos creator v2.4.10

2、creator热更新插件:热更新manifest生成工具(creator应用商店下载)
cocos提供的脚本也可以用,但这个集成好了,更便捷
在这里插入图片描述

3、静态资源服务器(用的win+tomcat)
tomcat下载安装

4、模拟器

5、Android打包环境

三、配置热更新

1、首先确保项目可以正常构建出apk,进行一次构建
2、商城下载插件,安装好
在这里插入图片描述
3、配置好地址,只支持一个固定的地址,对应服务器存放热更新文件的文件夹
在这里插入图片描述
在这里插入图片描述
4、点击生成热更新包,再点击导入manifest

2.4.10控制台会提示xx已弃用,建议使用xx,按照提示全局搜索替换即可
在这里插入图片描述

5、把下面的脚本放在加载界面

manifest url绑定上一步导入的version,默认在assets下面
在这里插入图片描述

在这里插入图片描述


const { ccclass, property } = cc._decorator;@ccclass
export default class HotUpdate extends cc.Component {@property(cc.Node) progressNode: cc.Node = null;@property(cc.Label) alartLbl: cc.Label = null;@property(cc.Asset) manifestUrl: cc.Asset = null;  // 以资源的形式,绑定初始的manifestprivate _am: jsb.AssetsManager; // 热更的对象private _storagePath: string = '';private _hotPath: string = 'HotUpdateSearchPaths'private _progress: number = 0;private _downFailList: Map<string, number> = new Map();protected start(): void {this.alartLbl.string = ""this._showProgress(0);this.alartLbl.string = '';this._init();}private _init() {console.log("hotUpdate检测更新", cc.sys.os);if (cc.sys.os != cc.sys.OS_ANDROID && cc.sys.os != cc.sys.OS_IOS) {// 非原生平台不处理this._enterGame();return;}this.alartLbl.string = "检测更新..."this._storagePath = ((jsb.fileUtils ? jsb.fileUtils.getWritablePath() : '/') + this._hotPath);// 创建一个热更对象,提供一个本地资源路径,用来存放远程下载的资源。一个版本比较方法,用来判断是否需要更新this._am = new jsb.AssetsManager('', this._storagePath, this._versionCompareHandle);// 设置MD5 校验回调,这里一般用来比对文件是否正常// 如果这个方法返回true表示正常, 返回false的会触发文件下载失败,失败时会抛出错误事件ERROR_UPDATING,后面还会提到this._am.setVerifyCallback(function (filePath, asset) {// When asset is compressed, we don't need to check its md5, because zip file have been deleted.let compressed = asset.compressed;// Retrieve the correct md5 value.let expectedMD5 = asset.md5;// asset.path is relative path and path is absolute.let relativePath = asset.path;// The size of asset file, but this value could be absent.let size = asset.size;if (compressed) {return true;}else {// let expectedMD5 = asset.md5; // 远程project.manifest文件下资源对应的MD5// let resMD5: string = calculateMD5(filePath);  // filePath是文件下载到本地的路径,需要自行提供方法来计算文件的MD5// return resMD5 == expectedMD5;return true;}});if (cc.sys.os === cc.sys.OS_ANDROID) {// Some Android device may slow down the download process when concurrent tasks is too much.// The value may not be accurate, please do more test and find what's most suitable for your game.// this._am.setMaxConcurrentTask(2);}this._progress = 0;//检测更新this._checkUpdate();}/*** 版本对比,返回值小于0则需要更新* @param versionA 当前游戏内版本  本地* @param versionB 需要更新的版本  服务器* @returns */_versionCompareHandle(versionA: string, versionB: string) {console.log("JS Custom Version Compare: version A is " + versionA + ', version B is ' + versionB);let vA = versionA.split('.');let vB = versionB.split('.');for (let i = 0; i < vA.length; ++i) {let a = parseInt(vA[i]);let b = parseInt(vB[i] || '0');if (a === b) {continue;} else {return a - b;}}if (vB.length > vA.length) {return -1;} else {return 0;}}private async _checkUpdate() {// 判断当前热更的状态,没有初始化才加载本地Manifest,加载完成后状态会改变if (this._am.getState() === jsb.AssetsManager.State.UNINITED) {let url = this.manifestUrl.nativeUrl;this._am.loadLocalManifest(url); // 加载本地manifest}if (!this._am.getLocalManifest() || !this._am.getLocalManifest().isLoaded()) {console.error('Failed to load local manifest ...');return;}this._am.setEventCallback(this._checkCb.bind(this));this._am.checkUpdate();}private _checkCb(event: jsb.EventAssetsManager) {switch (event.getEventCode()) {case jsb.EventAssetsManager.ERROR_NO_LOCAL_MANIFEST:case jsb.EventAssetsManager.ERROR_DOWNLOAD_MANIFEST:case jsb.EventAssetsManager.ERROR_PARSE_MANIFEST:// manifest文件相关的错误,这里可以去做一些错误相关的处理console.error('加载manifest文件失败', event.getEventCode())break;case jsb.EventAssetsManager.ALREADY_UP_TO_DATE:// 当前版本已经是最新版本cc.log('hotUpdate Already up to date with the latest remote version.');this._enterGame();break;case jsb.EventAssetsManager.NEW_VERSION_FOUND:// 找到新的版本console.log(`hotUpdate New version found, please try to update. (${this._am.getTotalBytes()})`);cc.log(`hotUpdate getDownloadedBytes${this._am.getDownloadedBytes()}`);cc.log(`hotUpdate getTotalBytes${this._am.getTotalBytes()}`);this._am.setEventCallback(null);this._downLoadAlart();break;default:return;}}private _downLoadAlart() {this.alartLbl.string = "开始更新..."// 检测到有新版本则直接更新,也可以改成点击按钮再更新,这一步可以根据自己的需求来处理this._hotUpdate();}// 开始更新private _hotUpdate() {console.log('hotUpdate开始更新')this._am.setEventCallback(this._updateCb.bind(this));this._am.update();}private _updateCb(event: jsb.EventAssetsManager) {var needRestart = false, file = null;console.log('hotUpdate _updateCb', event.getEventCode())this.alartLbl.string = "更新中..."switch (event.getEventCode()) {case jsb.EventAssetsManager.ERROR_NO_LOCAL_MANIFEST:case jsb.EventAssetsManager.ERROR_DOWNLOAD_MANIFEST:case jsb.EventAssetsManager.ERROR_PARSE_MANIFEST:// manifest文件相关的错误,这里可以去做一些错误相关的处理break;case jsb.EventAssetsManager.ERROR_UPDATING:file = event.getAssetId();// 文件更新出错,这里有可能是验证方法没有通过,也有可能是文件下载失败等等this._updateFailList(file, false);break;case jsb.EventAssetsManager.ERROR_DECOMPRESS:// 文件解压缩失败break;case jsb.EventAssetsManager.UPDATE_FAILED:cc.error(`Fail to Update -> ${event.getMessage()}`);let failed = false;this._downFailList.forEach((count) => {if (count > 3) {failed = true;}});if (failed) {// 超过3次失败,显示下载失败// this._showUpdateFalid();} else {cc.log(`HotUpdate failed...Restart Update`);this._am.downloadFailedAssets();  // 重新下载失败的文件}break;case jsb.EventAssetsManager.UPDATE_PROGRESSION:// 这里处理正常下载进度显示if (event.getPercent()) {let downloadBytes = event.getDownloadedBytes() || 0;let totalBytes = event.getTotalBytes() || 0;this._progress = Math.floor(downloadBytes / totalBytes * 100);if (this._progress <= 0) return;this._showProgress(this._progress);let unit = 1048576;/* 1MB = 1,024 KB = 1,048,576 Bytes */let downloadedMB = (downloadBytes / unit).toFixed(2) + 'MB';let totalMB = (totalBytes / unit).toFixed(2) + 'MB';this.alartLbl.string = `下载资源: ${this._progress}% (${downloadedMB}/${totalMB})`;cc.log('hotUpdate downloadBytes=>', this._progress, downloadedMB, totalMB);}break;case jsb.EventAssetsManager.ALREADY_UP_TO_DATE:// 已经是最新版本cc.log('hotUpdate Already up to date with the latest remote version...');this._enterGame();break;case jsb.EventAssetsManager.UPDATE_FINISHED:cc.log(`hotUpdate Update finished. ${event.getMessage()}`);// 更新完成,这里去做重启needRestart = true;break;case jsb.EventAssetsManager.ASSET_UPDATED:// 每次资源下载成功会回调到这里,可以根据需求做一些处理cc.log('hotUpdate Updated file: ' + event.getAssetId() + ' ' + event.getMessage());file = event.getAssetId();this._updateFailList(file, true);break;default:break;}if (needRestart) {this._am.setEventCallback(null);var newPaths = this._storagePath;// 将当前路径写入到本地,持久化数据以便下次游戏启动的时候能拿到cc.sys.localStorage.setItem(this._hotPath, JSON.stringify([newPaths]));this.alartLbl.string = "更新完成"cc.game.restart();}}private _updateFailList(file: string, success: boolean) {if (success) {this._downFailList.delete(file);cc.log('hotUpdate 更新成功', file);} else {if (this._downFailList.get(file)) {let count = this._downFailList.get(file);this._downFailList.set(file, count + 1);cc.log(`hotUpdate ${file} download fail count ${count + 1}`);} else {this._downFailList.set(file, 1);cc.log(`hotUpdate ${file} download fail count 1`);}}}private _showProgress(percent: number) {percent > 100 && (percent = 100);let progress = cc.winSize.width * percent / 100;this.progressNode.width = progress;}private _enterGame() {console.log("hotUpdate 加载游戏");//TODO  加载游戏场景cc.director.loadScene("game");}
}

本来还要在build\jsb-default\main.js开头添加热更新代码,插件集成好了,省略这一步

6、点击编译导出apk,安装在手机上
原始含有热更新功能的包就打包好了

7、修改一部分内容,再次构建(只需要构建)
然后在热更新工具把版本号增加一位,点击生成热更包

将导出的压缩包放到服务器解压
在这里插入图片描述
手机上打开之前安装的包,就会自动更新成最新版本

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/31853.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

深度评测阿里云操作系统控制台:功能全面,体验卓越!

&#x1f4dd;个人主页&#x1f339;&#xff1a;Eternity._ &#x1f339;&#x1f339;期待您的关注 &#x1f339;&#x1f339; ❀ 阿里云操作系统控制台 操作系统控制台操作系统实践体验服务的开通创建ESC实例组件管理功能体验&#xff1a;节点健康系统诊断系统观测订阅管…

Spring Boot 解析 LocalDateTime 失败?Uniapp 传输时间变 1970 的原因与解决方案

目录 前言1. 问题分析2. 时间戳&#xff08;推荐&#xff0c;可尝试&#xff09;3. 使用 JsonDeserialize & JsonSerialize&#xff08;中立&#xff09;4. 前端传 ISO-8601 格式&#xff08;不推荐&#xff0c;可尝试&#xff09;5. 用 String&#xff08;中立&#xff09…

【vitepress】如何搭建并部署自己的博客网站

文章目录 新的改变旧的github.io地址,现在不用更新netlify托管之后为这个 一 如何搭建[1]:安装vitepress初始化Vitepress启动项目 二 如何部署[2]视频教程 [3] 新的改变 旧的github.io地址,现在不用 https://dl-hx.github.io/myBlog/ 更新netlify托管之后为这个 https://dl…

Cursor新版0.47.x发布

0.47.x - 可靠性、键盘快捷键与提前体验选项功能 本次更新主要聚焦于稳定性和性能改进&#xff0c;以确保现有功能更好地运行。 新功能与改进 键盘快捷键&#xff1a;所有键盘快捷键现在都可以在键盘快捷键菜单中找到。前往 设置 > 键盘快捷键 来修改或添加新的快捷键。 …

docker 小记

一、卸载 查看当前版本 docker -v2. 如果有&#xff0c;先停止docker systemctl stop docker如果是yum安装&#xff0c;卸载方式为 #已防版本冲突&#xff0c;直接卸载 yum remove docker \docker-client \docker-client-latest \docker-common \docker-latest \docker-lat…

XGBoost介绍

XGBoost&#xff1a;是eXtreme Gradient Boosting(极端梯度提升)的缩写&#xff0c;是一种强大的集成学习(ensemble learning)算法&#xff0c;旨在提高效率、速度和高性能。XGBoost是梯度提升(Gradient Boosting)的优化实现。集成学习将多个弱模型组合起来&#xff0c;形成一个…

Aliyun CTF 2025 web ezoj

文章目录 ezoj ezoj 进来一看是算法题&#xff0c;先做了试试看,gpt写了一个高效代码通过了 通过后没看见啥&#xff0c;根据页面底部提示去/source看到源代码&#xff0c;没啥思路&#xff0c;直接看wp吧&#xff0c;跟算法题没啥关系,关键是去看源码 def audit_checker(even…

大数据hadoop课程笔记

1.课程导入 柯洁 Alpha Go是人工智能领域的里程碑。 深度学习 大模型deepseek chatgpt 大模型 和 大数据 之间有着非常紧密的关系。可以说&#xff0c;大数据是大模型发展的基石&#xff0c;而大模型是大数据价值挖掘的重要工具。 https://youtu.be/nN-VacxHUH8?sifj7Ltk…

Pandas数据清洗实战之清洗猫眼电影

本次案例所需要用到的模块 pandas(文件读取保存 操作表格的模块) 将上次Scrapy爬取下来的文件 做个数据清洗 变成我们想要的数据 确定目的&#xff1a;将此文件中的duration字段中的分钟 和publisher_time上映去掉 只保留纯数值 数据清洗题目如下: 修复 publish_time列中的错…

UDP-网络编程/socket编程

一&#xff0c;socket相关接口 1&#xff0c;socket 我们来介绍socket编程的第一个接口&#xff1a;socket&#xff0c;它需要用到的头文件如图&#xff1a; 其中domain表示域或者协议家族&#xff1a; 本次我就用AF_INET&#xff08;ipv4&#xff09;来做演示 type参数表示…

《人月神话》:软件工程的成本寓言与生存法则

1975年&#xff0c;Fred Brooks在《人月神话》中写下那句振聋发聩的断言——“向进度落后的项目增加人力&#xff0c;只会让进度更加落后”——时&#xff0c;他或许未曾料到&#xff0c;这一观点会在半个世纪后的人工智能与云原生时代&#xff0c;依然如达摩克利斯之剑般悬在每…

ROS云课基础题库-01C++案例-甜甜圈

效率是核心&#xff0c;但效率高的教程会忽略掉非常多的细节。 解决问题的思路和细节对于一个问题的有效求解至关重要。 资料 云课五分钟-02第一个代码复现-终端甜甜圈C-CSDN博客 从云课五分钟到五秒钟焦虑的甜甜圈向前冲-CSDN博客 说明 复现重要性没有那么大&#xff0c;…

Oracle RHEL 5.8 安装 - 呆瓜式

前言 Red Hat Enterprise Linux Server release 5.8 为企业级 SO 镜像。绝大部分企业如果使用Oracle数据库均会使用其企业版 OS &#xff0c;能够很好的支持数据库的运行 文档目的 当前文档仅针对 VMware Workstation Pro 进行 OS 介质安装。 镜像下载地址 注意&#xff1…

【数据分析大屏】基于Django+Vue汽车销售数据分析可视化大屏(完整系统源码+数据库+开发笔记+详细部署教程+虚拟机分布式启动教程)✅

目录 一、项目背景 二、项目创新点 三、项目功能 四、开发技术介绍 五、项目功能展示 六、权威视频链接 一、项目背景 汽车行业数字化转型加速&#xff0c;销售数据多维分析需求激增。本项目针对传统报表系统交互性弱、实时性差等痛点&#xff0c;基于DjangoVue架构构建…

软件IIC和硬件IIC的主要区别,用标准库举例!

学习交流792125321&#xff0c;欢迎一起加入讨论&#xff01; 在学习iic的时候&#xff0c;我们经常会遇到软件 IC和硬件 IC,它两到底有什么区别呢&#xff1f; 软件 IC&#xff08;模拟 IC&#xff09;和硬件 IC&#xff08;外设 IC&#xff09;是两种实现 IC 总线通信的方式…

CSS-三大特性,盒子模型,圆角边框,盒子阴影,文字阴影

一、 CSS 的三大特性 CSS 有三个非常重要的三个特性:层叠性&#xff64;继承性&#xff64;优先级&#xff61; 1.层叠性 相同选择器给设置相同的样式,此时一个样式就会覆盖(层叠)另一个冲突的样式&#xff61;层叠性主要解决样式冲突 的问题 层叠性原则: 样式冲突,遵循的原…

基于 Qwen2.5-14B + Elasticsearch RAG 的大数据知识库智能问答系统

AI 时代&#xff0c;如何从海量私有文档&#xff08;非公开&#xff09;中快速提取精准信息成为了许多企业和个人的迫切需求。 本文介绍了一款基于 Qwen2.5-14B 大语言模型&#xff08;换成 DeepSeek 原理一致&#xff09;与 Elasticsearch 搜索引擎构建的大数据知识库智能问答…

算法手记1

&#x1f984;个人主页:修修修也 &#x1f38f;所属专栏:数据结构 ⚙️操作环境:Visual Studio 2022 目录 一.NC313 两个数组的交集 题目详情: 题目思路: 解题代码: 二.AB5 点击消除 题目详情: 题目思路: 解题代码: 结语 一.NC313 两个数组的交集 牛客网题目链接(点击即可跳转)…

JMeter使用BeanShell断言

BeanShell简介 BeanShell是使用Java语法的一套脚本语言&#xff0c;在JMeter的多种组件中都有BeanShell的身影&#xff0c;如&#xff1a; 定时器&#xff1a;BeanShell Timer前置处理器&#xff1a;BeanShell PreProcessor采样器&#xff1a;BeanShell Sampler后置处理器&am…

【技海登峰】Kafka漫谈系列(五)Java客户端之生产者Producer核心组件与实现原理剖析

【技海登峰】Kafka漫谈系列(五)Java客户端之生产者Producer核心组件与实现原理剖析 向Kafka Broker服务节点中发送主题消息数据的应用程序被称为生产者,生产者与消费者均属于Kafka客户端,几乎所有主流语言都支持调用客户端API。官方提供了基于Java实现的kafka-clients,用于…