Unity项目开发笔记(十六)
目前产品在Unity端的开发任务已经趋于平稳 , 但是依旧达不到”非常稳定”的状态 , 随着用户量的增加 , 在复杂的PC网络和设备环境下 每次迭代版本都会频频中坑,这不 这次中招的是WWW和UnityWebRequest.
0x00. 基础介绍
我们知道现在的Unity (Mono2.0 + .NET 3.5) 在不增加第三方插件的情况下 有三种HTTP请求方式:
- WWW – 大家都知道,对于PC平台来说 底层封装了一遍Curl
- UnityWebRequest – Unity5.0以后新出来的一种方式
- HttpWebRequest – .net 自带的一种网络请求方式
以上三种请求方式各有优劣:
WWW 主要是封装得比较便捷 , 下载AssetBoundle Texture等比较方便 省了转换和加载的过程.
UnityWebRequest 主要是更加接近于 HttpWebRequest ,但是参数设置又比 HttpWebRequest 省事一点,适用于项目的大批量API实现 , 使用UnityWebRequest 可以比较自由的添加各种网络数据传输方式和格式.
HttpWebRequest .net框架自带 ,属于三种方式中自由度最大的 , 但是缺点也很明显 各种参数设置必须了然于胸 .
0x01. 问题重现
看下面一段代码
// 创建请求 WWW load = new WWW(url); float timing = 0; // 通过 isDone 判断是否结束 while (false == load.isDone) { // 设计超时 timing = timing + Time.deltaTime; if (timing > m_fTimeOut) { if (onDownloaded != null) { onDownloaded(null); } load.Dispose(); yield break; } yield return null; } if (true == string.IsNullOrEmpty(load.error)) { // 请求出错 } else { // 请求成功 }
大多数情况下 以上代码是没有问题的,当然在Unity比较老的版本 (4.3以前) 在编辑器模式下是不能通过 isDone 来判断是否结束的. 在极少的情况下 以上参数 isDone 永远都为false , 目前这个BUG触发的概率大概为:
概率 = 1/50 x 1 / 500 : 也就是 500个用户中会出现1个人 他在他的电脑中使用WWW发送50个请求其中有1个请求的isDone永远为false
通过对出现此问题的用户进行分析 ,发现大多数都集中在 联通和移动网络. 地区分布在云南贵州等等.
0x02. 解决方案 (虽然一般都把解决方案放在文末 但是写这篇文章不为分享解决方案)
这个问题的解决方案不值一提 , 直接当使用 WWW 或者 UnityWebRequest 请求超时的时候使用 HttpWebRequest 进行当前请求即可. 对的 就是那么简单,伪代码为:
if(WWW is TimeOut) { GoTo HttpWebRequest; } else { Goto Finish; }
深入WWW底层分析
虽然很轻易解决了问题 , 由于Unity的底层代码不开源 , 我们只能 绕道取荆州, 但是遇到问题绕道一直都不是我的性格 , 现在让我们扒一扒这个WWW底层到底是什么 , 基于某种机缘巧合我手上获得一份Unity老版本的底层C代码 , 同时在这两年Unity官方开源了C#部分代码: 传送门 , 有了这两份东西我们就可以深入WWW看看底层究竟长什么样:
自顶向下 WWW 是这样调用的:
(第一步第二步属于公开代码内容 , 这里以文字呈现 , 后面的深入步骤属于闭源部分 这里只能以图片的方式.)
// 第一步 上层创建 WWW load = new WWW(strURL); // 第二步 来到UnityEngine.dll public WWW(string url) { this.InitWWW(url, null, null); } [MethodImpl(MethodImplOptions.InternalCall)] public extern void InitWWW(string url, byte[] postData, string[] iHeaders);
第三步 来到C++底层 – 中间接口
第四步 来到C++底层 – 真正的WWW
这里主要注意 我们不是WebPlayer模式 . 也就是说会走到 CreatePlatformWWWBackend 中
第四步 来到C++底层 – CreatePlatformWWWBackend
一种跟踪 显然会走到 WWWCurl 里 , 注意 Curl,应该是指libcurl – 详解传送门 , 继续跟踪来到如下代码位置:
WWWCurl::WWWCurl -> DoInit -> StartThread -> WWW_ThreadEntryPoint -> GetURL
注意函数 WWW_ThreadEntryPoint 在 www.abortDownload 和 www.GetError() == NULL 都成立的情况下 才 www.FeedUnityWebStream(true) , 在curl整个调用过程中 都是 www.FeedUnityWebStream(false) , 说明isDone就在这个地方被改变.
最后我们看看这个GetURL是什么幺儿~
显然就是HTTP请求头的构造函数了.里面看到很多熟悉的字眼 , 使用fiddler随便抓一个Unity的请求就可以看到上面函数的数据头信息.
WWW底层代码的跟踪到此结束 . 基本上可以断定WWW之所以会发生isDone永远为false 和 curl有很大的关系 .下面我们看看isDone究竟是什么东西:
由上图 WWW 封装了一个 m_UnityWebStream 在 m_UnityWebStream 中保存 isDone数据. 看 IsFinished 函数即可确认:
而 IsFinished 函数里的数据 是根据上面 curl 的 WWWCurl::GetURL 函数中各种回调 WriteCallback,ReadCallback,ProgressCallback,HeaderCallback 中调用 FeedUnityWebStream 去修改的.
根据以上分析 ,如果要得出 isDone为何没有被设置为true, 只需重点跟踪 curl的几个回调函数即可. 根据以上代码 , 接下来准备分析一下libcurl的http请求 , 然后把这个模块单独抽出来 准备写个工具测试一下看看.
今天先到这里.
-EOF-
我的天,这么绕。自己用CurlSharp封一层就好了~原装无依赖
话说Unity还真喜欢各种IsReadyToPlay哈= =视频流的IsReadyToPlay就总出问题……
估计老外从未出现这些问题 , 我发现国内用户PC环境比移动环境复杂多了..最近出问题的用户几乎无一例外的是阉割的Win系统 或者奇葩网络. 而且新增某些第三方库的时候还要避过三大流氓:360/腾讯管家/金山杀毒