如何获取方法的参数 NAMES 。这些示例显示了如何获取参数的值,而不是 NAMES 。我想看到parma = 99,parmb = 1.不仅仅是99,1。
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Diagnostics;
using PostSharp.Aspects;
namespace GettingParmNames
{
public class Program
{
static void Main(string[] args)
{
Foo myfoo = new Foo();
int sum = myfoo.DoA(99, 1);
Console.WriteLine(sum.ToString());
Console.ReadKey();
}
}
public class Foo
{
[ExceptionAspect]
public int DoA(int parma, int parmb)
{
int retVal;
try
{
retVal = parma + parmb;
if (parma == 99)
{
throw new Exception("Fake Exception");
}
}
catch (Exception ex)
{
retVal = -1;
throw new Exception("There was a problem");
}
return retVal;
}
}
[Serializable]
public class ExceptionAspect : OnExceptionAspect
{
public override void OnException(MethodExecutionArgs args)
{
string parameterValues = "";
foreach (object arg in args.Arguments)
{
if (parameterValues.Length > 0)
{
parameterValues += ", ";
}
if (arg != null)
{
parameterValues += arg.ToString();
}
else
{
parameterValues += "null";
}
}
Console.WriteLine("Exception {0} in {1}.{2} invoked with arguments {3}", args.Exception.GetType().Name, args.Method.DeclaringType.FullName, args.Method.Name, parameterValues );
}
}
}
答案 0 :(得分:5)
您可以通过调用OnException
在args.Method.GetParameters()
方法中访问方法参数信息。但是出于性能原因,通常最好在编译期间初始化数据 - 在CompileTimeInitialize方法覆盖中。
[Serializable]
public class ExceptionAspect : OnExceptionAspect
{
private string[] parameterNames;
public override void CompileTimeInitialize(MethodBase method, AspectInfo aspectInfo)
{
parameterNames = method.GetParameters().Select(p => p.Name).ToArray();
}
public override void OnException(MethodExecutionArgs args)
{
StringBuilder parameterValues = new StringBuilder();
for (int i = 0; i < args.Arguments.Count; i++)
{
if ( parameterValues.Length > 0 )
{
parameterValues.Append(", ");
}
parameterValues.AppendFormat(
"{0} = {1}", parameterNames[i], args.Arguments[i] ?? "null");
}
// ...
}
答案 1 :(得分:0)
AlexD的答复很好。
以下是类似的方法,为您提供参数名称和值。
using System;
using System.Linq;
using System.Reflection;
using PostSharp.Aspects;
[Serializable]
public class LogWhenInvoked : MethodInterceptionAspect
{
private string[] parameterNames;
public override void CompileTimeInitialize(MethodBase method, AspectInfo aspectInfo)
{
parameterNames = method.GetParameters().Select(p => p.Name).ToArray();
}
public override void OnInvoke(MethodInterceptionArgs invocationArgs)
{
Console.WriteLine($"Entering {invocationArgs.Method.DeclaringType.Name}.{invocationArgs.Method.Name}");
var args = parameterNames.Select((param, index) => new
{
Name = param,
Value = invocationArgs.Arguments[index]
})
.ToList();
invocationArgs.Proceed();
}
}