加入收藏 | 设为首页 | 会员中心 | 我要投稿 李大同 (https://www.lidatong.com.cn/)- 科技、建站、经验、云计算、5G、大数据,站长网!
当前位置: 首页 > 百科 > 正文

c# – 如何在通用应用程序中禁用任务并行库的ETW事件源?

发布时间:2020-12-15 22:11:29 所属栏目:百科 来源:网络整理
导读:任务并行库使用 Event Tracing for Windows (ETW)进行日志记录.显然,在Windows Phone或Windows Store .NET Runtime下,存在与TPL或ETW中的日志记录相关的错误.原始问题描述于 here. 一种可能的解决方法是禁用TPL的ETW EventSource. 如果我真的想要如何在通用W
任务并行库使用 Event Tracing for Windows (ETW)进行日志记录.显然,在Windows Phone或Windows Store .NET Runtime下,存在与TPL或ETW中的日志记录相关的错误.原始问题描述于 here.

一种可能的解决方法是禁用TPL的ETW EventSource.

如果我真的想要如何在通用Windows应用程序中禁用它?

反射适用于桌面应用程序,但不适用于WP / WinRT应用程序. EventCommand.Disable未被识别为EventSource.SendCommand的有效命令(尽管它在ETW内部使用).这是要使用的代码(作为控制台应用程序):

using System;
using System.Threading.Tasks;
using System.Diagnostics.Tracing;
using System.Reflection;

namespace ConsoleApplication
{
    class Program
    {
        internal class MyEventListener : EventListener
        {
            protected override void OnEventSourceCreated(EventSource eventSource)
            {
                Console.WriteLine(eventSource);
                base.OnEventSourceCreated(eventSource);
                if (eventSource.Name == "System.Threading.Tasks.TplEventSource")
                {
                    Console.WriteLine("enabled: " + eventSource.IsEnabled());

                    // trying to disable with EventCommand.Disable: Invalid command
                    try
                    {
                        System.Diagnostics.Tracing.EventSource.SendCommand(
                            eventSource,EventCommand.Disable,new System.Collections.Generic.Dictionary<string,string>());
                        Console.WriteLine("enabled: " + eventSource.IsEnabled());
                    }
                    catch (Exception ex)
                    {
                        Console.WriteLine(ex.Message);
                    }

                    // reflection: doesn't work for Windows Phone/Store apps
                    try
                    {
                        var ti = typeof(EventSource).GetTypeInfo();
                        var f = ti.GetDeclaredField("m_eventSourceEnabled");
                        f.SetValue(eventSource,false);
                        Console.WriteLine("enabled: " + eventSource.IsEnabled());
                    }
                    catch (Exception ex)
                    {
                        Console.WriteLine(ex.Message);
                    }
                }
            }

            protected override void OnEventWritten(EventWrittenEventArgs eventData)
            {
                Console.WriteLine(eventData);
            }
        }

        static MyEventListener listener = new MyEventListener();

        static void Main(string[] args)
        {
            Task.Delay(1000).Wait();
            Console.ReadLine();
        }
    }
}

对于通用应用程序,可以将MyEventListener实例化为Application的一部分:

public sealed partial class App : Application
{
    static MyEventListener listener = new MyEventListener();
}

解决方法

我遇到了类似的问题并找到了一个潜在的解决方案.

您只需在EventSource上调用Dispose()即可!

这不会删除事件源,但会禁用它.并且它们都继承自的基本EventSource确实具有适当的检查,以防止在被禁用时调用其余的继承类.所以,从理论上讲,它应该是安全的.但是可能有一些EventSource实现,这将无法正常工作,所以要彻底测试它!

有关EventSource.Dispose(bool disposing)实现,请参见here.

(编辑:李大同)

【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!

    推荐文章
      热点阅读