文档章节

Unity定时器堆栈显示

梦想游戏人
 梦想游戏人
发布于 2017/09/04 16:36
字数 510
阅读 11
收藏 0

一般定时器是一个匿名函数,callback,在函数执行错误的话,很难查找到错误信息。

在这里可以利用添加定时器的时候 对调用栈进行一个快照,当发生错误时输出即可,这样调试起来方便多了。

 

AddTimer中栈 信息的一个快照为string 作为Timer对象的成员,release发行时可以去掉这种辅助机制

/*
* Author:  caoshanshan
* Email:   me@dreamyouxi.com
code copy from https://git.oschina.net/dreamyouxi/XYGame;
 */
using UnityEngine;
using System.Collections;
using System.Collections.Generic;
using ExtBase;

public sealed class TimerQueue : Singleton<TimerQueue>
{

    /// <summary>
    /// 添加一个真实时间定时器,返回函数调用即可 打断定时任务
    /// </summary>
    /// <param name="time_delay"></param>
    /// <param name="cb"></param>
    /// <param name="repeat_times">  <0 will be run forever</param>
    public VoidFuncVoid AddTimer(float time_delay, VoidFuncNN cb, int repeat_times = 1, params object[] objs)
    {
        return this.AddTimer(time_delay, () => { cb(objs); }, repeat_times);
    }
    /// <summary>
    /// 添加一个真实时间定时器,返回函数调用即可 打断定时任务
    /// </summary>
    /// <param name="time_delay"></param>
    /// <param name="cb"></param>
    /// <param name="repeat_times">  <0 will be run forever</param>
    public VoidFuncVoid AddTimer(float time_delay, VoidFuncVoid cb, int repeat_times = 1)
    {
        if (repeat_times == 0) return null;// () => { };
        Timer time = new Timer();
        time.delay = time_delay;
        time.cb = cb;
        time.repeat_times = repeat_times;
        //记录定时器添加的时候的快照栈信息,以 提供错误时 更多调试信息
        System.Diagnostics.StackTrace st = new System.Diagnostics.StackTrace(true);

        string ss = "\n";
        for (int i = 0; i < st.FrameCount; i++)
        {
            var f = st.GetFrame(i);
            string name = f.GetFileName();
            if (name.Length > Application.dataPath.Length)
            {
                name = name.Substring(Application.dataPath.Length, name.Length - Application.dataPath.Length);
            }
            ss += "   " + f.GetMethod().ToString() + " at " + name + " (" + f.GetFileLineNumber() + ") \n";
        }
        time.stackInfo = ss;
        time.Init();
        list.Add(time);
        return () =>
        {
            if (time != null && this.list.Contains(time))
                time.SetInValid();
        };
    }

    /// <summary>
    /// unity engine tick
    /// </summary>
    public void Tick()
    {
        this.Tick(ref list);
    }
    private void Tick(ref List<TimerBase> list)
    {
        if (list.Count < 1)
            return;

        for (int i = 0; i < list.Count; i++)
        {
            if (list[i] == null)
                continue;

            TimerBase timer = list[i] as TimerBase;
            timer.Tick();
        }

        for (int i = 0; i < list.Count; )
        {
            TimerBase b = list[i] as TimerBase;
            if (b.IsInValid())
            {
                list.Remove(b);
            }
            else
            {
                ++i;
            }
        }
    }
    public void Clear()
    {
        this.list_ms.Clear();
        this.list.Clear();
    }
    List<TimerBase> list_ms = new List<TimerBase>();
    List<TimerBase> list = new List<TimerBase>();

}


class TimerBase : GAObject
{
    public string stackInfo = "";
    public virtual void Tick()
    {

    }

    public VoidFuncVoid cb = null;
    public int repeat_times = 1;
    protected int repeat_times_current = 0;
}

sealed class Timer : TimerBase
{
    public override void Tick()
    {
        if (this.IsInValid()) return;

        if (current < delay)
        {
            current += Time.deltaTime;
            return;
        }
        if (cb != null)
        {
            try
            {
                cb();
            }
            catch (System.Exception e)
            {
                Debug.LogError(e.Message);
                Debug.LogError("add timer stackTrace:" + this.stackInfo);
                Debug.LogError("call stackTrace:" + e.StackTrace);
            }
            cb = null;
        }
        current = 0.0f;
        if (++repeat_times_current >= repeat_times && repeat_times >= 0)
        {
            this.SetInValid();
        }
    }

    public override bool Init()
    {
        return true;
    }
    float current = 0.0f;
    public float delay = 0.0f;

}

 

© 著作权归作者所有

共有 人打赏支持
梦想游戏人
粉丝 34
博文 420
码字总数 119565
作品 0
成都
学习Mono与Unity3D关系的笔记

Xamarin公司开发的跨平台 .NET运行环境,是.NET框架的另一种实现。区别在于,它可以跨平台。(Unity的跨平台特性依赖于Mono) Mono的组成。 1.C#编译器。 最新的Momo版本(5.0+)c#编译器完全...

lishengxu159
05/05
0
0
unity3D -Mac- 破解

3.用root用户给破解文件权限 shell命令:sudo su 然后输入密码(密码为当前用户密码) 接着更改破解文件Unity的所属用户,以及给Unityv4.x.ulf文件执行权限 shell命令:chown root:admin Un...

MrLovelyCbb
2015/04/27
0
0
【Untiy3D 游戏开发之一】Unity3D For Mac最新3.4.1版本破解教程分享并将Unity3D项目在iphone上运行;

声明:本破解教程为学习其他别人文章进行的,如有侵犯等请通知我,我删除对应段落; Himi 原创, 欢迎转载,转载请在明显处注明! 谢谢。 原文地址:http://blog.csdn.net/xiaominghimi/art...

迷途d书童
2012/03/19
0
0
Unity-UnityScript转换C#工具(js转c#)

如何使用 首先,从https://github.com/Unity-Technologies/unityscript2csharp/releases下载工具。 运行转换工具之前: 备份您的项目 请记住,如果您的UnityScripts将#pragma严格应用于它们,...

qq_21153225
05/09
0
0
UnityShader源码2017---学习笔记与自我拓展006

源自CubeBlend 首先映入眼帘的是 稍微解释一下[NoScaleOffset]吧,就是在material的面板上没有tilling和offset控制部分的GUI显示。 然后我看到了 这个应该是unity声明的吧,比如想MainTexST...

u012871784
05/29
0
0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

流量劫持是如何产生的?

流量劫持,这种古老的攻击沉寂了一段时间后,最近又开始闹的沸沸扬扬。众多知名品牌的路由器相继爆出存在安全漏洞,引来国内媒体纷纷报道。只要用户没改默认密码,打开一个网页甚至帖子,路由...

谢思华
22分钟前
0
0
Hadoop Client无法使用maven下载源码

最近在学习hadoop,使用maven的时候想看一下源码的注释,结果IDEA一直提示无法下载 搞得我一度以为maven坏掉了。 但是通过搜索,发现在maven仓库里确实没有源码.... 而2.8.1以及之前的版本是...

Iceberg_XTY
23分钟前
0
0
为什么程序员千万不要重写代码?

你所做的事情,也许暂时看不到成果,但不要灰心或焦虑,你不是没有成长,而是在扎根。 图片来自网络 0 前言 程序员都有一颗工程师的心,所以当他们到一片新的场地想做的第一件事就是,将旧的...

Java小铺
25分钟前
0
0
VUE集成AdminLte

1. 安装需要到插件 npm i admin-lte -Snpm i jquery -Snpm i axios -Snpm i vue-router -S 2. 配置webpack.config.js 2.1 module.exports.module.rules修改字体loader: {test: /\.(p......

Pasenger
今天
0
0
Spring Aop原理之切点表达式解析

在前面的文章(Spring AOP切点表达式详解)中,我们总结了Spring Aop切点表达式的用法,而在上文(Spring Aop原理之Advisor过滤)中我们讲到,切点表达式的解析主要是在PatternParser.parse...

爱宝贝丶
今天
0
0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

返回顶部
顶部