Unity性能优化之代码优化小技巧_018

发表于2018-09-09
评论0 5.9k浏览
参与“Unity性能优化”征文活动

对于Unity性能优化,目前接触到的大概有这几个方面:
1. Draw Call
2. 资源(模型、贴图、粒子
3. 渲染(相机、光照、Shader)
4. 网络
5. 代码(代码编写、资源加载、物理系统)
可以在Unity自带的Profiler窗口查看项目性能消耗主要在哪几个地方,然后有针对性的进行优化。

作为一个程序,这里跟大家分享一下最近常用到代码方面的一点技巧,如有不足之处,欢迎大大们提出宝贵意见~

1. 在场景中有大量物体频繁的激活或隐藏时,不使用SetActive(),在需要隐藏时移到屏幕外 ,显示时再移到屏幕内,即修改transform.position。还有一个方法,把需要隐藏的物体设为一个已隐藏的物体的子物体,因为父物体是未激活状态,子物体会自动隐藏,不过这种方法消耗与SetActive()差不多,不推荐使用。
下面是测试代码:
    void Start()
    {
        //准备工作
        List<GameObject> objs = new List<GameObject>();
        DateTime begainTime;
        //生成预设GameObject, 空物体或简单的GameObject,性能消耗较小,GameObject越复杂,消耗越大
        GameObject prefab = GameObject.CreatePrimitive(PrimitiveType.Cube);
        for (int i = 0; i < 100; i++)
        {
            GameObject.CreatePrimitive(PrimitiveType.Cube).transform.parent = prefab.transform;
        }
        //加载一定数量GameObject到场景, 大量操作才会对性能产生明显影响,GameObject数量越多,消耗越大
        for (int i = 0; i < 100; i++)
        {
            objs.Add(Instantiate(prefab, transform));
        }
        //开始测试
        begainTime = DateTime.Now;
        for (int i = 0; i < objs.Count; i++)
        {
            objs[i].SetActive(false);
        }
  Debug.Log("SetActive(false)  " + (DateTime.Now - begainTime).ToString());
        begainTime = DateTime.Now;
        for (int i = 0; i < objs.Count; i++)
        {
            objs[i].SetActive(true);
        }
  Debug.Log("SetActive(true)  " + (DateTime.Now - begainTime).ToString());
        begainTime = DateTime.Now;
        for (int i = 0; i < objs.Count; i++)
        {
            objs[i].transform.position = Vector3.one * 1000;
        }
Debug.Log("transform.position  " + (DateTime.Now - begainTime).ToString());
        Transform idleParent = new GameObject().transform;
        idleParent.gameObject.SetActive(false);
        begainTime = DateTime.Now;
        for (int i = 0; i < objs.Count; i++)
        {
            objs[i].transform.parent = idleParent;
        }
  Debug.Log("transform.parent  " + (DateTime.Now - begainTime).ToString());
    }
输出结果:
SetActive(true)消耗最大,SetActive(false)与transform.parent其次,transform.position消耗最小,占用的时间可以忽略不计。

2. 动态实例化到场景的物体,名字都会有一个后缀(Clone),有时候为了方便识别,会修改其名字,同样会产生性能消耗。比如:
        //修改名字, 字符串操作
        for (int i = 0; i < objs.Count; i++)
        {
            objs[i].name = "New Obj " + i;
        }

3. 在不影响正常运行结果的情况下,减少Update或FixUpdate的调用次数
假设现在将所有的Updata刷新逻辑写在DoUpdata中
 void DoUpdata() { Debug.Log("Do Updata"); }

3.1 每隔一定数量帧,执行一次DoUpdata
    int spaceCount = 10;
    int updateCount;
    void Update()
    {
        if (updateCount % spaceCount == 0)
        {
            updateCount = 0;
            //DoUpdata();
        }
        updateCount++;
    }

3.2 使用协程While(true)循环,每次循环间隔一定时间,调用DoUpdata,需要在Start函数中开启协程
   float spaceTime = 0.1f;
    IEnumerator StartUpdata()
    {
        while(true)
        {
            DoUpdata();
            yield return new WaitForSeconds(spaceTime);
        }
    }

3.3 使用InvokeRepeating循环调用DoUpdata
    float repeatTime = 0.1f;
    public void Start()
    {
        InvokeRepeating("DoUpdata", 0, repeatTime);
        //StartCoroutine(StartUpdata());
    }

4. 使用对象池
操作目标相对较少,可简化对象池,实现效果
    GameObject prefab;
    Transform parent;

4.1 简化一
List<GameObject> objList = new List<GameObject>();
    GameObject CreatObj01()
    {
        //先遍历已有
        for (int i = 0; i < objList.Count; i++)
        {
            if (!objList[i].activeSelf)
            {
                objList[i].SetActive(true);
                /*
                 *  在此进行初始化
                 */
                return objList[i];
            }
        }
        //创建
        GameObject newObj = Instantiate(prefab, parent);
        /*
         *  在此进行初始化
         */
        objList.Add(newObj);
        return newObj;
    }
    void DestroyObj01(GameObject obj)
    {
        obj.SetActive(false);
        /*
         *  在此进行重置
         */
    }

4.2 简化二
 List<GameObject> workList = new List<GameObject>();
    List<GameObject> idleList = new List<GameObject>();
    GameObject CreatObj02()
    {
        GameObject newObj;
        //是否有空闲
        if (idleList.Count > 0)
        {
            newObj = idleList[0];
            idleList.RemoveAt(0);
            workList.Add(newObj);
        }
        else
        {
            //创建
            newObj = Instantiate(prefab, parent);
            workList.Add(newObj);
        }
        /*
         *  newObj 在此进行初始化
         */
        return newObj;
    }
    void DestroyObj02(GameObject obj)
    {
        if (workList.Contains(obj))
            workList.Remove(obj);
        idleList.Add(obj);
        obj.SetActive(false);
        /*
         *  在此进行重置
         */
    }
如果目标对象较多,就需要写一个正经的对象池了,因为之前写过,这里就直接给链接啦

5. 音效播放时,为避免频繁创建、销毁播放器,可以对音效统一管理,同样之前写过,上链接~

6. 场景中经常需要动态生成GameObject,当一次创建的数量较多时,在不影响使用的情况下,可以使用协程,在多帧内完成创建,可参照:

7. 部分简单的物理计算可以不使用Unity提供的物理系统,简化物理计算量,下面是关于向量计算的一个栗子:

8. 选择合理的数据结构存储数据。
在数据结构转换时,可以避免如下的for循环,利用C#提供的方法
        int[] array = new int[] { 1, 2, 3, 4, 5 };
        List<int> list = new List<int>();
        for (int i = 0; i < array.Length; i++)
        {
            list.Add(array[i]);
        }

9. 其他
9.1 尽量避免在Update和for循环内创建临时变量。
9.2 尽量避免创建临时字符串。
9.3 可以使用for循环的情况,就不用foreach。
9.4 每个继承MonoBehaviour的类,都会自动生成Update方法,但很多类是用不到Update的,这时候需要将其删除,毕竟实时调用空方法,多少还是有消耗的。
9.5 数值计算中使用乘法而不用触发,比如 a / 2, 可以写成 a * 0.5f。
9.6 比较他Tag值时,使用if(gameObject.CompareTag("Tag")),而不是if(gameObject.tag == "Tag")。
9.7 开发过程中会各种Debug,这也会有一定的消耗,可以对Debug进行封装,设置一个bool值,不过这样console面板双击不会跳转到调用Debug的代码行,最好的做法是将其做成dll文件。

OK~ 关于Unity性能优化中的代码优化技巧暂时就想到了这么多,期待大大们补充~~


如社区发表内容存在侵权行为,您可以点击这里查看侵权投诉指引