.NET 中获取调用方法名
在写记录日志功能时,需要记录日志调用方所在的模块名、命名空间名、类名以及方法名,想到使用的是反射(涉及到反射请注意性能),但具体是哪一块儿还不了解,于是搜索,整理如下:
需要添加相应的命名空间:
using System;
using System.Diagnostics;
using System.Reflection;
如果仅是获取当前方法名使用 MethodBase.GetCurrentMethod 方法(返回表示当前正在执行的方法的 MethodBase 对象),可以使用如下代码:
public static void WriteSysLog(int level, string content)
{
MethodBase mb = MethodBase.GetCurrentMethod();
string systemModule = Environment.NewLine;
systemModule += "模块名:" + mb.Module.ToString() + Environment.NewLine;
systemModule += "命名空间名:" + mb.ReflectedType.Namespace + Environment.NewLine;
//完全限定名,包括命名空间
systemModule += "类名:" + mb.ReflectedType.FullName + Environment.NewLine;
systemModule += "方法名:" + mb.Name; Console.WriteLine("LogDate: {0}{1}Level: {2}{1}systemModule: {3}{1}content: {4}", DateTime.Now, Environment.NewLine, level, systemModule, content);
Console.WriteLine();
}
本文地址:http://www.cnblogs.com/Interkey/p/GetMethodName.html
但一般情况下是获取此记录日志方法的调用方,因此需要使用下面的代码:(此方法仅为演示)
public static void WriteSysLog(string content)
{
const int level = ; StackTrace ss = new StackTrace(true);
//index:0为本身的方法;1为调用方法;2为其上上层,依次类推
MethodBase mb = ss.GetFrame().GetMethod(); //本文地址:http://www.cnblogs.com/Interkey/p/GetMethodName.html
StackFrame[] sfs = ss.GetFrames();
string systemModule = Environment.NewLine;
systemModule += "模块名:" + mb.Module.ToString() + Environment.NewLine;
systemModule += "命名空间名:" + mb.DeclaringType.Namespace + Environment.NewLine;
//仅有类名
systemModule += "类名:" + mb.DeclaringType.Name + Environment.NewLine;
systemModule += "方法名:" + mb.Name; Console.WriteLine("LogDate: {0}{1}Level: {2}{1}systemModule: {3}{1}content: {4}", DateTime.Now, Environment.NewLine, level, systemModule, content);
Console.WriteLine();
}
对于这一点儿,感觉有意思的是Main的调用方,System.AppDomain._nExecuteAssembly(Assembly assembly, String[] args)。
通过
StackTrace ss = new StackTrace(true);
StackFrame[] sfs = ss.GetFrames();
可以得知.NET程序的执行顺序:
- System.Threading.ThreadHelper.ThreadStart()
- System.Threading.ExecutionContext.Run(ExecutionContext executionContext, ContextCallback callback, Object state)
- Microsoft.VisualStudio.HostingProcess.HostProc.RunUsersAssembly()
- System.AppDomain._nExecuteAssembly(Assembly assembly, String[] args)
然后进入方法Main中。
StackFrame指的是一个.net运行的时候堆栈上的一个帧(Frame),每次进入一个方法的时候就会有一个新的方法帧压入线程执行堆栈,可以通过StackFrame获取相关的信息,比如当前代码所在文件名和行号。
拓展方法是在.NET Framework 3.5以上版本才支持的,需要组件System.Core。
相关类:
- 在System.Diagnostics 命名空间中的 StackTrace 类 ,表示一个堆栈跟踪,它是一个或多个堆栈帧的有序集合。
- 在System.Diagnostics 命名空间中的 StackFrame 类 ,提供关于 StackFrame(表示当前线程的调用堆栈中的一个函数调用)的信息。
- 在System.Reflection 命名空间中的 MethodBase 类 ,提供有关方法和构造函数的信息。
另外,从 MethodBase 类 还可以获取很多其他属性,可以自行定位到System.Reflection.MethodBase 查看。
使用反射可以遍历获得类的所有属性名,方法名,成员名,其中一个有趣的小例子:通过反射将变量值转为变量名本身
文件:GetCurrentMethodName.zip [VS版本为2013,本例仅为演示]
后续:
感谢博友的提醒,使用CallerMemberNameAttribute 也可以实现获取调用方法名。示例请参考:调用方信息(C# 和 Visual Basic)。
C# 5.0 给我们带来了三个非常有用的编译器特性:CallerMemberNameAttribute、CallerFilePathAttribute、CallerLineNumberAttribute (在System.Runtime.CompilerServices 命名空间中),但这三个Attribute 是定义.NET Framework 4.5里的。
如果在低于.NET4.5中使用需要定义这些特性:
namespace System.Runtime.CompilerServices
{
[AttributeUsage(AttributeTargets.Parameter, AllowMultiple = false, Inherited = false)]
public class CallerMemberNameAttribute : Attribute { } [AttributeUsage(AttributeTargets.Parameter, AllowMultiple = false, Inherited = false)]
public class CallerFilePathAttribute : Attribute { } [AttributeUsage(AttributeTargets.Parameter, AllowMultiple = false, Inherited = false)]
public class CallerLineNumberAttribute : Attribute { }
}
关于 Attribute 和 AttributeUsageAttribute 的使用可自行搜索,有些偏题了…
参考:
- Logging method name in .NET
- c# 获取命名空间 类名 方法名
- 得到当前正在运行的方法或属性名[C#]
- C#如何获取被调用的方法的名称及参数列表?
- 获得当前程序的 空间名.类名.方法名
- C#获得调用方法的名称和类名
- 浅析StackTrace
- C# 5.0 CallerMemberName CallerFilePath CallerLineNumber 在.NET4中的使用
.NET 中获取调用方法名的更多相关文章
- 在IE中,JS方法名和input的name重名时,调用该方法无效
在IE中,JS方法名和input的name重名时,调用该方法无效.提示:网页错误详细信息 用户代理: Mozilla/4.0 (compatible; MSIE 8.0; Windows NT 6.1 ...
- delphi中获取调用堆栈信息
异常堆栈有利于分析程序的错误,Delphi的Exception有StackTrace属性,但是值为空,因为StackTrace的信息收集Delphi委托给了第三方组件来完成,真是脑子有毛病! 借助于m ...
- Java中返回类型方法名
继承父类,子类含有两个分别为boy.Girl类名. 返回是需要返回方法 则返回变量名Person class Person { void eat() {} void speak() {} } clas ...
- JAVA中获取当前运行的类名,方法名,行数
JAVA中获取当前运行的类名,方法名,行数 public static String getTraceInfo(){ StringBuffer sb = new StringBuffer(); Sta ...
- C# 5.0中使用CallerMemberName、CallerFilePath和CallerLineNumber获取代码的调用方信息(转载)
很多时候,我们需要在运行过程中记录一些调测的日志信息,如下所示: public void DoProcessing() { TraceMessage("DoProcessing()被XXX调 ...
- Java中获取运行代码的类名、方法名
以下是案例,已运行通过 package com.hdys; /* * 1.获取当前运行代码的类名,方法名,主要是通过java.lang.StackTraceElement类 * * 2. * [1]获 ...
- ThinkPHP 3.2 中获取所有函数方法名,以及注释,完整可运行
<?php namespace Home\Controller; use Common\Controller\BaseController; class AuthController exten ...
- js中获取方法名
var tmp = arguments.callee.toString(); var re = /function\s*(\w*)/i; var matches = re.exec(tmp);//方法 ...
- [No000085]C#反射Demo,通过类名(String)创建类实例,通过方法名(String)调用方法
using System; using System.Collections.Generic; using System.Linq; using System.Reflection; using Sy ...
随机推荐
- 虚拟化平台cloudstack(6)——使用maven:jetty调试
调试环境 ubuntu 12.04 JDK1.7 apache-maven-3.10 eclipse 4.2 Juno mysql 5 apache ant JDK的配置和安装 安装可以参考: htt ...
- Git学习笔记(7)——多人协作
本文主要记录了,多人协作时,产生冲突时的解决情况. 多人环境创建 首先我们需要模拟一个多人环境.前面的Git的学习都是在Ubuntu上面,现在我们也搭建一个win环境吧.安装win环境下的Git,很简 ...
- Oracle建表脚本记录
--删除 drop table dianfei; --创建表 create table dianfei ( uon ) not null, mmonth ) not null, ddf ,) not ...
- 每天一个linux命令(27):linux chmod命令
chmod命令用于改变linux系统文件或目录的访问权限.用它控制文件或目录的访问权限.该命令有两种用法.一种是包含字母和操作符表达式的文字设定法:另一种是包含数字的数字设定法. Linux系统中的每 ...
- HTML基础笔记-02
---恢复内容开始--- 学习网站:W3School 一.HTML的认识 纯文本语言:只显示内容,不显示样式,也不能描述语义的文档,但是也不会乱码 语义:数据的含义就是语义,数据是符号,在这表示标签 ...
- Javascript快速入门(上篇)
Javascript的熟练之路,小弟来了. JavaScript简介:JavaScript一种直译式脚本语言,是一种动态类型.弱类型.基于原型的语言,内置支持类型.它的解释器被称为JavaScript ...
- webpack学习笔记
1.安装webpack npm install webpack -g 2.进入项目目录,初始化 npm init 3.将webpack安装到项目依赖中 npm install webpack --sa ...
- 前端工程师技能之photoshop巧用系列第四篇——图片格式
× 目录 [1]图片格式 [2]保存设置 前面的话 对于前端来说,图片格式是需要重要掌握的知识.本文是photoshop巧用系列第四篇——图片格式 图片格式 目前在前端的开发中常用的图片格式有jpg. ...
- Linux:文件权限
Linux:文件权限 1.文件权限 ☆文件所有者 ☆用户组 ☆其他人 ☆ROOT 说明: Linux系统中默认所有系统上的账号与一般身份用户,还有那个root的相关信息记录在/etc/passwd文件 ...
- Azure ARM (6) ARM Template简单介绍
<Windows Azure Platform 系列文章目录> Azure ARM (1) 概览 Azure ARM (2) 概览 Azure ARM (3) ...