4

我希望从 C# 中的任务中获取方法/操作名称。具体来说,我正在实现一个自定义任务调度程序,并希望生成有关任务运行持续时间的统计信息,然后我将通过任务内部运行的方法对其进行汇总。在 Visual Studio 调试器中,您可以访问它并查看 m_action 私有变量,以及调试器显示注释,将其显示为 Method={0}。有没有办法从任务本身访问这个?

4

3 回答 3

4

您可以从 Task 继承以使这变得非常容易......我将在这里实现第一个构造函数作为示例:

public class NamedTask : Task {
    public string MethodName { get; set; }
    public NamedTask(Action action) : base(action) {
        MethodName = action.Method.Name;
    }
    public NamedTask(Action action, CancellationToken cancellationToken) : base(action, cancellationToken) {}
    public NamedTask(Action action, TaskCreationOptions creationOptions) : base(action, creationOptions) {}
    public NamedTask(Action action, CancellationToken cancellationToken, TaskCreationOptions creationOptions) : base(action, cancellationToken, creationOptions) {}
    public NamedTask(Action<object> action, object state) : base(action, state) {}
    public NamedTask(Action<object> action, object state, CancellationToken cancellationToken) : base(action, state, cancellationToken) {}
    public NamedTask(Action<object> action, object state, TaskCreationOptions creationOptions) : base(action, state, creationOptions) {}
    public NamedTask(Action<object> action, object state, CancellationToken cancellationToken, TaskCreationOptions creationOptions) : base(action, state, cancellationToken, creationOptions) {}
}

在那之后...

NamedTask task = new NamedTask(() => AsyncMethod(arg1, arg2, argN));
string methodName = task.MethodName; // there's the name!

更多示例。继承自Task<T>

public class NamedTask<T> : Task<T> {
    public string MethodName { get; set; }
    public NamedTask(Func<T> function) : base(function) {
        MethodName = function.Method.Name;
    }
    public NamedTask(Func<T> function, string methodName) : base(function) {
        MethodName = methodName;
    }
    ...
}

处理匿名方法:

NamedTask<bool> task2 = new NamedTask<bool>(() => {
                // some arbitrary code
                return true;
    });

NamedTask<bool> task3 = new NamedTask<bool>(() => {
                // some arbitrary code
                return true;
    }, "ReturnTrueMethod");

string methodName2 = task2.MethodName; // returns "<LongRunning_Async>b__19"
string methodName3 = task3.MethodName; // returns "ReturnTrueMethod"
于 2012-08-23T06:22:59.183 回答
3

好吧,给定一个变量,您可以使用反射来获取私有m_action字段:Tasktask

    var fieldInfo = typeof(Task).GetField("m_action", BindingFlags.Instance | BindingFlags.NonPublic);
    Delegate action = fieldInfo.GetValue(task) as Delegate;

然后获取Name方法和DeclaringType

    var name = action.Method.Name;
    var type = action.Method.DeclaringType.FullName;

要获得完全限定的方法 ( type + "." + name)...

但是,一旦任务执行完成,m_action就是null. 我不确定这将如何适用于 TaskFactory.StartNew ...

于 2012-08-23T19:25:46.740 回答
0
  1. 尝试反射以获取 m_action 变量。
  2. 尝试从Envorinment.StackTrace任务内部获取信息或直接调用它的方法。
于 2012-08-23T06:01:35.043 回答