C#:让您知道您的方法是被何“人”调用

禁用过期方法调用的实现与栈跟踪解析
本文介绍了如何在.NET中通过使用StackFrame类来限制具有Obsolete属性的方法调用,并详细解释了如何获取当前函数调用堆栈中其他函数的信息,以了解程序流程。同时展示了如何简化代码利用StackTrace类实现相同功能。
我们要在DisabledObsoleteMethod函数里限制具有“Obsolete”属性的方法调用,我们如何去做呢?在.Net中提供了一个"StackFrame"类用于表示当前线程上的函数调用堆栈中的某个具体函数,所以我们通过它就可继续编写我们的DisabledObsoleteMethod函数
也许在某些场合下我们想知道自己的某个方法是被谁(哪个方法)调用的?比如下面的例子:

        /// <summary>
        /// 正常方法
        /// </summary>

         static   void  Method1()
         {
            DisabledObsoleteMethod();
        }


         /// <summary>
        /// 过期方法
        /// </summary>

        [Obsolete]
         static   void  Method2()
         {
            DisabledObsoleteMethod();
        }


         /// <summary>
        /// 禁止过期方法调用此方法
        /// </summary>

         static   void  DisabledObsoleteMethod()
         {
            //如果调用此方法的方法中有"Obsolete"标记则不允许继续运行
        }


在上面代码中,我们要在DisabledObsoleteMethod函数里限制具有“Obsolete”属性的方法调用,我们如何去做呢?

在.Net中提供了一个"StackFrame"类用于表示当前线程上的函数调用堆栈中的某个具体函数,所以我们通过它就可继续编写我们的DisabledObsoleteMethod函数,代码如下:

         /// <summary>
        /// 禁止过期方法调用此方法
        /// </summary>

         static   void  DisabledObsoleteMethod()
         {
            StackFrame frame = new StackFrame(1);       //偏移一个函数位,也即是获取当前函数的前一个调用函数
            MethodBase method = frame.GetMethod();      //取得调用函数
            //反射获取其特性
            object[] attributes = method.GetCustomAttributes(typeof(ObsoleteAttribute), false);
            if (attributes.Length > 0)
            {
                //包含有"Obsolete"标记抛出错误或做其它处理
                throw new Exception(string.Format("方法{0}包括有Obsolete属性已被禁止调用",method.Name));
            }


            //继续做其它操作
        }


到此,当运行Method1时我们的DisabledObsoleteMethod函数就可以正常运作,而Method2就会被抛出异常警告了


因为StackFrame的构造函数可以指定偏移量,所以我们可以使用它获取调用我们的函数时函数调用堆栈里都有些什么函数,也即是可以了解到当前程序的一个流程是如何的,示例代码如下:

     class  Test
     {
        static void Main()
        {
            int offset = 0;
            do
            {
                StackFrame frame = new StackFrame(offset++);
                MethodBase method = frame.GetMethod();
                if (method == null) break;       //如果偏移位置没有函数时,则GetMethod方法返回null
                Console.WriteLine(method.Name);

            }
 while (true);
            Console.Read();
        }

    }


其实.NET已经为我们提供了一个StackTrace类,其可以获取函数调用堆栈里的所有函数的有序集合,通过它我们就能将上面的代码简化为下面的代码了,如下:

    class  Test
     {
        static void Main()
        {
            StackTrace trace = new StackTrace();
            foreach (StackFrame frame in trace.GetFrames())
            {
                Console.WriteLine(frame.GetMethod().Name);
            }

            Console.Read();
        }

    }


两者输出的结果还是一样的 ,如下:

Main
_nExecuteAssembly
ExecuteAssembly
RunUsersAssembly
ThreadStart_Context
Run
ThreadStart


看来在控制台程序中也是由某个线程委托开始运作的

来自:http://www.cnblogs.com/kingthy/archive/2008/04/19/1160816.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值