Cocos Creator 热更新
参考博客:https://www.jianshu.com/p/cec263b6b9ac
官方文档:https://docs.cocos.com/creator/manual/zh/advanced-topics/hot-update.html
官方提供的那个21点热更新demo,在ios和andriod真机端都没有跑出来,黑屏,粉屏,无语。。。
只好按参考博客的作者新建项目直接开搞
用真机测试,Creator自带的模拟器,能更新资源,但是更新后加载不了资源,没法测试效果
测试下来安卓和iOS是可以共用一套代码一次更新构建,安卓或者iOS构建的新版本,两端都会提示更新,且能正常更新和进入
Creator版本:2.1.0
HotUpdate.js 使用的是官方demo 最新里的
UpdatePanel.js 使用的是官方demo 最新里的
资源也是借用官方demo里的,整的目录结构如下
因为没有其他更多依赖,代码也放一下
HotUpdate.js
var UpdatePanel = require('./UpdatePanel'); // Custom manifest removed the following assets: // 1. res/raw-assets/2a/2a40e5e7-4c4a-4350-9e5d-76757755cdd2.png // 2. res/raw-assets/2d/2d86a854-63c4-4b90-8b88-a4328b8526c2.png // So when custom manifest used, you should be able to find them in downloaded remote assets var customManifestStr = JSON.stringify({ "packageUrl": "http://192.168.50.220:5555/tutorial-hot-update/remote-assets/", "remoteManifestUrl": "http://192.168.50.220:5555/tutorial-hot-update/remote-assets/project.manifest", "remoteVersionUrl": "http://192.168.50.220:5555/tutorial-hot-update/remote-assets/version.manifest", "version": "1.10", "assets": { "src/cocos2d-jsb.js": { "size": 3341465, "md5": "fafdde66bd0a81d1e096799fb8b7af95" }, "src/project.dev.js": { "size": 97814, "md5": "ed7f5acd411a09d4d298db800b873b00" }, "src/settings.js": { "size": 3849, "md5": "deb03998a4cfb8f8b468fba8575cb1c9" }, "res/import/03/0379fb962.json": { "size": 1107, "md5": "d102d0f14ed6b6cb42cc28d88b3b9069" }, "res/import/0c/0cd5de143.json": { "size": 80883, "md5": "f06347880038a1381043ed505d6f8a9a" }, "res/import/0d/0d756af45.json": { "size": 10137, "md5": "02dc8b795e79b9fd62e00d4a2c70c8c1" }, "res/import/0d/0dc6a4e59.json": { "size": 14970, "md5": "a500f696892df6869341dff5f31b1a33" }, "res/import/41/4128b78b-00ae-4d8a-ae35-4e5ca5c5cde9.json": { "size": 76, "md5": "3f79d93ce8d42b186ecd43d868c8d023" }, "res/import/49/49539cb0-3893-459a-b310-7cc1b7f6d335.json": { "size": 72, "md5": "8a36388cda7c3773b5bf7a53d8824535" }, "res/import/9e/9e2ae507-fae5-4511-940b-f2e46f81b790.json": { "size": 74, "md5": "98f6b1d93a4ee3a1f2074be9ce00fbb2" }, "res/raw-assets/0e/0ed8cf6e-8c04-4569-8d17-626a26e1099f.png": { "size": 4665, "md5": "9e8bf9af30ac7a9ea9d3b72f37a193e1" }, "res/raw-assets/13/137d1ca6-e90c-440b-9fa2-4b9ffff569f7.png": { "size": 1627, "md5": "75060291e24294abd6a52553fa22317e" }, "res/raw-assets/15/15d5f3f0-f965-4c00-945b-d2c8faee78b6.png": { "size": 3840, "md5": "cb525edab8063a845e6bd1e9d29b8cde" }, "res/raw-assets/19/19509bb1-dc08-4cbf-ab8f-2460e207265c.png": { "size": 9638, "md5": "6e159c9cc1b971d3921bc8908071a70b" }, "res/raw-assets/26/26e9a867-3d2f-4981-8a33-82d440de7aff.png": { "size": 6417, "md5": "5c139729708dd26bd461bcd3e8201823" }, "res/raw-assets/2d/2ddfe005-2129-41d8-aeec-2b1f51f02962.png": { "size": 2290, "md5": "874dccfd88108a9f0188bda59c5df183" }, "res/raw-assets/34/3459ab36-782c-4c4e-8aef-7280aff8b272.png": { "size": 18969, "md5": "3a810a636f3779b357e854155eafa4b6" }, "res/raw-assets/36/36b6ea73-ff48-430e-a0c7-0e5e8defe341.png": { "size": 2711, "md5": "e64625aeb59a1de225e718a7126634ad" }, "res/raw-assets/39/394bac82-54fb-472f-a27f-b5107821bfb8.png": { "size": 1641, "md5": "049d2201d7d99fc6dbdb017d8d8bd9b8" }, "res/raw-assets/3c/3cedb8b4-8532-4037-a00e-b8d3e0013158.png": { "size": 94313, "md5": "a2e763866c1bdd6b189be69f3d37eedd" }, "res/raw-assets/41/4128b78b-00ae-4d8a-ae35-4e5ca5c5cde9.manifest": { "size": 6358, "md5": "c1d18879851e567545ea04bf135a325f" }, "res/raw-assets/49/49539cb0-3893-459a-b310-7cc1b7f6d335.mp3": { "size": 971644, "md5": "f45ec6666f06b729d8c0461bc89d4b94" }, "res/raw-assets/4e/4e06c7f1-72ac-4e4e-90de-683e16905156.png": { "size": 2406, "md5": "5f0c28e0eed7ec0cb75e45f5937dd7c6" }, "res/raw-assets/50/50da5486-dfa1-46d2-9d4f-686eb5527c1a.png": { "size": 6911, "md5": "51cf32529c923146f06019a58398c98d" }, "res/raw-assets/52/5245e25c-010c-45fb-84a3-f3bce95793e7.png": { "size": 3963, "md5": "0f050ba45e09986b3d785b7b23ffcc1e" }, "res/raw-assets/6d/6de06a23-d0de-4766-a9e1-a0314136d62e.png": { "size": 10878, "md5": "9f89eec7a1b0f615a3c1bab0857aefff" }, "res/raw-assets/70/700faa17-11a6-46cd-aeb5-d6900bc264f8.png": { "size": 3765, "md5": "878e89a0a3e02b13beee9f3274f2ca39" }, "res/raw-assets/71/71561142-4c83-4933-afca-cb7a17f67053.png": { "size": 1050, "md5": "c06a93f5f1a8a1c6edc4fd8b52e96cbf" }, "res/raw-assets/80/8071df9d-029b-40e8-98f3-8eab08dbf6ca.png": { "size": 25205, "md5": "f688777a92fba11bfe85c3061a4476e5" }, "res/raw-assets/82/82fe58d4-ae13-4806-9a41-2e73902ea811.png": { "size": 24298, "md5": "b807df8ffcb540f3dd20db75ac95b73b" }, "res/raw-assets/83/83cc2086-d713-47a0-8d86-a8d6068b6258.png": { "size": 3782, "md5": "9827ce705349caa604e1aba1d53b0fd9" }, "res/raw-assets/96/96e3e293-4e36-426d-a0a6-eb8d025c0d5b.png": { "size": 15379, "md5": "d6ce47aed38348a1ea0f003fa0063079" }, "res/raw-assets/97/97a6316c-7fcb-4ffe-9045-35625bc6abf6.png": { "size": 2187, "md5": "f3f41b4c0783a751e561f1b84d91a70b" }, "res/raw-assets/97/97bb9c9c-5568-4419-af04-4ed5a2969a02.png": { "size": 10370, "md5": "48ab94f1c34b0e9a047297cab1aeabc4" }, "res/raw-assets/99/99170b0b-d210-46f1-b213-7d9e3f23098a.png": { "size": 1177, "md5": "d1118d133683bb4227d5e60c79c846b7" }, "res/raw-assets/99/99acc716-33df-4c4c-879d-cc3407f0cd8c.png": { "size": 9754, "md5": "23e7221934021f3fbe6c6a52b023ded8" }, "res/raw-assets/9e/9e2ae507-fae5-4511-940b-f2e46f81b790.mp3": { "size": 3179, "md5": "90d17b1a25200c90e292d9a3748c9fec" }, "res/raw-assets/ac/ac11439d-3758-49f5-8728-81ed22c1ed96.png": { "size": 11935, "md5": "c20ae4a74c42b2aed28bb8c9247eb5d5" }, "res/raw-assets/ae/ae4e2188-2b7b-42a9-85e1-8fb987600b04.png": { "size": 634171, "md5": "07b03f7145b75579708ae05ea2a2c029" }, "res/raw-assets/af/afe329a6-e85e-46a0-98ed-8a34e128907b.png": { "size": 2209, "md5": "30ae2fe844c7c53f1d00291051230607" }, "res/raw-assets/b2/b2037f34-04ff-4351-b9da-5be4bb557017.png": { "size": 1530, "md5": "bb96dacb8b09e0443d83462cc7b20095" }, "res/raw-assets/b4/b43ff3c2-02bb-4874-81f7-f2dea6970f18.png": { "size": 1114, "md5": "83fcc9912e01ae5411c357651fb8b1cf" }, "res/raw-assets/c3/c39ea496-96eb-4dc5-945a-e7c919b77c21.png": { "size": 2548, "md5": "ae7a04af25e238a5478170759b55a7ba" }, "res/raw-assets/ca/caaaf9ff-5036-4232-a8a7-88b80b2e4c88.png": { "size": 1829, "md5": "94d761c4626df88053787f17fa09914d" }, "res/raw-assets/ca/cacafa85-d8e9-4716-bcdb-7eba457e409c.png": { "size": 7380, "md5": "e6bb0f4d041257653f07da2dfe1edd09" }, "res/raw-assets/ce/ce6d2de9-7056-4ba8-a1b1-40b00bb6f469.png": { "size": 10982, "md5": "52aa0df577edafe11de1cfdb44422895" }, "res/raw-assets/cf/cfef78f1-c8df-49b7-8ed0-4c953ace2621.png": { "size": 1140, "md5": "a4b5953dffeb145b4b70072d91c4052b" }, "res/raw-assets/d5/d5dfe6a8-eb19-4aae-a74f-83b71eaa57dc.png": { "size": 8755, "md5": "aeb1055ced334ce20fe030579e187494" }, "res/raw-assets/da/da3e556f-1bce-4c31-87dc-897ea2d788e2.png": { "size": 11636, "md5": "d81124346c110eb1377f7b56346b31e4" }, "res/raw-assets/e8/e851e89b-faa2-4484-bea6-5c01dd9f06e2.png": { "size": 1082, "md5": "90cf45d059d0408bec327f66eae5764c" }, "res/raw-assets/ec/ec244ee5-6f1f-4920-9b69-d4df0e78ec2d.png": { "size": 55581, "md5": "68fdff7430b1b02f3a6e76bea92c6372" }, "res/raw-assets/fc/fccc4d85-6ad4-496d-9b33-ea76e69da132.png": { "size": 82257, "md5": "df4359cdcb956f52f2e5b4ef777bbb7d" } }, "searchPaths": [] }); cc.Class({ extends: cc.Component, properties: { panel: UpdatePanel, manifestUrl: { type: cc.Asset, default: null }, updateUI: cc.Node, _updating: false, _canRetry: false, _storagePath: '' }, checkCb: function (event) { cc.log('Code: ' + event.getEventCode()); switch (event.getEventCode()) { case jsb.EventAssetsManager.ERROR_NO_LOCAL_MANIFEST: this.panel.info.string = "No local manifest file found, hot update skipped."; break; case jsb.EventAssetsManager.ERROR_DOWNLOAD_MANIFEST: case jsb.EventAssetsManager.ERROR_PARSE_MANIFEST: this.panel.info.string = "Fail to download manifest file, hot update skipped."; break; case jsb.EventAssetsManager.ALREADY_UP_TO_DATE: this.panel.info.string = "Already up to date with the latest remote version."; break; case jsb.EventAssetsManager.NEW_VERSION_FOUND: this.panel.info.string = 'New version found, please try to update.'; this.panel.checkBtn.active = false; this.panel.fileProgress.progress = 0; this.panel.byteProgress.progress = 0; break; default: return; } this._am.setEventCallback(null); this._checkListener = null; this._updating = false; }, updateCb: function (event) { var needRestart = false; var failed = false; switch (event.getEventCode()) { case jsb.EventAssetsManager.ERROR_NO_LOCAL_MANIFEST: this.panel.info.string = 'No local manifest file found, hot update skipped.'; failed = true; break; case jsb.EventAssetsManager.UPDATE_PROGRESSION: this.panel.byteProgress.progress = event.getPercent(); this.panel.fileProgress.progress = event.getPercentByFile(); this.panel.fileLabel.string = event.getDownloadedFiles() + ' / ' + event.getTotalFiles(); this.panel.byteLabel.string = event.getDownloadedBytes() + ' / ' + event.getTotalBytes(); var msg = event.getMessage(); if (msg) { this.panel.info.string = 'Updated file: ' + msg; // cc.log(event.getPercent()/100 + '% : ' + msg); } break; case jsb.EventAssetsManager.ERROR_DOWNLOAD_MANIFEST: case jsb.EventAssetsManager.ERROR_PARSE_MANIFEST: this.panel.info.string = 'Fail to download manifest file, hot update skipped.'; failed = true; break; case jsb.EventAssetsManager.ALREADY_UP_TO_DATE: this.panel.info.string = 'Already up to date with the latest remote version.'; failed = true; break; case jsb.EventAssetsManager.UPDATE_FINISHED: this.panel.info.string = 'Update finished. ' + event.getMessage(); needRestart = true; break; case jsb.EventAssetsManager.UPDATE_FAILED: this.panel.info.string = 'Update failed. ' + event.getMessage(); this.panel.retryBtn.active = true; this._updating = false; this._canRetry = true; break; case jsb.EventAssetsManager.ERROR_UPDATING: this.panel.info.string = 'Asset update error: ' + event.getAssetId() + ', ' + event.getMessage(); break; case jsb.EventAssetsManager.ERROR_DECOMPRESS: this.panel.info.string = event.getMessage(); break; default: break; } if (failed) { this._am.setEventCallback(null); this._updateListener = null; this._updating = false; } if (needRestart) { this._am.setEventCallback(null); this._updateListener = null; // Prepend the manifest's search path var searchPaths = jsb.fileUtils.getSearchPaths(); var newPaths = this._am.getLocalManifest().getSearchPaths(); console.log(JSON.stringify(newPaths)); Array.prototype.unshift.apply(searchPaths, newPaths); // This value will be retrieved and appended to the default search path during game startup, // please refer to samples/js-tests/main.js for detailed usage. // !!! Re-add the search paths in main.js is very important, otherwise, new scripts won't take effect. cc.sys.localStorage.setItem('HotUpdateSearchPaths', JSON.stringify(searchPaths)); jsb.fileUtils.setSearchPaths(searchPaths); cc.audioEngine.stopAll(); cc.game.restart(); } }, loadCustomManifest: function () { if (this._am.getState() === jsb.AssetsManager.State.UNINITED) { var manifest = new jsb.Manifest(customManifestStr, this._storagePath); this._am.loadLocalManifest(manifest, this._storagePath); this.panel.info.string = 'Using custom manifest'; } }, retry: function () { if (!this._updating && this._canRetry) { this.panel.retryBtn.active = false; this._canRetry = false; this.panel.info.string = 'Retry failed Assets...'; this._am.downloadFailedAssets(); } }, checkUpdate: function () { if (this._updating) { this.panel.info.string = 'Checking or updating ...'; return; } if (this._am.getState() === jsb.AssetsManager.State.UNINITED) { // Resolve md5 url var url = this.manifestUrl.nativeUrl; if (cc.loader.md5Pipe) { url = cc.loader.md5Pipe.transformURL(url); } this._am.loadLocalManifest(url); } if (!this._am.getLocalManifest() || !this._am.getLocalManifest().isLoaded()) { this.panel.info.string = 'Failed to load local manifest ...'; return; } this._am.setEventCallback(this.checkCb.bind(this)); this._am.checkUpdate(); this._updating = true; }, hotUpdate: function () { if (this._am && !this._updating) { this._am.setEventCallback(this.updateCb.bind(this)); if (this._am.getState() === jsb.AssetsManager.State.UNINITED) { // Resolve md5 url var url = this.manifestUrl.nativeUrl; if (cc.loader.md5Pipe) { url = cc.loader.md5Pipe.transformURL(url); } this._am.loadLocalManifest(url); } this._failCount = 0; this._am.update(); this.panel.updateBtn.active = false; this._updating = true; } }, show: function () { if (this.updateUI.active === false) { this.updateUI.active = true; } }, // use this for initialization onLoad: function () { // Hot update is only available in Native build if (!cc.sys.isNative) { return; } this._storagePath = ((jsb.fileUtils ? jsb.fileUtils.getWritablePath() : '/') + 'blackjack-remote-asset'); cc.log('Storage path for remote asset : ' + this._storagePath); // Setup your own version compare handler, versionA and B is versions in string // if the return value greater than 0, versionA is greater than B, // if the return value equals 0, versionA equals to B, // if the return value smaller than 0, versionA is smaller than B. this.versionCompareHandle = function (versionA, versionB) { cc.log("JS Custom Version Compare: version A is " + versionA + ', version B is ' + versionB); var vA = versionA.split('.'); var vB = versionB.split('.'); for (var i = 0; i < vA.length; ++i) { var a = parseInt(vA[i]); var b = parseInt(vB[i] || 0); if (a === b) { continue; } else { return a - b; } } if (vB.length > vA.length) { return -1; } else { return 0; } }; // Init with empty manifest url for testing custom manifest this._am = new jsb.AssetsManager('', this._storagePath, this.versionCompareHandle); var panel = this.panel; // Setup the verification callback, but we don't have md5 check function yet, so only print some message // Return true if the verification passed, otherwise return false this._am.setVerifyCallback(function (path, asset) { // When asset is compressed, we don't need to check its md5, because zip file have been deleted. var compressed = asset.compressed; // Retrieve the correct md5 value. var expectedMD5 = asset.md5; // asset.path is relative path and path is absolute. var relativePath = asset.path; // The size of asset file, but this value could be absent. var size = asset.size; if (compressed) { panel.info.string = "Verification passed : " + relativePath; return true; } else { panel.info.string = "Verification passed : " + relativePath + ' (' + expectedMD5 + ')'; return true; } }); this.panel.info.string = 'Hot update is ready, please check or directly update.'; 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.panel.info.string = "Max concurrent tasks count have been limited to 2"; } this.panel.fileProgress.progress = 0; this.panel.byteProgress.progress = 0; }, onDestroy: function () { if (this._updateListener) { this._am.setEventCallback(null); this._updateListener = null; } } });
注意:
var UpdatePanel = require('./UpdatePanel'); 这个require的相对路径,根据项目路径自己调整
UpdatPanel.js
module.exports = cc.Class({ extends: cc.Component, properties: { info: cc.Label, fileProgress: cc.ProgressBar, fileLabel: cc.Label, byteProgress: cc.ProgressBar, byteLabel: cc.Label, close: cc.Node, checkBtn: cc.Node, retryBtn: cc.Node, updateBtn: cc.Node }, onLoad () { this.close.on(cc.Node.EventType.TOUCH_END, function () { cc.director.loadScene("helloworld"); }, this); } });
提示:拖到update_panel节点上后,对应属性参考官方demo,下面截个图
这是一个用于生成 Manfiest 文件的 NodeJS 脚本
使用样例:node version_generator.js -v 2.2.0 -u http://192.168.1.103:8081/remote-assets/ -s build/jsb-link/ -d assets/
-v 2.2.0 版本号
-u http://192.168.1.103:8081/remote-assets/ 远程地址
-s build/jsb-link/ 源文件路径build/jsb-link/或build/jsb-default/
-d assets/ 生成后目标路径
官方说明如下:(链接:version_generator.js 文件)
-v
指定 Manifest 文件的主版本号。-u
指定服务器远程包的地址,这个地址需要和最初发布版本中 Manifest 文件的远程包地址一致,否则无法检测到更新。-s
本地原生打包版本的目录相对路径。-d
保存 Manifest 文件的地址。
version_generator.js
var fs = require('fs'); var path = require('path'); var crypto = require('crypto'); var manifest = { packageUrl: 'http://localhost/tutorial-hot-update/remote-assets/', remoteManifestUrl: 'http://localhost/tutorial-hot-update/remote-assets/project.manifest', remoteVersionUrl: 'http://localhost/tutorial-hot-update/remote-assets/version.manifest', version: '1.0.0', assets: {}, searchPaths: [] }; var dest = './remote-assets/'; var src = './jsb/'; // Parse arguments var i = 2; while ( i < process.argv.length) { var arg = process.argv[i]; switch (arg) { case '--url' : case '-u' : var url = process.argv[i+1]; manifest.packageUrl = url; manifest.remoteManifestUrl = url + 'project.manifest'; manifest.remoteVersionUrl = url + 'version.manifest'; i += 2; break; case '--version' : case '-v' : manifest.version = process.argv[i+1]; i += 2; break; case '--src' : case '-s' : src = process.argv[i+1]; i += 2; break; case '--dest' : case '-d' : dest = process.argv[i+1]; i += 2; break; default : i++; break; } } function readDir (dir, obj) { var stat = fs.statSync(dir); if (!stat.isDirectory()) { return; } var subpaths = fs.readdirSync(dir), subpath, size, md5, compressed, relative; for (var i = 0; i < subpaths.length; ++i) { if (subpaths[i][0] === '.') { continue; } subpath = path.join(dir, subpaths[i]); stat = fs.statSync(subpath); if (stat.isDirectory()) { readDir(subpath, obj); } else if (stat.isFile()) { // Size in Bytes size = stat['size']; md5 = crypto.createHash('md5').update(fs.readFileSync(subpath, 'binary')).digest('hex'); compressed = path.extname(subpath).toLowerCase() === '.zip'; relative = path.relative(src, subpath); relative = relative.replace(/\\/g, '/'); relative = encodeURI(relative); obj[relative] = { 'size' : size, 'md5' : md5 }; if (compressed) { obj[relative].compressed = true; } } } } var mkdirSync = function (path) { try { fs.mkdirSync(path); } catch(e) { if ( e.code != 'EEXIST' ) throw e; } } // Iterate res and src folder readDir(path.join(src, 'src'), manifest.assets); readDir(path.join(src, 'res'), manifest.assets); var destManifest = path.join(dest, 'project.manifest'); var destVersion = path.join(dest, 'version.manifest'); mkdirSync(dest); fs.writeFile(destManifest, JSON.stringify(manifest), (err) => { if (err) throw err; console.log('Manifest successfully generated'); }); delete manifest.assets; delete manifest.searchPaths; fs.writeFile(destVersion, JSON.stringify(manifest), (err) => { if (err) throw err; console.log('Version successfully generated'); });
启本地服务器:这里使用NodeJS
1. 新建一个文件夹nodejs
2. 在nodejs中新建hotUpdate文件夹
3. 把构建后的res和src复制到hotUpdate文件夹中
4. 在nodejs文件夹中新建一个app.js脚本
var express = require('express'); var path = require('path'); var app = express(); app.use(express.static(path.join(__dirname, 'hotUpdate'))); app.listen(8081); // 端口自己起一个不常用的,以免被占用
5. 命令行进入到nodejs文件夹下,执行node app.js命令,启动服务器,
6. 可以访问http://192.168.1.103:8081/remote-assets/project.manifest,如果成功访问则服务器启动成功。
注意:192.168.1.103:8081 根据自己的本机地址更改,可以命令行ifconfig查看,也可以在这里查看
整理操作流程:
1、选择ios或者android平台构建
2、构建完成后拷贝官方的version_generator.js到项目根目录下(和assets同级)(后者使用上面贴的version_generator.js代码)
3、命令行进到version_generator.js目录,执行node version_generator.js -v 1.0.0 -u http://192.168.1.103:8081/remote-assets/ -s build/jsb-link/ -d assets/
4、在项目assets目录下会生成两个文件:project.manifest 和 version.manifest
5、回到Creator把Canvas节点上的脚本ManifestUrl属性关联上生成的project.manifest文件
6、xcode打开项目,运行到ios设备上
7、接下来制作新版本(hotUpdate页面左上角按钮会跳helloworld页面,我们再helloworld页面做点改动)
8、helloworld页面改动后,重新构建(构建属性不要调整)
9、把构建好的(在build/jsb-link/下)res和src拷贝到 启服务器的路径下(nodejs/hotUpdate/remote-assets/)
10、再一次执行:命令行进到version_generator.js目录,执行node version_generator.js -v 1.0.1 -u http://192.168.1.103:8081/remote-assets/ -s build/jsb-link/ -d assets/
11、在项目assets目录下会生成两个文件:project.manifest 和 version.manifest
12、将这两个文件同样拷贝到 启服务器的路径下(nodejs/hotUpdate/remote-assets/),目前目录结构如下
13、开始测试热更新,ios设备启动进入原来运行的app,点击检测更新,正常会提示有版本可以更新,然后点击更新,就能更新了,然后点击左上角按钮进入helloworld,查看改成所做修改
这个过程中会遇到的坑:ip不对,端口被占用,官方demo运行不了,版本混乱。。。