关于mui的热更新。

    热更新,也就是打开软件的时候软件自动判断是否需要更新,区别于在应用市场上更新。最近项目上用到了mui的移动app,需要自动更新。特地去查看了自带的示例,发现在Hello H5+和Hello mui中有2种不同的检查更新方式。其中mui的比较简单,H5+的比较复杂。 官方给出的文档讲解的很少,所以自己便开始研究了。
    首先mui中的update.js很简单。自带的主要代码如下:
var server = ""; //获取升级描述文件服务器地址,官方的网址为:http://www.dcloud.io/check/update,内容为:
/*官方给出的服务端地址为http://www.dcloud.io/check/update;
  内容模板为:{
{ "appid":"H5291D269", "iOS":{ "version":"1.2.0","title":"更新标题","note":"更新内容描述","url":"地址" },"Android":{ "version":"2.4.0", "title":"Hello MUI版本更新", "note": "修复轮播组件和原生滚动冲突的bug;\n修复列表控件不显示上边框的bug;","url":"软件地址" } }
}*/


function update() {mui.getJSON(server, {"appid": plus.runtime.appid, //此为项目的id,"version": plus.runtime.version,//项目版本号"imei": plus.device.imei//移动设备身份码}, function(data) {if (data.status) {plus.nativeUI.confirm(data.note, function(event) {if (0 == event.index) {plus.runtime.openURL(data.url);}}, data.title, ["立即更新", "取  消"]);}});}
    按照官方给的示例是在服务端判断是否需要更新。很简单,但是一定要确保appid保持一致。示例中是直接获取的appid,在调试过程中(并未打包)appid默认为HBuilder。和服务端的不一致,所以一致不行。
  另外一种H5+比较复杂,是在js里面判断是否需要更新,同时可操作性更强。完整代码如下:
/**
 * 判断应用升级模块,从url地址下载升级描述文件到本地local路径
 * [email protected]
 * 
 * 升级文件为JSON格式数据,如下:
{
	"appid":"HelloH5",
    "iOS":{
    	"version":"iOS新版本号,如:1.0.0",
    	"note":"iOS新版本描述信息,多行使用\n分割",
    	"url":"Appstore路径,如:itms-apps://itunes.apple.com/cn/app/hello-h5+/id682211190?l=zh&mt=8"
    },
    "Android":{
    	"version":"Android新版本号,如:1.0.1",
    	"note":"Android新版本描述信息,多行使用\n分割",
    	"url":"apk文件下载地址,如:http://www.dcloud.io/helloh5p/HelloH5.apk"
    }
}
 *
 */
(function(w){
var server="http://www.dcloud.io/helloh5/update.json",//获取升级描述文件服务器地址
localDir="update",localFile="update.json",//本地保存升级描述目录和文件名
keyUpdate="updateCheck",//取消升级键名
keyAbort="updateAbort",//忽略版本键名
checkInterval=604800000,//升级检查间隔,单位为ms,7天为7*24*60*60*1000=604800000, 如果每次启动需要检查设置值为0
dir=null;

/**
 * 准备升级操作
 * 创建升级文件保存目录
 */
function initUpdate(){
	// 在流应用模式下不需要检测升级操作
	if(navigator.userAgent.indexOf('StreamApp')>=0){
		return;
	}
	// 打开doc根目录
	plus.io.requestFileSystem( plus.io.PRIVATE_DOC, function(fs){
		fs.root.getDirectory( localDir, {create:true}, function(entry){
			dir = entry;
			checkUpdate();
		}, function(e){
			console.log( "准备升级操作,打开update目录失败:"+e.message );
		});
	},function(e){
		console.log( "准备升级操作,打开doc目录失败:"+e.message );
	});
}

/**
 * 检测程序升级
 */
function checkUpdate() {
	// 判断升级检测是否过期
	var lastcheck = plus.storage.getItem( keyUpdate );
	if ( lastcheck ) {
		var dc = parseInt( lastcheck );
		var dn = (new Date()).getTime();
		if ( dn-dc < checkInterval ) {	// 未超过上次升级检测间隔,不需要进行升级检查
			return;
		}
		// 取消已过期,删除取消标记
		plus.storage.removeItem( keyUpdate );
	}
	// 读取本地升级文件
	dir.getFile( localFile, {create:false}, function(fentry){
		fentry.file( function(file){
			var reader = new plus.io.FileReader();
			reader.onloadend = function ( e ) {
				fentry.remove();
				var data = null;
				try{
					data=JSON.parse(e.target.result);
				}catch(e){
					console.log( "读取本地升级文件,数据格式错误!" );
					return;
				}
				checkUpdateData( data );
			}
			reader.readAsText(file);
		}, function(e){
			console.log( "读取本地升级文件,获取文件对象失败:"+e.message );
			fentry.remove();
		} );
	}, function(e){
		// 失败表示文件不存在,从服务器获取升级数据
		getUpdateData();
	});
}

/**
 * 检查升级数据
 */
function checkUpdateData( j ){
	var curVer=plus.runtime.version,
	inf = j[plus.os.name];//判断软件是Android还是ios
	if ( inf ){
		var srvVer = inf.version;
		// 判断是否存在忽略版本号
		var vabort = plus.storage.getItem( keyAbort );
		if ( vabort && srvVer==vabort ) {
			// 忽略此版本
			return;
		}
		// 判断是否需要升级
		if ( compareVersion(curVer,srvVer) ) {
			// 提示用户是否升级
			plus.nativeUI.confirm( inf.note, function(i){
				if ( 0==i.index ) {
					plus.runtime.openURL( inf.url );
				} else if ( 1==i.index ) {
					plus.storage.setItem( keyAbort, srvVer );
					plus.storage.setItem( keyUpdate, (new Date()).getTime().toString() );
				} else {
					plus.storage.setItem( keyUpdate, (new Date()).getTime().toString() );
				}
			}, inf.title, ["立即更新","跳过此版本","取  消"] );
		}
	}
}

/**
 * 从服务器获取升级数据
 */
function getUpdateData(){
	var xhr = new plus.net.XMLHttpRequest();
	xhr.onreadystatechange = function () {
        switch ( xhr.readyState ) {
            case 4:
                if ( xhr.status == 200 ) {
                	// 保存到本地文件中
                	dir.getFile( localFile, {create:true}, function(fentry){
                		fentry.createWriter( function(writer){
                			writer.onerror = function(){
                				console.log( "获取升级数据,保存文件失败!" );
                			}
                			writer.write( xhr.responseText );
                		}, function(e){
                			console.log( "获取升级数据,创建写文件对象失败:"+e.message );
                		} );
                	}, function(e){
                		console.log( "获取升级数据,打开保存文件失败:"+e.message );
                	});
                } else {
                	console.log( "获取升级数据,联网请求失败:"+xhr.status );
                }
                break;
            default :
                break;
        }
	}
	xhr.open( "GET", server );
	xhr.send();
}

/**
 * 比较版本大小,如果新版本nv大于旧版本ov则返回true,否则返回false
 * @param {String} ov
 * @param {String} nv
 * @return {Boolean} 
 */
function compareVersion( ov, nv ){
	if ( !ov || !nv || ov=="" || nv=="" ){
		return false;
	}
	var b=false,
	ova = ov.split(".",4),
	nva = nv.split(".",4);
	for ( var i=0; ino || sn.length>so.length  ) {
			return true;
		} else if ( nnova.length && 0==nv.indexOf(ov) ) {
		return true;
	}
}

if ( w.plus ) {
	initUpdate();
} else {
	document.addEventListener("plusready", initUpdate, false );
}

})(window);

    下面来看代码, 先是定义一些列参数,可以看到除了服务端地址,还多了本地保存的地址以及是否忽略本次更新,以及提醒的间隔。
var server="http://www.dcloud.io/helloh5/update.json",//获取升级描述文件服务器地址
localDir="update",localFile="update.json",//本地保存升级描述目录和文件名
keyUpdate="updateCheck",//取消升级键名
keyAbort="updateAbort",//忽略版本键名
checkInterval=604800000,//升级检查间隔,单位为ms,7天为7*24*60*60*1000=604800000, 如果每次启动需要检查设置值为0
dir=null;
之后是创建本地文件,存放获取的update.json。之后打开会首先通过initUpdate()和checkUpdate方法查打开此文件,如果没有的话通过getUpdateData()方法请求服务端获取数据保存到本地;如果本地有数据,则会依次执行initUpdate()>>checkUpdate()>>checkUpdateData()>>compareVersion()。在compareVersion()判断是否需要更新,如要则在checkUpdateData()中打开下载链接进行软件下载。

你可能感兴趣的:(关于mui的热更新。)