Unity 工具类 之 简单定时任务系统 TimerTaskSystem 的实现

Unity 工具类 之 简单定时任务系统 TimerTaskSystem 的实现

Unity 工具类 之 简单定时任务系统 TimerTaskSystem 的实现

目录


一、简单介绍

Unity 工具类,自己整理的一些游戏开发可能用到的模块,单独独立使用,方便游戏开发。

定时任务系统,顾名思义就是在指定时间执行指定任务,这里包括定时(ms,s,min,hour,day)任务和定时帧任务两大类定时任务。

二、关键技术

1、实现原理很简单,在 Update 中计时计数看任务是否到达定时执行条件,到了执行,没到继续监测;

2、单例,使用单例很方便的进行定时任务添加;

3、定时任务,有循环次数,可以无限循环执行,也可以制定次数循环;

4、AddTimerTask/AddTimerFrameTask 添加定时/帧任务;

5、DeleteTimerTask/DeleteTimerFrameTask 删除定时/帧任务;

6、ReplaceTimerTask/ReplaceTimerFrameTask 替换定时/帧任务

三、注意事项

1、帧任务执行,计数帧可能会越界;

四、效果预览

www.zeeklog.com  - Unity 工具类 之 简单定时任务系统 TimerTaskSystem 的实现

五、实现步骤

1、打开Unity,新建一个工程,如下图

www.zeeklog.com  - Unity 工具类 之 简单定时任务系统 TimerTaskSystem 的实现

2、在工程中新建脚本,MonoSingleton 单列类,TimerTaskSystem 定时任务系统实现定时的核心代码逻辑,Test_TimerTaskSystem 测试 定时任务系统,如下图

www.zeeklog.com  - Unity 工具类 之 简单定时任务系统 TimerTaskSystem 的实现

3、把测试脚本和定时任务系统脚本挂载到场景中,如下图

www.zeeklog.com  - Unity 工具类 之 简单定时任务系统 TimerTaskSystem 的实现

4、测试效果如下图

www.zeeklog.com  - Unity 工具类 之 简单定时任务系统 TimerTaskSystem 的实现

五、关键代码

1、MonoSingleton

using UnityEngine;

public abstract class MonoSingleton<T> : MonoBehaviour where T : MonoBehaviour
{
    private static T instance = null;

    private static readonly object locker = new object();

    private static bool bAppQuitting;

    public static T Instance
    {
        get
        {
            if (bAppQuitting)
            {
                instance = null;
                return instance;
            }

            lock (locker)
            {
                if (instance == null)
                {
                    // 保证场景中只有一个 单例
                    T[] managers = Object.FindObjectsOfType(typeof(T)) as T[];
                    if (managers.Length != 0)
                    {
                        if (managers.Length == 1)
                        {
                            instance = managers[0];
                            instance.gameObject.name = typeof(T).Name;
                            return instance;
                        }
                        else
                        {
                            Debug.LogError("Class " + typeof(T).Name + " exists multiple times in violation of singleton pattern. Destroying all copies");
                            foreach (T manager in managers)
                            {
                                Destroy(manager.gameObject);
                            }
                        }
                    }


                    var singleton = new GameObject();
                    instance = singleton.AddComponent<T>();
                    singleton.name = "(singleton)" + typeof(T);
                    singleton.hideFlags = HideFlags.None;
                    DontDestroyOnLoad(singleton);

                }
                instance.hideFlags = HideFlags.None;
                return instance;
            }
        }
    }

    protected virtual void Awake()
    {
        bAppQuitting = false;
    }

    protected virtual void OnDestroy()
    {
        bAppQuitting = true;
    }
}

2、TimerTaskSystem

using System;
using System.Collections.Generic;
using UnityEngine;

/// <summary>
/// 定时任务系统功能有
/// 1、添加定时任务
/// 2、删除定时任务
/// 3、替换定时任务
/// </summary>
public class TimerTaskSystem : MonoSingleton<TimerTaskSystem>
{
    // 任务 ID
    private int taskId;
    // taskId 列表
    private List<int> taskIdList = new List<int>();
    // 回收任务结束不用了的 taskId 的缓存列表
    private List<int> recycleTaskIdList = new List<int>();

    // 定时任务列表
    private List<TimerTask> taskTimeList = new List<TimerTask>();
    // 临时 定时任务列表 (作为缓存列表)
    private List<TimerTask> tmpTaskTimeList = new List<TimerTask>();

    // 获取任务id 的锁
    private static readonly string objLock = "lock";

    // 帧计数
    int frameCounter = 0;
    // 帧定时任务列表
    private List<TimerFrameTask> taskFrameTimeList = new List<TimerFrameTask>();
    // 帧临时 定时任务列表 (作为缓存列表)
    private List<TimerFrameTask> tmpTaskFrameTimeList = new List<TimerFrameTask>();



    // Update is called once per frame
    void Update()
    {
        // 检测定时任务是否到达可执行
        CheckTimerTask();

        // 检测定时帧任务是否到达可执行
        CheckTimerFrameTask();

        // 回收任务ID
        RecycleTaskId();
    }

    /// <summary>
    /// 检测定时任务是否到达可执行
    /// </summary>
    private void CheckTimerTask() {
        // 加入临时定时任务列表的定时任务 (这样做保证同帧不干扰下面遍历任务列表)
        for (int tempIndex = 0; tempIndex < tmpTaskTimeList.Count; tempIndex++)
        {
            taskTimeList.Add(tmpTaskTimeList[tempIndex]);
        }
        tmpTaskTimeList.Clear();

        // 遍历检测定时任务是否满足执行条件
        for (int index = 0; index < taskTimeList.Count; index++)
        {
            TimerTask task = taskTimeList[index];
            if (Time.realtimeSinceStartup * 1000 < task.destTime)
            {
                continue;
            }
            else
            {
                Action callback = task.callback;

                // 异常捕获,避免回调函数出错
                try
                {
                    if (callback != null)
                    {
                        callback();
                    }
                }
                catch (Exception e)
                {

                    Debug.Log(e.Message);
                }


                // 任务循环只剩一次
                if (task.loopCount == 1)
                {
                    // 把结束任务不用的 taskid 缓存到 回收任务 id 列表中
                    recycleTaskIdList.Add(task.taskId);
                    taskTimeList.RemoveAt(index);
                    index--;
                }
                else
                {
                    // 不是无限循环,循环次数 -1
                    if (task.loopCount != 0)
                    {
                        task.loopCount -= 1;
                    }
                    // 目标时间更新一个 delay 时间
                    task.destTime += task.delay;
                }
            }

        }
    }

    /// <summary>
    /// 检测定时帧任务是否到达可执行
    /// </summary>
    private void CheckTimerFrameTask()
    {
        // 帧计数(会不会越界)
        frameCounter += 1;

        // 加入临时定时任务列表的定时任务 (这样做保证同帧不干扰下面遍历任务列表)
        for (int tempIndex = 0; tempIndex < tmpTaskFrameTimeList.Count; tempIndex++)
        {
            taskFrameTimeList.Add(tmpTaskFrameTimeList[tempIndex]);
        }
        tmpTaskFrameTimeList.Clear();

        // 遍历检测定时任务是否满足执行条件
        for (int index = 0; index < taskFrameTimeList.Count; index++)
        {
            TimerFrameTask task = taskFrameTimeList[index];
            if (frameCounter < task.destFrame)
            {
                continue;
            }
            else
            {
                Action callback = task.callback;

                // 异常捕获,避免回调函数出错
                try
                {
                    if (callback != null)
                    {
                        callback();
                    }
                }
                catch (Exception e)
                {

                    Debug.Log(e.Message);
                }


                // 任务循环只剩一次
                if (task.loopCount == 1)
                {
                    // 把结束任务不用的 taskid 缓存到 回收任务 id 列表中
                    recycleTaskIdList.Add(task.taskId);
                    taskFrameTimeList.RemoveAt(index);
                    index--;
                }
                else
                {
                    // 不是无限循环,循环次数 -1
                    if (task.loopCount != 0)
                    {
                        task.loopCount -= 1;
                    }
                    // 目标时间更新一个 delay 时间
                    task.destFrame += task.delay;
                }
            }

        }
    }


    #region TimeTask    

    /// <summary>
    /// 添加定时的任务
    /// </summary>
    /// <param name="callback">任务回调</param>
    /// <param name="delay">延迟时间</param>
    /// <param name="loopCount">循环次数(0 为 无限循环)</param>
    /// <param name="timerUnit">时间单位(ms,s,m,h,Day)</param>
    /// <returns>返回任务id</returns>
    public int AddTimerTask(Action callback, float delay,int loopCount = 1, TimerUnit timerUnit = TimerUnit.Millisecond) {
        if (timerUnit != TimerUnit.Millisecond) {
            switch (timerUnit)
            {
                case TimerUnit.Second:
                    delay = delay * 1000;
                    break;
                case TimerUnit.Minute:
                    delay = delay * 1000 * 60;
                    break;
                case TimerUnit.Hour:
                    delay = delay * 1000 * 60 * 60;
                    break;
                case TimerUnit.Day:
                    delay = delay * 1000 * 60 * 60 * 24;
                    break;
                default:
                    Debug.Log("Add Tssk Unit Type Error...");

                    break;
            }
        }
        int tid = GetTaskId();

        float destTime = Time.realtimeSinceStartup * 1000 + delay;
        TimerTask timerTask = new TimerTask(tid, callback, destTime, delay, loopCount);                

        // 添加新定时任务的时候,先添加到临时列表(避免同帧遍历任务列表时干扰遍历列表)
        tmpTaskTimeList.Add(timerTask);

        // tid 添加到使用列表中
        taskIdList.Add(tid);

        return tid;
    }

    /// <summary>
    /// 删除指定任务
    /// </summary>
    /// <param name="taskId">任务id</param>
    /// <returns>返回是否删除</returns>
    public bool DeleteTimerTask(int taskId) {
        bool exist = false;
        for (int i = 0; i < taskTimeList.Count; i++) {
            TimerTask task = taskTimeList[i];

            if (task.taskId == taskId) {
                taskTimeList.RemoveAt(i);
                for (int j = 0; j < taskIdList.Count; j++) {
                    if (taskIdList[j] == taskId) {
                        taskIdList.RemoveAt(j);
                        break;
                    }
                }

                exist = true;
                break;
            }
        }

        // 从临时缓存任务列表中查找
        if (exist == false) {
            for (int i =0; i<tmpTaskTimeList.Count; i++) {
                TimerTask task = tmpTaskTimeList[i];
                if (task.taskId == taskId) {

                    tmpTaskTimeList.RemoveAt(i);
                    for (int j = 0; j < taskIdList.Count; j++)
                    {
                        if (taskIdList[j] == taskId)
                        {
                            taskIdList.RemoveAt(j);
                            break;
                        }
                    }

                    exist = true;
                    break;
                }
            }
        }

        return exist;
    }

    /// <summary>
    /// 替换存在的任务
    /// </summary>
    /// <param name="taskId">任务id</param>
    /// <param name="callback">任务回调</param>
    /// <param name="delay">延迟时间</param>
    /// <param name="loopCount">循环次数(0 为 无限循环)</param>
    /// <param name="timerUnit">时间单位(ms,s,m,h,Day)</param>
    /// <returns>返回是否替换成功</returns>
    public bool ReplaceTimerTask(int taskId, Action callback, float delay, int loopCount = 1, TimerUnit timerUnit = TimerUnit.Millisecond)
    {
        if (timerUnit != TimerUnit.Millisecond)
        {
            switch (timerUnit)
            {
                case TimerUnit.Second:
                    delay = delay * 1000;
                    break;
                case TimerUnit.Minute:
                    delay = delay * 1000 * 60;
                    break;
                case TimerUnit.Hour:
                    delay = delay * 1000 * 60 * 60;
                    break;
                case TimerUnit.Day:
                    delay = delay * 1000 * 60 * 60 * 24;
                    break;
                default:
                    Debug.Log("Add Tssk Unit Type Error...");

                    break;
            }
        }
        

        float destTime = Time.realtimeSinceStartup * 1000 + delay;
        TimerTask newTimerTask = new TimerTask(taskId, callback, destTime, delay, loopCount);

        bool isReplace = false;
        // 从任务列表中查找对应的 taskId 进行任务替换
        for (int i = 0; i < taskTimeList.Count; i++)
        {
            TimerTask task = taskTimeList[i];

            if (task.taskId == taskId)
            {
                taskTimeList[i] = newTimerTask;           

                isReplace = true;
                break;
            }
        }

        // 从临时缓冲任务列表中替换
        if (isReplace == false) {
            for (int i = 0; i < tmpTaskTimeList.Count; i++)
            {
                TimerTask task = tmpTaskTimeList[i];
                if (task.taskId == taskId)
                {

                    tmpTaskTimeList[i] = newTimerTask;                 
                    isReplace = true;
                    break;
                }
            }
        }


        return isReplace;
    }
    #endregion

    #region TimerFrameTask

    /// <summary>
    /// 添加定时帧任务
    /// </summary>
    /// <param name="callback">任务回调</param>
    /// <param name="delay">延迟帧数</param>
    /// <param name="loopCount">循环次数(0 为 无限循环)</param>
    /// <returns>返回任务id</returns>
    public int AddTimerFrameTask(Action callback, int delay, int loopCount = 1)
    {
        
        int tid = GetTaskId();

        int destTime = frameCounter + delay;
        TimerFrameTask timerTask = new TimerFrameTask(tid, callback, destTime, delay, loopCount);

        // 添加新定时帧任务的时候,先添加到临时列表(避免同帧遍历任务列表时干扰遍历列表)
        tmpTaskFrameTimeList.Add(timerTask);

        // tid 添加到使用列表中
        taskIdList.Add(tid);

        return tid;
    }

    /// <summary>
    /// 删除指定任务
    /// </summary>
    /// <param name="taskId">任务id</param>
    /// <returns>返回是否删除</returns>
    public bool DeleteTimerFrameTask(int taskId)
    {
        bool exist = false;
        for (int i = 0; i < taskFrameTimeList.Count; i++)
        {
            TimerFrameTask task = taskFrameTimeList[i];

            if (task.taskId == taskId)
            {
                taskFrameTimeList.RemoveAt(i);
                for (int j = 0; j < taskIdList.Count; j++)
                {
                    if (taskIdList[j] == taskId)
                    {
                        taskIdList.RemoveAt(j);
                        break;
                    }
                }

                exist = true;
                break;
            }
        }

        // 从临时缓存任务列表中查找
        if (exist == false)
        {
            for (int i = 0; i < tmpTaskFrameTimeList.Count; i++)
            {
                TimerFrameTask task = tmpTaskFrameTimeList[i];
                if (task.taskId == taskId)
                {

                    tmpTaskFrameTimeList.RemoveAt(i);
                    for (int j = 0; j < taskIdList.Count; j++)
                    {
                        if (taskIdList[j] == taskId)
                        {
                            taskIdList.RemoveAt(j);
                            break;
                        }
                    }

                    exist = true;
                    break;
                }
            }
        }

        return exist;
    }

    /// <summary>
    /// 替换存在的任务
    /// </summary>
    /// <param name="taskId">任务id</param>
    /// <param name="callback">任务回调</param>
    /// <param name="delay">延迟帧数</param>
    /// <param name="loopCount">循环次数(0 为 无限循环)</param>
    /// <returns>返回是否替换成功</returns>
    public bool ReplaceTimerFrameTask(int taskId, Action callback, int delay, int loopCount = 1)
    {

        int destTime = frameCounter+ delay;
        TimerFrameTask newTimerFrameTask = new TimerFrameTask(taskId, callback, destTime, delay, loopCount);

        bool isReplace = false;
        // 从任务列表中查找对应的 taskId 进行任务替换
        for (int i = 0; i < taskFrameTimeList.Count; i++)
        {
            TimerFrameTask task = taskFrameTimeList[i];

            if (task.taskId == taskId)
            {
                taskFrameTimeList[i] = newTimerFrameTask;

                isReplace = true;
                break;
            }
        }

        // 从临时缓冲任务列表中替换
        if (isReplace == false)
        {
            for (int i = 0; i < tmpTaskFrameTimeList.Count; i++)
            {
                TimerFrameTask task = tmpTaskFrameTimeList[i];
                if (task.taskId == taskId)
                {

                    tmpTaskFrameTimeList[i] = newTimerFrameTask;
                    isReplace = true;
                    break;
                }
            }
        }


        return isReplace;
    }

    #endregion

    #region Tools

    /// <summary>
    /// 回收任务结束,不用的taskId,以备下次使用
    /// </summary>
    private void RecycleTaskId() {
        for (int i = 0; i < recycleTaskIdList.Count; i++)
        {
            int taskId = recycleTaskIdList[i];
            for (int j = 0; j < taskIdList.Count; j++)
            {
                if (taskIdList[j] == taskId)
                {
                    taskIdList.RemoveAt(j);
                    break;
                }
            }
        }

        
        recycleTaskIdList.Clear();
    }

    /// <summary>
    /// 获得任务ID
    /// </summary>
    /// <returns>返回一个任务ID</returns>
    private int GetTaskId()
    {
        lock (objLock) {
            taskId += 1;

            while (true) {
                // 防止越界
                if (taskId == int.MaxValue) {
                    taskId = 0;
                }

                // 遍历循环检查taskId是否被使用
                bool isUsed = false;
                for (int i =0;i < taskIdList.Count; i++) {
                    if (taskId == taskIdList[i]) {
                        isUsed = true;
                        break;
                    }
                }
                // 如果该 taskId 没有被使用,跳出循环
                if (isUsed == false)
                {
                    break;
                }
                else {
                    taskId += 1;
                }
            }
        }

        return taskId;
    }

    #endregion

    #region 任务模型

    /// <summary>
    /// 任务模型
    /// </summary>
    class TimerTask {
        public int taskId;
        public Action callback;
        public float destTime;
        public float delay;
        public int loopCount;

        public TimerTask(int taskId, Action callback, float destTime, float delay, int loopCount)
        {
            this.taskId = taskId;
            this.callback = callback;
            this.destTime = destTime;
            this.delay = delay;
            this.loopCount = loopCount;
        }
    }

    /// <summary>
    /// 帧任务模型
    /// </summary>
    class TimerFrameTask
    {
        public int taskId;
        public Action callback;
        public int destFrame;
        public int delay;
        public int loopCount;

        public TimerFrameTask(int taskId, Action callback, int destFrame, int delay, int loopCount)
        {
            this.taskId = taskId;
            this.callback = callback;
            this.destFrame = destFrame;
            this.delay = delay;
            this.loopCount = loopCount;
        }
    }

    #endregion


}

/// <summary>
/// 时间单位枚举
/// </summary>
public enum TimerUnit
{
    Millisecond,
    Second,
    Minute,
    Hour,
    Day
}

3、Test_TimerTaskSystem

using System.Collections;
using System.Collections.Generic;
using UnityEngine;

public class Test_TimerTaskSystem : MonoBehaviour
{
    // Start is called before the first frame update
    void Start()
    {
        
    }

    // 任务 ID
    private int taskId;
    void Update() {

        #region 测试定时任务

        
        if (Input.GetKeyDown(KeyCode.A)) {
            Debug.Log("AddTimerTask");
            taskId = TimerTaskSystem.Instance.AddTimerTask(TestFunA, 1000, 0);
        }

        if (Input.GetKeyDown(KeyCode.D)){
            Debug.Log("DeleteTimerTask " + taskId);
            TimerTaskSystem.Instance.DeleteTimerTask(taskId);
        }

        if (Input.GetKeyDown(KeyCode.R))
        {
            Debug.Log("ReplaceTimerTask " + taskId);
            TimerTaskSystem.Instance.ReplaceTimerTask(taskId, TestFunB, 500, 2);
        }
        #endregion

        #region 测试定时帧任务

        
        if (Input.GetKeyDown(KeyCode.Alpha1))
        {
            Debug.Log("AddTimerFrameTask");
            taskId = TimerTaskSystem.Instance.AddTimerFrameTask(TestFrameFunA, 50, 0);
        }

        if (Input.GetKeyDown(KeyCode.Alpha2))
        {
            Debug.Log("DeleteTimerFrameTask " + taskId);
            TimerTaskSystem.Instance.DeleteTimerFrameTask(taskId);
        }

        if (Input.GetKeyDown(KeyCode.Alpha3))
        {
            Debug.Log("ReplaceTimerFrameTask " + taskId);
            TimerTaskSystem.Instance.ReplaceTimerFrameTask(taskId, TestFrameFunB, 500, 2);
        }
        #endregion

    }

    #region 测试用到的函数

    
    void TestFunA() {
        Debug.Log(" TestFunA taskId = " + taskId + "  "+System.DateTime.Now);
    }

    void TestFunB()
    {
        Debug.Log(" TestFunB taskId = " + taskId + "  " + System.DateTime.Now);
    }

    void TestFrameFunA()
    {
        Debug.Log(" TestFrameFunA taskId = " + taskId + "  " + System.DateTime.Now);
    }

    void TestFrameFunB()
    {
        Debug.Log(" TestFrameFunB taskId = " + taskId + "  " + System.DateTime.Now);
    }

    #endregion
}