我们要在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

最新文章

  1. jQuery入门(2)使用jQuery操作元素的属性与样式
  2. zk系列-zookeeper概述
  3. 白话Https
  4. PHP和JS实现多按钮提交表单
  5. How to evaluate a transimpedance amplifier (part 2)
  6. dedecms的特性-----不完整
  7. opengl基础学习专题 (二) 点直线和多边形
  8. AutoCAD.NET二次开发:创建自定义菜单的两种方法比较
  9. gcc,g++,extern “C” :一些编译错误的缘由
  10. mongodb授权登录
  11. Linux 高性能server编程——高级I/O函数
  12. RocketMQ源码 — 一、 quikstart
  13. 【MySQL源码】源码安装和启动mysql
  14. 使用WinDbg获取SSDT函数表对应的索引再计算得出地址
  15. JS变量声明方式
  16. C. Books Queries
  17. map的使用方式之一。
  18. loj#6491. zrq 学反演
  19. 20155217《网络对抗》Exp07 网络欺诈防范
  20. Maven发布war包到Tomcat

热门文章

  1. leetcode 之Remove Duplicates from Sorted Array(1)
  2. POJ 1218 THE DRUNK JAILER(类开灯问题,完全平方数)
  3. IEnumerable的几个简单用法
  4. 一张图解AlphaGo原理及弱点
  5. CCTF部分赛题分析
  6. 洛谷 P1202 [USACO1.1]黑色星期五Friday the Thirteenth 题解
  7. linux 命令route add default dev eth0和route add default gw eth0的区别?
  8. &lt;&lt;Javascript Patterns&gt;&gt;阅读笔记 -- 第2章 基本技巧(二)
  9. Linux Shell 文本处理工具
  10. 《Android虚拟机》----Android系统的结构