unity3d 自己主动文件更新系统
游戏内容变更之后。一般而言不会想让玩家下载整个游戏包又一次安装,由于这样会流失大量玩家。全部游戏更新是必须的。
更新的内容包含 数据、资源、代码。
基本原理:
1、将须要更新的文件打包成AssetBundle文件,并计算各个文件的crc值。
以下代码将选择的文件分别导出为AssetBundle文件,并将每一个文件的crc值写入到crc.txt文件里,在Editor文件夹建立一个类,并复制以下代码。
能够在Project文件夹导出选择的文件。
public class ExportAssetBundles { [MenuItem("Assets/Build AssetBundle From Selection Respective - Track dependencies")] static void ExportResourceRespective() { // Bring up save panel string path = EditorUtility.SaveFilePanel ("Save Resource", "", "New Resource", "unity3d"); string dirpath = path.Substring(0,path.LastIndexOf("/")+1); string filename = path.Substring(path.LastIndexOf("/")+1); if (path.Length != 0) { #if UNITY_ANDROID string targetDir = "Android/"; BuildTarget targetBuild = BuildTarget.Android; #elif UNITY_IPHONE string targetDir = "iPhone/"; BuildTarget targetBuild = BuildTarget.iPhone; #elif UNITY_STANDALONE_WIN string targetDir = "StandaloneWindows/"; BuildTarget targetBuild = BuildTarget.StandaloneWindows; #endif JSDocument.JSNode node = new JSDocument.JSNode("root"); Document.SNode[] nodes = node.putChildren("filehash",Selection.objects.Length); for(int i=0;i<Selection.objects.Length;i++) { string name = Selection.objects[i].name+".unity3d"; uint crc = 0; if(!Directory.Exists(dirpath+targetDir)) Directory.CreateDirectory(dirpath+targetDir); BuildPipeline.BuildAssetBundle(Selection.objects[i],new Object[]{ Selection.objects[i]}, dirpath+targetDir+name,out crc,BuildAssetBundleOptions.CollectDependencies,targetBuild); nodes[i].put("name",name); nodes[i].put("crc",crc); } System.IO.File.WriteAllText(dirpath+targetDir+filename+".crc.txt",node.toJSONString()); } } } |
2.须要一个资源更新server,将导出的AssetBundle文件和crc文件上传到资源更新server。能够用一个简单的httpserver。比如nginx。
3.client进入游戏之前。首先向更新server请求crc.txt文件。然后从本地的磁盘文件夹中查找crc.txt文件,检查须要更新的文件列表。然后从server下载须要更新的文件。这样,假设server没有更改文件,则仅仅须要下载一次。
4.最新的crc.txt文件到本地,以便下次查询。
以下代码演示 3,4 步骤,当中
Engine.Instance.server_datapath = 服务器下载地址。
Engine.Instance.local_datapath = Application.persistentDataPath+"/;
public static IEnumerator UpdateDataFromServer(UpdateProgress up) { string server_datapath = Engine.Instance.server_datapath; string local_datapath = Engine.Instance.local_datapath; byte[] server_crc_data = null; Dictionary<string,long> filehash_server=new Dictionary<string, long>(); Dictionary<string,long> filehash_local=new Dictionary<string, long>(); List<string> needUpdateFile = new List<string>(); Debug.Log("Load Server FileHash"); using(WWW www = new WWW(server_datapath+"crc.txt")) { yield return www; if (!String.IsNullOrEmpty(www.error)) { Debug.Log("Load Filehash Failed"); up(1.0f); yield break; } JSDocument.JSNode node = new JSDocument.JSNode("filehash",www.text); Document.SNode[] data = node.getChildren("filehash"); for(int i=0;i<data.Length;i++) { filehash_server[data[i].get("name","")] = data[i].get("crc",(long)0); } server_crc_data = www.bytes; } Debug.Log("Load Local FileHash"); //从本地载入文件MD5表。可能没有 try { JSDocument.JSNode node = new JSDocument.JSNode("filehash",System.IO.File.ReadAllText(local_datapath+"crc.txt")); Document.SNode[] data = node.getChildren("filehash"); for(int i=0;i<data.Length;i++) { filehash_local[data[i].get("name","")] = data[i].get("crc",(long)0); } } catch(Exception e) { Debug.Log(e.Message); } Debug.Log("Check FileHash"); //计算须要更新的文件 foreach(KeyValuePair<string,long> data in filehash_server) { //更新须要的文件 if(!filehash_local.ContainsKey(data.Key) || filehash_local[data.Key] != data.Value) { needUpdateFile.Add(data.Key); } } Debug.Log("Update File"); //下载并存储 for(int i=0;i<needUpdateFile.Count;i++) { using(WWW www = new WWW(server_datapath+needUpdateFile[i])) { yield return www; byte[] bytes = null; if (!String.IsNullOrEmpty(www.error)) { Debug.Log(www.error); yield break; } else { bytes = www.bytes; } up((float)i/needUpdateFile.Count); string path = local_datapath+ needUpdateFile[i]; Debug.Log(path); FileStream fs = new FileStream(path,FileMode.Create); fs.Write(bytes,0,bytes.Length); fs.Flush(); fs.Close(); // //保存 // BinaryWriter writer; // FileInfo t =new FileInfo(local_datapath+ needUpdateFile[i]); // if(!t.Exists) // { // writer = new BinaryWriter(t.Open(FileMode.OpenOrCreate)); // } // else // { // t.Delete(); // writer = new BinaryWriter(t.Open(FileMode.Create)); // } // writer.Write(bytes); // writer.Close(); } } Debug.Log("Save FileHash"); if(needUpdateFile.Count>0) { //保存最新的文件MD5值表 // FileStream fs = new FileStream(local_datapath+"crc.txt",FileMode.Create); // fs.Write(server_crc_data,0,server_crc_data.Length); // fs.Flush(); // fs.Close(); BinaryWriter writer; FileInfo t =new FileInfo(local_datapath+"crc.txt"); if(!t.Exists) { writer = new BinaryWriter(t.Open(FileMode.Create)); } else { t.Delete(); writer = new BinaryWriter(t.Open(FileMode.Create)); } writer.Write(server_crc_data); writer.Close(); Debug.Log(local_datapath+"crc.txt"); } } |
5.载入已经更新完毕的存储在本地的AssetBundle,须要注意的是。Unity3d同样的文件同一时候仅仅能有一个AssetBundle在内存中,所以我们对于同样文件的载入做了同步。
(loadRefCount)。
static HashSet<string> loadRefCount = new HashSet<string>(); public delegate void delegateLoadFinish(GameObject go); public static IEnumerator LoadModel(string res,delegateLoadFinish onLoadFinish) { GameObject resObject = Resources.Load<GameObject>("cards/"+res); if(resObject !=null) { onLoadFinish((GameObject)GameObject.Instantiate(resObject)); yield break; } //假设包括资源。返回 while(loadRefCount.Contains(res)) { yield return true; } loadRefCount.Add(res); Debug.Log(Engine.Instance.local_datapath+res+".unity3d"); AssetBundleCreateRequest crcLocalBundle = AssetBundle.CreateFromMemory( System.IO.File.ReadAllBytes(Engine.Instance.local_datapath+res+".unity3d")); yield return crcLocalBundle; { GameObject cardObject = GameObject.Instantiate(crcLocalBundle.assetBundle.mainAsset)as GameObject; crcLocalBundle.assetBundle.Unload(false); onLoadFinish(cardObject); } loadRefCount.Remove(res); } |
unity3d 自己主动文件更新系统的更多相关文章
- 艺萌文件上传下载及自动更新系统(基于networkComms开源TCP通信框架)
1.艺萌文件上传下载及自动更新系统,基于Winform技术,采用CS架构,开发工具为vs2010,.net2.0版本(可以很容易升级为3.5和4.0版本)开发语言c#. 本系统主要帮助客户学习基于TC ...
- 制作SD更新系统时和用mfgtool工具烧录时,文件如何替换?
制作SD更新系统时和用mfgtool工具烧录时,文件如何替换? 答:制作SD更新系统时,请按照需求选择不同mfgimages-myd*文件夹.每个文件夹里面有一个Manifest文件, 里面规定了ub ...
- CentOS更改yum源与更新系统
[1] 首先备份/etc/yum.repos.d/CentOS-Base.repo mv /etc/yum.repos.d/CentOS-Base.repo /etc/yum.repos.d/Cent ...
- CentOS 更改yum源与更新系统
FROM:http://www.cnblogs.com/lightnear/archive/2012/10/03/2710952.html [1] 首先备份/etc/yum.repos.d/CentO ...
- [转]CentOS更改yum源与更新系统
[1] 首先备份/etc/yum.repos.d/CentOS-Base.repo mv /etc/yum.repos.d/CentOS-Base.repo /etc/yum.repos.d/Cent ...
- 引擎设计跟踪(九.9) 文件包系统(Game Package System)
很早之前,闪现过写文件包系统的想法, 但是觉得还没有到时候. 由于目前工作上在做android ndk开发, 所以业余时间趁热做了android的移植, 因为android ndk提供的mountab ...
- CentOS 7更改yum源与更新系统
在CentOS 7下更改yum源与更新系统. [1] 首先备份/etc/yum.repos.d/CentOS-Base.repo cp /etc/yum.repos.d/CentOS-Base.rep ...
- MAC EI Capitan上更新系统自带SVN版本号(关闭SIP方能sudo rm)
继昨晚之后.决定更新系统自带的svn.自带的svn版本号是1.7.看官网svn:http://www.wandisco.com/subversion/download#osx 最新版本号是1.9.13 ...
- expect脚本同步文件 expect脚本指定host和要同步的文件 构建文件分发系统 批量远程执行命令
自动同步文件 #!/usr/bin/expect set " spawn rsync -av root@.txt /tmp/ expect { "yes/no" { se ...
随机推荐
- u-boot启动代码start.S详解
(1)定义入口.由于一个可执行的Image必须有一个入口点,并且只能有一个全局入口,通常这个入口放在ROM(Flash)的0x0地址,因此,必须通知编译器以使其知道这个入口,该工作可通过修改连接器脚本 ...
- spring mvc 下载
1.代码: response.setContentType("application/msword;charset=GBK"); try { response.setHeader( ...
- NOIP2018提高组省一冲奖班模测训练(三)
NOIP2018提高组省一冲奖班模测训练(三) 自己按照noip的方式考,只在最后一两分钟交了一次 第一题过了,对拍拍到尾. 第二题不会.考试时往组合计数的方向想,推公式,推了一个多小时,大脑爆炸,还 ...
- 阿里云Linux系统安装配置Tomcat方法
本文将tomcat安装到了/alidata/server/目录下,当然也可以安装到其他目录. 1. 下载tomcat:#wget http://apache.fayea.com/tomcat/tomc ...
- js实现鼠标吸附线条效果
如图,箭头→为鼠标位置,鼠标会带有吸附着一些线条的效果,具体效果可在我的博客查看,当然,这也是可很受欢迎很常见的效果了=3= <script> !function(){ function ...
- oracle 表空间Tablespaces
1.表空间 一个数据库可以有多个表空间,一个表空间里可以有多个表.表空间就是存多个表的物理空间: 可以指定表空间的大小位置等. 创建表空间:create tablespace ts1 datafile ...
- Myeclipse学习总结(9)——MyEclipse2014安装插件的几种方式(适用于Eclipse或MyEclipse其他版本)
众所周知MyEclipse是一个很强大的Java IDE,而且它有许多开源免费又好用的插件,这些插件给我们开发过程中带来了许多方便.插件具有针对性,例如,你如果做安卓开发,可能需要一个ADT(Andr ...
- tddl
淘宝根据自己的业务特点开发了TDDL(Taobao Distributed Data Layer 外号:头都大了 ?_Ob)框架,主要解决了分库分表对应用的透明化以及异构数据库之间的数据复制,它是一个 ...
- hdu2838Cow Sorting(树状数组+逆序数)
题目链接:点击打开链接 题意描写叙述:给定一个长度为100000的数组,每一个元素范围在1~100000,且互不同样,交换当中的随意两个数须要花费的代价为两个数之和. 问怎样交换使数组有序.花费的代价 ...
- Hadoop解析--MapReduce
从本篇博客開始咱们一起来具体了解Hadoop的每一个部分.我们在上篇博客中介绍了HDFS,MapReduce,MapReduce为了更有效率事实上是建立在HDFS之上的.有了分布式的文件系统,我们就能 ...