当前位置: 首页 > article >正文

【unity进阶知识4】封装unity协程工具,避免 GC(垃圾回收)

文章目录

  • 前言
  • 封装协程工具类,避免 GC(垃圾回收)
  • 使用
    • 1.使用默认方式使用协程
    • 2.使用自定义的 CoroutineTool 工具类来等待不同的时间
  • 完结

前言

在 Unity 中,使用 yield return null 、yield return new WaitForEndOfFrame()等会导致 GC(垃圾回收)开销。

  • yield return null: 每次调用这个语句,Unity 会创建一个新的迭代器状态机。当你执行协程时,如果你在协程中使用 yield return null,它会生成一个新的迭代器,这样会产生额外的内存分配。

  • new WaitForEndOfFrame(): 每次使用 new WaitForEndOfFrame() 都会创建一个新的 WaitForEndOfFrame 对象。这种频繁的对象创建会增加内存开销,并可能导致 GC 的触发。

但是注意,启动协程时会创建一个 Coroutine 对象,这本身会导致一次内存分配,进而可能引发垃圾回收(GC)。这是协程在 Unity 中的一个固有特性,无法完全避免。但是它还是解决了协程的主要痛点,毕竟一个项目启动的协程一般不会很多。如果你在意,可以选择使用 UniTask:
【推荐100个unity插件之33】比 Unity 自带协程更高效的异步处理方式,提供一个高性能和0GC的async/await异步方案——UniTask插件

封装协程工具类,避免 GC(垃圾回收)

提前new好协程所需要的WaitForEndOfFrame、WaitForFixedUpdate、WaitForFrameStruct类的对象,避免GC。

/// <summary>
/// 协程工具类,避免 GC(垃圾回收)
/// </summary>
public static class CoroutineTool
{
    // 定义一个结构体,用于表示等待一帧的状态
    private struct WaitForFrameStruct : IEnumerator
    {
        public object Current => null;

        public bool MoveNext() { return false; } // 一旦调用,立即返回 false,停止迭代

        public void Reset() { } // 重置方法,不做任何操作
    }

    // 预定义的等待结束帧对象,避免多次创建
    private static WaitForEndOfFrame waitForEndOfFrame = new WaitForEndOfFrame();
    // 预定义的等待固定更新对象,避免多次创建
    private static WaitForFixedUpdate waitForFixedUpdate = new WaitForFixedUpdate();

    /// <summary>
    /// 获取等待结束帧的对象
    /// </summary>
    public static WaitForEndOfFrame WaitForEndOfFrame()
    {
        return waitForEndOfFrame;
    }

    /// <summary>
    /// 获取等待固定更新的对象
    /// </summary>
    public static WaitForFixedUpdate WaitForFixedUpdate()
    {
        return waitForFixedUpdate;
    }

    /// <summary>
    /// 等待指定时间(以秒为单位)
    /// </summary>
    /// <param name="time">等待的时间</param>
    public static IEnumerator WaitForSeconds(float time)
    {
        float currTime = 0;
        while (currTime < time)
        {
            currTime += Time.deltaTime;
            yield return new WaitForFrameStruct(); // 等待一帧
        }
    }

    /// <summary>
    /// 等待指定的实时时间(不受时间缩放影响)
    /// </summary>
    /// <param name="time">等待的时间</param>
    public static IEnumerator WaitForSecondsRealtime(float time)
    {
        float currTime = 0; // 当前经过的时间
        while (currTime < time) // 当经过的时间小于指定时间时
        {
            currTime += Time.unscaledDeltaTime; // 增加经过的时间(不受时间缩放影响)
            yield return new WaitForFrameStruct(); // 等待一帧
        }
    }

    /// <summary>
    /// 等待指定帧数
    /// </summary>
    /// <param name="count">等待的帧数,默认为1</param>
    public static IEnumerator WaitForFrame(int count = 1)
    {
        for (int i = 0; i < count; i++)
        {
            yield return new WaitForFrameStruct(); // 等待一帧
        }
    }
}

使用

1.使用默认方式使用协程

IEnumerator DelayedAction()
{
    yield return new WaitForEndOfFrame();// 等待到当前帧的结束(即所有渲染操作完成后)
    yield return new WaitForFixedUpdate();// 等待到下一个固定更新(适用于物理计算)
    yield return new WaitForSeconds(2.0f); // 等待2秒(游戏时间)
    yield return new WaitForSecondsRealtime(2.0f);// 等待2秒(现实时间,不受游戏时间缩放影响)
 	yield return null;// 等待一帧
}

2.使用自定义的 CoroutineTool 工具类来等待不同的时间

IEnumerator DelayedAction()
{
    yield return CoroutineTool.WaitForEndOfFrame(); // 等待到当前帧的结束
    yield return CoroutineTool.WaitForFixedUpdate(); // 等待到下一个固定更新
    yield return CoroutineTool.WaitForSeconds(2.0f); // 等待2秒(游戏时间)
    yield return CoroutineTool.WaitForSecondsRealtime(2.0f); // 等待2秒(现实时间,不受游戏时间缩放影响)
    yield return CoroutineTool.WaitForFrame(); // 等待一帧
    yield return CoroutineTool.WaitForFrame(3); // 等待3帧
}

完结

赠人玫瑰,手有余香!如果文章内容对你有所帮助,请不要吝啬你的点赞评论和关注,你的每一次支持都是我不断创作的最大动力。当然如果你发现了文章中存在错误或者有更好的解决方法,也欢迎评论私信告诉我哦!

好了,我是向宇,https://xiangyu.blog.csdn.net

一位在小公司默默奋斗的开发者,闲暇之余,边学习边记录分享,站在巨人的肩膀上,通过学习前辈们的经验总是会给我很多帮助和启发!如果你遇到任何问题,也欢迎你评论私信或者加群找我, 虽然有些问题我也不一定会,但是我会查阅各方资料,争取给出最好的建议,希望可以帮助更多想学编程的人,共勉~
在这里插入图片描述


http://www.kler.cn/a/325160.html

相关文章:

  • 3. 后端验证前端Token
  • 在.NET用C#将Word文档转换为HTML格式
  • PyBroker:利用 Python 和机器学习助力算法交易
  • Nmap之企业漏洞扫描(Enterprise Vulnerability Scanning for Nmap)
  • Ability Kit-程序框架服务(类似Android Activity)
  • turtle教学课程课堂学习考试在线网站
  • Vue76 编程式路由导航
  • 云手机的默认ip地址是什么
  • (补充)3DMAX初级小白班第三课:创建物体+物体材质编辑
  • gateway--网关
  • 【spring】 -Dlog4j.configurationFile配置log4j2的自定义路径
  • mac Wireshark You do not have permission to capture on device “rvio“.
  • Java 编码系列:集合框架(List、Set、Map 及其常用实现类)
  • 从0到1教你学会写测试总结
  • Emiya 家今天的饭C++
  • 封装axios请求
  • C++ 中是#pragma once
  • cefsharp新版本OnBeforeResourceLoad 禁止http自动跳转https显示404错误解决办法 含代码
  • 在Ubuntu中自动挂载SMB/CIFS共享
  • Springboot2笔记核心技术——1.基础入门
  • Java-数据结构-Map和Set-(二)-哈希表 |ू・ω・` )
  • 第八届蓝桥杯嵌入式省赛程序设计题解析(基于HAL库)
  • SQL_having_pandas_filter
  • 天童美语:全国爱牙日|健康护“齿”知识
  • 从0学习React(5)---通过例子体会setState
  • 使用Docker快速本地部署RSSHub结合内网穿透访问RSS订阅源