[unity3d]手游资源热更新策略探讨

原地址:http://blog.csdn.net/dingxiaowei2013/article/details/20079683

我们学习了如何将资源进行打包。这次就可以用上场了,我们来探讨一下手游资源的增量更新策略。注意哦,只是资源哦。关于代码的更新,我们稍后再来研究。理论上这个方案可以使用各种静态资源的更新,不仅仅是assetbundle打包的。

(转载请注明原文地址http://blog.csdn.net/janeky/article/details/17666409)

  • 原理

现在的手游安装有几种方式。一种是安装的时候就把程序和资源安装到本地。另外一种是只安装程序和少量的必要资源,然后在启动的时候再把缺少的资源下载完整。手游一般不建议和传统页游一样,在运行过程中加载资源,那样做会导致用户体验会比较差些。上述的两种安装模式,在更新资源上本质都是相同的。都是比较服务器资源的版本和本地资源的版本,以确定哪些资源要下载(包括需要更新的和新增的)。

  • 实践

        1.资源打包。
资源打包之前,要先规划好资源之间的相互依赖关系。把一些共性的东西抽取出来,尽量减少不必要的耦合。一些比较好的做法有,所有物件尽可能做成Prefab,场景上的东西越少越好,“一切都是动态加载”。
        2.生成文件MD5
关于文件的MD5,这里就不详细描述了。大家可以简单理解它为一个文件的状态标记。如果文件有更改,那么它的md5一定是改变的,单纯的移动文件是不会更改的。md5验证还可以起到安全验证的作用,保证本地文件不被篡改。举个例子,我们经常从网上上下载软件时,一般都会给出一个md5值,你下载后,对比一下已下载文件的md5值,就可以知道文件有没有被篡改。在版本发布时,我们需要对所有打包好的文件计算md5值,然后保存在一个配置文件中。关于这部分的工作,我之前写过一个可视化小工具(https://github.com/kenro/File_Md5_Generator),现在分享给大家。如果大家觉得有用,记得打星哦:)
        3.版本比较
先加载本地的version.txt,将结果缓存起来。下载服务器的version.txt,与本地的version进行比较,筛选出需要更新和新增的资源
        4.下载资源
依次下载更新的资源,如果本地已经有旧资源,则替换之,否则就新建保存起来

        5.更新本地版本配置文件version.txt

用服务器的version.txt替换掉本地的version.txt。这样做是为了确保下次启动的时候,不会再重复更新了。

        6.从本地加载assetbundle进行测试显示。

这里将一个模型制成Prefab,打包成assetbundle。程序从本地加载后,显示在场景中

        7.更新服务器的assetbundle,重新生成版本号文件。

        8.重复6的步骤

我们可以验证,我们的程序不用任何改动,资源已经实现了更新。场景中显示的已经是最新的模型了。

 

关于上述的流程,我写了一个小的演示demo。我这里没有用到web服务器,而是将本地的另外一个文件夹作为资源服务器目录。这里的目录只针对windows下的版本进行测试。如果要在手机平台上,需要记得更新相关的路径。

[csharp] view plaincopy在CODE上查看代码片派生到我的代码片
  1. using UnityEngine;  
  2. using System.Collections;  
  3. using System.Collections.Generic;  
  4. using System.Text;  
  5. using System.IO;  
  6.   
  7. public class ResUpdate : MonoBehaviour  
  8. {  
  9.     public static readonly string VERSION_FILE = "version.txt";  
  10.     public static readonly string LOCAL_RES_URL = "file://" + Application.dataPath + "/Res/";  
  11.     public static readonly string SERVER_RES_URL = "file:///C:/Res/";  
  12.     public static readonly string LOCAL_RES_PATH = Application.dataPath + "/Res/";  
  13.   
  14.     private Dictionary<string, string> LocalResVersion;  
  15.     private Dictionary<string, string> ServerResVersion;  
  16.     private List<string> NeedDownFiles;  
  17.     private bool NeedUpdateLocalVersionFile = false;  
  18.   
  19.     void Start()  
  20.     {  
  21.         //初始化  
  22.         LocalResVersion = new Dictionary<string, string>();  
  23.         ServerResVersion = new Dictionary<string, string>();  
  24.         NeedDownFiles = new List<string>();  
  25.   
  26.         //加载本地version配置  
  27.         StartCoroutine(DownLoad(LOCAL_RES_URL + VERSION_FILE, delegate(WWW localVersion)  
  28.         {  
  29.             //保存本地的version  
  30.             ParseVersionFile(localVersion.text, LocalResVersion);  
  31.             //加载服务端version配置  
  32.             StartCoroutine(this.DownLoad(SERVER_RES_URL + VERSION_FILE, delegate(WWW serverVersion)  
  33.             {  
  34.                 //保存服务端version  
  35.                 ParseVersionFile(serverVersion.text, ServerResVersion);  
  36.                 //计算出需要重新加载的资源  
  37.                 CompareVersion();  
  38.                 //加载需要更新的资源  
  39.                 DownLoadRes();  
  40.             }));  
  41.   
  42.         }));  
  43.     }  
  44.   
  45.     //依次加载需要更新的资源  
  46.     private void DownLoadRes()  
  47.     {  
  48.         if (NeedDownFiles.Count == 0)  
  49.         {  
  50.             UpdateLocalVersionFile();  
  51.             return;  
  52.         }  
  53.   
  54.         string file = NeedDownFiles[0];  
  55.         NeedDownFiles.RemoveAt(0);  
  56.   
  57.         StartCoroutine(this.DownLoad(SERVER_RES_URL + file, delegate(WWW w)  
  58.         {  
  59.             //将下载的资源替换本地就的资源  
  60.             ReplaceLocalRes(file, w.bytes);  
  61.             DownLoadRes();  
  62.         }));  
  63.     }  
  64.   
  65.     private void ReplaceLocalRes(string fileName, byte[] data)  
  66.     {  
  67.         string filePath = LOCAL_RES_PATH + fileName;  
  68.         FileStream stream = new FileStream(LOCAL_RES_PATH + fileName, FileMode.Create);  
  69.         stream.Write(data, 0, data.Length);  
  70.         stream.Flush();  
  71.         stream.Close();  
  72.     }  
  73.   
  74.     //显示资源  
  75.     private IEnumerator Show()  
  76.     {  
  77.         WWW asset = new WWW(LOCAL_RES_URL + "cube.assetbundle");  
  78.         yield return asset;  
  79.         AssetBundle bundle = asset.assetBundle;  
  80.         Instantiate(bundle.Load("Cube"));  
  81.         bundle.Unload(false);  
  82.     }  
  83.   
  84.     //更新本地的version配置  
  85.     private void UpdateLocalVersionFile()  
  86.     {  
  87.         if (NeedUpdateLocalVersionFile)  
  88.         {  
  89.             StringBuilder versions = new StringBuilder();  
  90.             foreach (var item in ServerResVersion)  
  91.             {  
  92.                 versions.Append(item.Key).Append(",").Append(item.Value).Append("\n");  
  93.             }  
  94.   
  95.             FileStream stream = new FileStream(LOCAL_RES_PATH + VERSION_FILE, FileMode.Create);  
  96.             byte[] data = Encoding.UTF8.GetBytes(versions.ToString());  
  97.             stream.Write(data, 0, data.Length);  
  98.             stream.Flush();  
  99.             stream.Close();  
  100.         }  
  101.         //加载显示对象  
  102.         StartCoroutine(Show());  
  103.     }  
  104.   
  105.     private void CompareVersion()  
  106.     {  
  107.         foreach (var version in ServerResVersion)  
  108.         {  
  109.             string fileName = version.Key;  
  110.             string serverMd5 = version.Value;  
  111.             //新增的资源  
  112.             if (!LocalResVersion.ContainsKey(fileName))  
  113.             {  
  114.                 NeedDownFiles.Add(fileName);  
  115.             }  
  116.             else  
  117.             {  
  118.                 //需要替换的资源  
  119.                 string localMd5;  
  120.                 LocalResVersion.TryGetValue(fileName, out localMd5);  
  121.                 if (!serverMd5.Equals(localMd5))  
  122.                 {  
  123.                     NeedDownFiles.Add(fileName);  
  124.                 }  
  125.             }  
  126.         }  
  127.         //本次有更新,同时更新本地的version.txt  
  128.         NeedUpdateLocalVersionFile = NeedDownFiles.Count > 0;  
  129.     }  
  130.   
  131.     private void ParseVersionFile(string content, Dictionary<string, string> dict)  
  132.     {  
  133.         if (content == null || content.Length == 0)  
  134.         {  
  135.             return;  
  136.         }  
  137.         string[] items = content.Split(new char[] { '\n' });  
  138.         foreach (string item in items)  
  139.         {  
  140.             string[] info = item.Split(new char[] { ',' });  
  141.             if (info != null && info.Length == 2)  
  142.             {  
  143.                 dict.Add(info[0], info[1]);  
  144.             }  
  145.         }  
  146.   
  147.     }  
  148.   
  149.     private IEnumerator DownLoad(string url, HandleFinishDownload finishFun)  
  150.     {  
  151.         WWW www = new WWW(url);  
  152.         yield return www;  
  153.         if (finishFun != null)  
  154.         {  
  155.             finishFun(www);  
  156.         }  
  157.         www.Dispose();  
  158.     }  
  159.   
  160.     public delegate void HandleFinishDownload(WWW www);  
  161. }  

 

  • 总结

资源更新的原理,本质上都是相似的。我之前也从事过页游的开发,资源更新流程也类似。所以技术的本质是掌握思维方式,平台和语言都是永远在变的。我们最后归纳一下流程:比较服务端的资源版本和本地的资源版本,找出需要更新的资源,然后依次下载。如果大家有更好的策略,欢迎分享探讨 ken@iamcoding.com。

 

 

  • 源码

 

http://pan.baidu.com/s/1mgNnR8O

 

  • 参考资料

 

Unity3d官网文档

转载于:https://www.cnblogs.com/123ing/p/3823015.html

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

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

相关文章

PostgreSQL学习手册(二) 模式(Schema)

2019独角兽企业重金招聘Python工程师标准>>> 一个数据库包含一个或多个命名的模式&#xff0c;模式又包含表。模式还包含其它命名的对象&#xff0c;包括数据类型、函数&#xff0c;以及操作符。同一个对象名可以在不同的模式里使用而不会导致冲突&#xff1b; 比如…

软件工作第4次作业

软件工作第4次作业 信管141 宋乃佳 1425052010 基于我们列出的 7 条UX评价准则&#xff0c;分析“师路南通” 在用户体验设计方面让你觉得满意的地方&#xff08;不少于2点&#xff09;&#xff1b;&#xff08;20分&#xff09;&#xff0c;请陈述理由。 同样&#xff0c;分析…

phoneGap2.9+eclipse开发环境和helloword案例

不同机器安装和使用各不相同&#xff0c;这里也只是记录一下自己机器上面的使用过程。 android安装环境前面的文章有些&#xff0c;这里不再说&#xff0c;直接上phoneGap的过程。因为phoneGap2.9.1需要安装nodejs和Git&#xff0c;比较复杂&#xff0c;没有太多精力去折腾&…

Ubuntu14.04下搭建Bochs仿真平台,同时用该平台安装Linux0.11内核

因为Linux0.11内核需要在80X86硬件平台上运行&#xff0c;现在已经没有该硬件系统了&#xff0c;所以需要搭建Bochs这个仿真平台。Bochs是一个X86硬件平台的开源模拟器。 安装步骤参考的是如下一篇文章&#xff1a;http://os.51cto.com/art/201407/446838_all.htm&#xff0c;非…

java web与android互通的aes算法

2019独角兽企业重金招聘Python工程师标准>>> ####Java实现代码 //可自定义保证16btye即可private static final byte[] IV {16, 26, -35, 23, 34, 125, -5, -4, -8, -9, -15, -78, 90, -8, -99, 100};public static byte[] encrypt(String content, String passwor…

mysql第三方工具binlog_mysql 开发进阶篇系列 33 工具篇(mysqlbinlog日志管理工具)

一.概述由于服务器生成的二进制日志文件以二进制格式保存&#xff0c;所以如果要想检查这些文件的文本格式&#xff0c;就会用到mysqlbinlog日志管理工具。mysqlbinlog的语法如下:mysqlbinlog [options] log-files log-files2...其中options有很多选项&#xff0c;常用如下&…

JMeter部分功能详解

JMeter 介绍&#xff1a; 一个非常优秀的开源免费的性能测试工具。 优点&#xff1a;你用着用着就会发现它的重多优点&#xff0c;当然不足点也会呈现出来。 从性能工具的原理划分&#xff1a; Jmeter工具和其他性能工具在原理上完全一致&#xff0c;工具包含4个部分&#xff1…

登录时记住用户名和密码及cookie案例应用

文章原址&#xff1a;http://www.jb51.net/article/33588.htm 登录样子&#xff0c;可以参考某一论坛的登录介面&#xff1a; 记住这些信息&#xff0c;可以使用Cookie来实现&#xff0c;更多Cookie应用&#xff0c;可参考 http://jb51.net/article/33590.htm http://jb51.net…

退出登录后点返回键 是登录状态_看了这50条登录的测试点,你还敢说测试很容易吗...

条件&#xff1a;一个用户名输入框 (要求15个字符以内)一个密码输入框 (要求8个字符以内)一个登录按钮针对以上条件进行测试用例的设计先回顾一下测试用例的设计方法&#xff1a;等价类&#xff0c;边界值&#xff0c;错误猜测法&#xff0c;因果图&#xff0c;场景法测试功能点…

HTML5原生拖拽/拖放(drag drop)详解

前言 拖放&#xff08;drap && drop&#xff09;在我们平时的工作中&#xff0c;经常遇到。它表示&#xff1a;抓取对象以后拖放到另一个位置。目前&#xff0c;它是HTML5标准的一部分。我从几个方面学习并实践这个功能。 拖放的流程对应的事件 我们先看下拖放的流程&a…

Linux vmstat命令详解

vmstat命令是最常见的Linux/Unix监控工具&#xff0c;可以展现给定时间间隔的服务器的状态值,包括服务器的CPU使用率&#xff0c;内存使用&#xff0c;虚拟内存交换情况,IO读写情况。这个命令是我查看Linux/Unix最喜爱的命令&#xff0c;一个是Linux/Unix都支持&#xff0c;二是…

python中如何比较两个列表_python中如何比较两个列表

cmp() 方法用于比较两个列表的元素。cmp()方法语法&#xff1a;cmp(list1, list2)参数&#xff1a;list1 -- 比较的列表。list2 -- 比较的列表。返回值&#xff1a;如果比较的元素是同类型的,则比较其值,返回结果。如果两个元素不是同一种类型,则检查它们是否是数字。如果是数字…

rc mysql common_RR与RC隔离级别下MySQL不同的加锁解锁方式

作者 韩杰沃趣科技MySQL数据库工程师出品 沃趣科技| RC与RR隔离级别下MySQL不同的加锁解锁方式MySQL5.7.21数据准备rootlocalhost : pxs 05:26:27> show create table dots\G*************************** 1. row ***************************Table: dotsCreate Table: …

[Javascript_库编写]创建自己的“JavaScript库”

一.编写JavaScript库要注意的问题 为了让自己的JS库构建的更加优雅、合理&#xff0c;我们编写JS库时要注意两方面的内容&#xff1a; 1.不要使用版本检测&#xff0c;而要使用能力检测 由于浏览器的类型和版本太多&#xff0c;以及不断的新的浏览器出现&#xff0c;我们不可能…

mysql5.6.24怎么打开_mysql 5.6.24 安装配置方法图文教程

由于工作需要&#xff0c;开始使用mysql数据库&#xff0c;已经好久没有使用了。基本已经忘了差不多。今天重新安装配置了一下&#xff0c;写个随笔记录一下&#xff0c;以免自己以后需要的时候翻看&#xff0c;如有不正确或需要补充的&#xff0c;希望大家多多留言。首先下载m…

sublime text 3安装及使用

安装配置 安装&#xff1a; 安装环境:Ubuntu 16.04 官网下载sublime text 3 https://www.sublimetext.com/3 解压&#xff1a; tar xvf 文件名 进入 sublime_text_3文件夹&#xff0c;运行 ./sublime_text 注册码&#xff1a;我用的时候有用 —– BEGIN LICENSE —–TwitterInc…

微信公众号新功能-原创声明、赞赏功能、评论管理、页面模版

原文&#xff1a;http://www.shichangbu.com/portal.php?modview&aid25931 公众号如何申请这些功能&#xff1f; 在微信公众号后台和QQ经常都有人问我诸如此类的问题&#xff1a;”微信公众平台原创声明怎么开通?“、”微信赞赏等功能是怎么回事…

为什么闹钟设置了却不响_又被iPhone闹钟坑了?解决闹钟不响问题看这里!

原标题&#xff1a;又被iPhone闹钟坑了&#xff1f;解决闹钟不响问题看这里&#xff01;你有没有过这样的体验&#xff0c;早上醒来&#xff0c;闹钟还没响&#xff0c;内心想&#xff1a;真好&#xff0c;还可以再睡会。拿起手机想看看还能再睡多久&#xff0c;结果……我的天…

CSS3实现纸张边角卷起效果

html代码 1 <body>2 <div class"page">3 <div class"page-box">4 <h1>5 兔子先生6 </h1>7 <p>8 这几…