亚洲激情专区-91九色丨porny丨老师-久久久久久久女国产乱让韩-国产精品午夜小视频观看

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

.Net?Core日志記錄器如何實現

發布時間:2022-10-21 10:03:59 來源:億速云 閱讀:98 作者:iii 欄目:開發技術

這篇文章主要介紹“.Net Core日志記錄器如何實現”的相關知識,小編通過實際案例向大家展示操作過程,操作方法簡單快捷,實用性強,希望這篇“.Net Core日志記錄器如何實現”文章能幫助大家解決問題。

    日志記錄器工廠

     ILoggerFactory 接口
    public interface ILoggerFactory : IDisposable
    {
        ILogger CreateLogger(string categoryName);
    
        void AddProvider(ILoggerProvider provider);
    }

    ILoggerFactory是日志記錄器的工廠接口類,用于配置日志記錄系統并創建Logger實例的類,默認實現兩個接口方法為,通過CreateLogger()方法來創建ILogger實例,(其中參數categoryName是一個日志類別,用于調用Logger所在類的全名,類別指明日志消息是誰寫入的,一般我們將日志所屬的的組件、服務或者消息類型名稱作為日志類別。) 而AddProvider()添加日志記錄提供程序,向日志系統注冊添加一個ILoggerProvider。工廠接口類的默認實現類為LoggerFactory , 我們繼續往下看:

    LoggerFactory 實現

    ILoggerFactory 的默認實現是 LoggerFactory ,在構造函數中,如下:

        public class LoggerFactory : ILoggerFactory
        {
            private static readonly LoggerRuleSelector RuleSelector = new LoggerRuleSelector();
    
            private readonly Dictionary<string, Logger> _loggers = new Dictionary<string, Logger>(StringComparer.Ordinal);
            
            private readonly List<ProviderRegistration> _providerRegistrations = new List<ProviderRegistration>();
            
            private readonly object _sync = new object();
            
            private volatile bool _disposed;
            
            private IDisposable _changeTokenRegistration;
            
            private LoggerFilterOptions _filterOptions;
            
            private LoggerExternalScopeProvider _scopeProvider;
            
            public LoggerFactory() : this(Enumerable.Empty<ILoggerProvider>())
            {
            }
            
            public LoggerFactory(IEnumerable<ILoggerProvider> providers) : this(providers, new StaticFilterOptionsMonitor(new LoggerFilterOptions()))
            {
            }
            
            public LoggerFactory(IEnumerable<ILoggerProvider> providers, LoggerFilterOptions filterOptions) : this(providers, new StaticFilterOptionsMonitor(filterOptions))
            {
            }
            
            public LoggerFactory(IEnumerable<ILoggerProvider> providers, IOptionsMonitor<LoggerFilterOptions> filterOption)
            {
                foreach (var provider in providers)
                {
                    AddProviderRegistration(provider, dispose: false);
                }
    
                _changeTokenRegistration = filterOption.OnChange(RefreshFilters);
                RefreshFilters(filterOption.CurrentValue);
            }
            
            private void AddProviderRegistration(ILoggerProvider provider, bool dispose)
            {
                _providerRegistrations.Add(new ProviderRegistration
                {
                    Provider = provider,
                    ShouldDispose = dispose
                });
    
                if (provider is ISupportExternalScope supportsExternalScope)
                {
                    if (_scopeProvider == null)
                    {
                        _scopeProvider = new LoggerExternalScopeProvider();
                    }
    
                    supportsExternalScope.SetScopeProvider(_scopeProvider);
                }
            }
        }

    LoggerFactory 中 的構造函數中可以發現,通過注入的方式獲取到ILoggerProvider(這個在下文中會說明),并調用AddProviderRegistration方法添加注冊程序,將ILoggerProvider保存到ProviderRegistration集合中。

    AddProviderRegistration 方法:

    這是一個日志程序提供器,將ILoggerProvider保存到ProviderRegistration集合中。當日志提供器實現 ISupportExternalScope 接口將單例 LoggerExternalScopeProvider 保存到 provider._scopeProvider 中。

    ProviderRegistration集合:

    private struct ProviderRegistration
    {
       public ILoggerProvider Provider;
       public bool ShouldDispose;
    }

    其中的 ShouldDispose 字段標識在在LoggerFactory生命周期結束之后,該ILoggerProvider是否需要釋放。雖然在系統中LoggerFactory為單例模式,但是其提供了一個靜態方法生成一個可釋放的DisposingLoggerFactory

    LoggerFactory 實現默認的接口方法CreateLogger(),AddProvider()

    查看源碼如下:

    CreateLogger

    創建ILogger實例,CreateLogger() 源碼如下:

        public class LoggerFactory : ILoggerFactory
        { 
            private readonly Dictionary<string, Logger> _loggers = new Dictionary<string, Logger>(StringComparer.Ordinal);
            
             private readonly List<ProviderRegistration> _providerRegistrations = new List<ProviderRegistration>();
            
            private struct ProviderRegistration
            {
                public ILoggerProvider Provider;
                public bool ShouldDispose;
            }
            
            public ILogger CreateLogger(string categoryName)
            {
                if (CheckDisposed())
                {
                    throw new ObjectDisposedException(nameof(LoggerFactory));
                }
                lock (_sync)
                {
                    if (!_loggers.TryGetValue(categoryName, out var logger))
                    {
                        logger = new Logger
                        {
                            Loggers = CreateLoggers(categoryName),
                        };
                        (logger.MessageLoggers, logger.ScopeLoggers) = ApplyFilters(logger.Loggers);
    
                        _loggers[categoryName] = logger;
                    }
                    return logger;
                }
            }
            
            private LoggerInformation[] CreateLoggers(string categoryName)
            {
                var loggers = new LoggerInformation[_providerRegistrations.Count];
                for (var i = 0; i < _providerRegistrations.Count; i++)
                {
                    loggers[i] = new LoggerInformation(_providerRegistrations[i].Provider, categoryName);
                }
                return loggers;
            }
        }

    從源碼可以看出,CreateLogger方法中,會檢測資源是否被釋放,在方法中,根據內部定義的字典集合Dictionary<string, Logger> _loggers,判斷字典中是否存在對應的Logger屬性對象,如果不存在,會調用CreateLoggers方法根據之前注冊的的所有ILoggerProvider 所創建出來 ProviderRegistration 集合來實現創建Logger屬性集合(根據日志類別生成了對應實際的日志寫入類FileLoggerConsoleLogger等),并通過字典集合的方式保存categoryName和對應的Logger

    創建 Logger 需要的 LoggerInformation[]

    internal readonly struct LoggerInformation
    {
        public LoggerInformation(ILoggerProvider provider, string category) : this()
        {
            ProviderType = provider.GetType();
            Logger = provider.CreateLogger(category);
            Category = category;
            ExternalScope = provider is ISupportExternalScope;
        }
    
        public ILogger Logger { get; }
        
        public string Category { get; }
        
        public Type ProviderType { get; }
        
        public bool ExternalScope { get; }
    }

    根據注冊的ILoggerProvider,創建ILogger 其中的字段說明:

    Logger :具體日志類別寫入途徑實現類

    Category : 日志類別名稱

    ProviderType : 日志提供器Type

    ExternalScope :是否支持 ExternalScope

    繼續看CreateLogger方法,在創建Logger之后,還調用了ApplyFilters方法:

            private (MessageLogger[] MessageLoggers, ScopeLogger[] ScopeLoggers) ApplyFilters(LoggerInformation[] loggers)
            {
                var messageLoggers = new List<MessageLogger>();
                var scopeLoggers = _filterOptions.CaptureScopes ? new List<ScopeLogger>() : null;
    
                foreach (var loggerInformation in loggers)
                {
                    RuleSelector.Select(_filterOptions,
                        loggerInformation.ProviderType,
                        loggerInformation.Category,
                        out var minLevel,
                        out var filter);
    
                    if (minLevel != null && minLevel > LogLevel.Critical)
                    {
                        continue;
                    }
    
                    messageLoggers.Add(new MessageLogger(loggerInformation.Logger, loggerInformation.Category, loggerInformation.ProviderType.FullName, minLevel, filter));
    
                    if (!loggerInformation.ExternalScope)
                    {
                        scopeLoggers?.Add(new ScopeLogger(logger: loggerInformation.Logger, externalScopeProvider: null));
                    }
                }
    
                if (_scopeProvider != null)
                {
                    scopeLoggers?.Add(new ScopeLogger(logger: null, externalScopeProvider: _scopeProvider));
                }
    
                return (messageLoggers.ToArray(), scopeLoggers?.ToArray());
            }

    由源碼可以看出,

    MessageLogger[] 集合取值:

    在獲取LoggerInformation[]后進行傳參,進行遍歷,根據RuleSelector過濾器,從配置文件中讀取對應的日志級別,過濾器會返回獲取最低級別和對應的一條過濾規則,如果配置文件中沒有對應的配置,默認取全局最低級別(MinLevel),如果讀取到的日志級別大于LogLevel.Critical,則將其加入MessageLogger[]

    過濾器的規則:

    選擇當前記錄器類型的規則,如果沒有,請選擇未指定記錄器類型的規則

    選擇最長匹配類別的規則

    如果沒有與類別匹配的內容,則采用所有沒有類別的規則

    如果只有一條規則,則使用它的級別和過濾器

    如果有多個規則,請選擇使用最后一條。

    如果沒有適用的規則,請使用全局最低級別

    通過MessageLogger[]添加消息日志集合

    internal readonly struct MessageLogger
    {
        public MessageLogger(ILogger logger, string category, string providerTypeFullName, LogLevel? minLevel, Func<string, string, LogLevel, bool> filter)
        {
            Logger = logger;
            Category = category;
            ProviderTypeFullName = providerTypeFullName;
            MinLevel = minLevel;
            Filter = filter;
        }
    
        public ILogger Logger { get; }
    
        public string Category { get; }
    
        private string ProviderTypeFullName { get; }
    
        public LogLevel? MinLevel { get; }
    
        public Func<string, string, LogLevel, bool> Filter { get; }
    
        public bool IsEnabled(LogLevel level)
        {
            if (MinLevel != null && level < MinLevel)
            {
                return false;
            }
    
            if (Filter != null)
            {
                return Filter(ProviderTypeFullName, Category, level);
            }
    
            return true;
        }
    }
    
    internal readonly struct ScopeLogger
    {
        public ScopeLogger(ILogger logger, IExternalScopeProvider externalScopeProvider)
        {
            Logger = logger;
            ExternalScopeProvider = externalScopeProvider;
        }
    
        public ILogger Logger { get; }
    
        public IExternalScopeProvider ExternalScopeProvider { get; }
    
        public IDisposable CreateScope<TState>(TState state)
        {
            if (ExternalScopeProvider != null)
            {
                return ExternalScopeProvider.Push(state);
            }
            return Logger.BeginScope<TState>(state);
        }
    }

    MessageLogger[]中帶有MinLevel屬性和Filter委托兩種過濾配置,而這兩種配置的來源,在上一章中可以看到,分別是從配置文件(AddConfiguration)和直接使用委托(AddFilter)來進行配置的。

    再由上面的IsEnabled方法可以看出,會先使用 MinLevel 過濾,再使用 Filter 進行過濾。所以這兩者存在優先級。

    ScopeLogger[ ] 取值 :

    如果 ILoggerProvider實現了ISupportExternalScope接口,那么使用LoggerExternalScopeProvider作為Scope功能的實現。反之,使用ILogger作為其Scope功能的實現。

    LoggerExternalScopeProvider :

    • 通過 Scope 組成了一個單向鏈表,每次 beginscope 向鏈表末端增加一個新的元素,Dispose的時候,刪除鏈表最末端的元素。我們知道LoggerExternalScopeProvider 在系統中是單例模式,多個請求進來,加入線程池處理。通過使用AsyncLoca來實現不同線程間數據獨立。

    • 有兩個地方開啟了日志作用域:

    • 1、通過socket監聽到請求后,將KestrelConnection加入線程池,線程池調度執行IThreadPoolWorkItem.Execute()方法。在這里開啟了一次

    • 2、在構建請求上下文對象的時候(HostingApplication.CreateContext()),開啟了一次

    由上源碼可以得出:在工廠記錄器類中,通過系統依賴注入的方式解析所有注冊的ILoggerProvider,然后調用其中的CreateLogger方法實現創建一個Logger實例對象,而這個Logger實例對象會根據根據注冊的ILoggerProvider創建需要的 LoggerInformation[],并將此對象作為參數進行ApplyFilters過濾器篩選,得到對應的最低等級或過濾規則,最后通過調用Log方法日志記錄的時候,會遍歷MessageLogger[]集合,根據logger日志類別對應實際不同的日志寫入類,調用ILoggerProvider具體實現類 (可以看下文說明) 中的Log方法。

    AddProviderRegistration&rarr;CreateLoggers&rarr;LoggerInformation[]&rarr;ApplyFilters&rarr;MessageLogger[]&rarr;Log&rarr;ILoggerProvider ( 執行具體類中的Log方法 )

    ILoggerFactory 來源

    在上一篇中我們在對日志配置進行說明的時候,應用程序在啟動初始化的時候會通過注入的方式CreateDefaultBuilder&rarr;ConfigureLogging&rarr;AddLogging

    public static IServiceCollection AddLogging(this IServiceCollection services, Action<ILoggingBuilder> configure)
    {
        if (services == null)
        {
           throw new ArgumentNullException(nameof(services));
        }
        
        services.AddOptions();
        services.TryAdd(ServiceDescriptor.Singleton<ILoggerFactory, LoggerFactory>());
        services.TryAdd(ServiceDescriptor.Singleton(typeof(ILogger<>), typeof(Logger<>)));
        
        services.TryAddEnumerable(ServiceDescriptor.Singleton<IConfigureOptions<LoggerFilterOptions>>(
           new DefaultLoggerLevelConfigureOptions(LogLevel.Information)));
        
        configure(new LoggingBuilder(services));
        return services;
    }

    實現將把ILoggerFactory對象以依賴注入的方式托管到集合容器中,為程序調用提供使用。

    日志記錄提供器

    ILoggerProvider 接口

    創建ILogger實例的類型,根據日志類別名稱創建一個新的ILogger實例

    public interface ILoggerProvider : IDisposable
    {
        ILogger CreateLogger(string categoryName);
    }

    這個是具體的日志寫入類,在工廠記錄器中我們已經提到了這個,在LoggerInformation[]中會根據日志類別注冊對應的ILoggerProvider,在系統中我們就可以通過ILogger同時向多個途經寫入日志信息。(這也是對上一篇中留下的問題進行再次說明)

    ILoogerProvider繼承了IDisposable接口,如果某個具體的ILoggerProvider對象需要釋放資源,就可以將相關的操作實現在Dispose方法中。

    默認的實現方式為多個,官方實現的由ConsoleLoggerProvider 、DebugLoggerProvider 、EventSourceLoggerProviderEventLogLoggerProvider TraceSourceLoggerProvider

    ConsoleLoggerProvider為列

        [ProviderAlias("Console")]
        public class ConsoleLoggerProvider : ILoggerProvider, ISupportExternalScope
        {
            private readonly IOptionsMonitor<ConsoleLoggerOptions> _options;
            private readonly ConcurrentDictionary<string, ConsoleLogger> _loggers;
            private readonly ConsoleLoggerProcessor _messageQueue;
    
            private IDisposable _optionsReloadToken;
            private IExternalScopeProvider _scopeProvider = NullExternalScopeProvider.Instance;
     
            public ConsoleLoggerProvider(IOptionsMonitor<ConsoleLoggerOptions> options)
            {
                _options = options;
                _loggers = new ConcurrentDictionary<string, ConsoleLogger>();
    
                ReloadLoggerOptions(options.CurrentValue);
                _optionsReloadToken = _options.OnChange(ReloadLoggerOptions);
    
                _messageQueue = new ConsoleLoggerProcessor();
                if (RuntimeInformation.IsOSPlatform(OSPlatform.Windows))
                {
                    _messageQueue.Console = new WindowsLogConsole();
                    _messageQueue.ErrorConsole = new WindowsLogConsole(stdErr: true);
                }
                else
                {
                    _messageQueue.Console = new AnsiLogConsole(new AnsiSystemConsole());
                    _messageQueue.ErrorConsole = new AnsiLogConsole(new AnsiSystemConsole(stdErr: true));
                }
            }
            private void ReloadLoggerOptions(ConsoleLoggerOptions options)
            {
                foreach (var logger in _loggers)
                {
                    logger.Value.Options = options;
                }
            }
            public ILogger CreateLogger(string name)
            {
                return _loggers.GetOrAdd(name, loggerName => new ConsoleLogger(name, _messageQueue)
                {
                    Options = _options.CurrentValue,
                    ScopeProvider = _scopeProvider
                });
            } 
            public void Dispose()
            {
                _optionsReloadToken?.Dispose();
                _messageQueue.Dispose();
            }
     
            public void SetScopeProvider(IExternalScopeProvider scopeProvider)
            {
                _scopeProvider = scopeProvider;
    
                foreach (var logger in _loggers)
                {
                    logger.Value.ScopeProvider = _scopeProvider;
                }
    
            }
        }

    ConsoleLoggerProvider類型定義中,標注了ProviderAliasAttribute特性,并設置別名為Console,所以在配置過濾規則的時候,可以直接使用這個名稱。ILogger的創建實現了具體日志類ConsoleLogger。 

    日志記錄器

    ILogger 接口

    表示用于執行日志記錄的類型,是系統中寫入日志的統一入口。

    public interface ILogger
    { 
        void Log<TState>(LogLevel logLevel, EventId eventId, TState state, Exception exception, Func<TState, Exception, string> formatter); 
        
        bool IsEnabled(LogLevel logLevel); 
        
        IDisposable BeginScope<TState>(TState state);
    }

    定義了三個方法,Log<TState>() 用于寫入日志,IsEnabled()用于檢查判斷日志級別是否開啟,BeginScope() 用于指日志作用域。

    Logger 實現

    ILogger執行記錄接口類的具體實現Logger如下:

    internal class Logger : ILogger
    {
        public LoggerInformation[] Loggers { get; set; }
        public MessageLogger[] MessageLoggers { get; set; }
        public ScopeLogger[] ScopeLoggers { get; set; }
    
        public void Log<TState>(LogLevel logLevel, EventId eventId, TState state, Exception exception, Func<TState, Exception, string> formatter)
        {
            var loggers = MessageLoggers;
            if (loggers == null)
            {
                return;
            }
    
            List<Exception> exceptions = null;
            for (var i = 0; i < loggers.Length; i++)
            {
                ref readonly var loggerInfo = ref loggers[i];
                if (!loggerInfo.IsEnabled(logLevel))
                {
                    continue;
                }
    
                LoggerLog(logLevel, eventId, loggerInfo.Logger, exception, formatter, ref exceptions, state);
            }
    
            if (exceptions != null && exceptions.Count > 0)
            {
                ThrowLoggingError(exceptions);
            }
    
            static void LoggerLog(LogLevel logLevel, EventId eventId, ILogger logger, Exception exception, Func<TState, Exception, string> formatter, ref List<Exception> exceptions, in TState state)
            {
                try
                {
                    logger.Log(logLevel, eventId, state, exception, formatter);
                }
                catch (Exception ex)
                {
                    if (exceptions == null)
                    {
                        exceptions = new List<Exception>();
                    }
    
                    exceptions.Add(ex);
                }
            }
        }
        public bool IsEnabled(LogLevel logLevel)
        {
            var loggers = MessageLoggers;
            if (loggers == null)
            {
                return false;
            }
    
            List<Exception> exceptions = null;
            var i = 0;
            for (; i < loggers.Length; i++)
            {
                ref readonly var loggerInfo = ref loggers[i];
                if (!loggerInfo.IsEnabled(logLevel))
                {
                    continue;
                }
    
                if (LoggerIsEnabled(logLevel, loggerInfo.Logger, ref exceptions))
                {
                    break;
                }
            }
    
            if (exceptions != null && exceptions.Count > 0)
            {
                ThrowLoggingError(exceptions);
            }
    
            return i < loggers.Length ? true : false;
    
            static bool LoggerIsEnabled(LogLevel logLevel, ILogger logger, ref List<Exception> exceptions)
            {
                try
                {
                    if (logger.IsEnabled(logLevel))
                    {
                        return true;
                    }
                }
                catch (Exception ex)
                {
                    if (exceptions == null)
                    {
                        exceptions = new List<Exception>();
                    }
    
                    exceptions.Add(ex);
                }
    
                return false;
            }
        }
    }

    源碼中MessageLogger[]在上文已經提到了,其中保存了在配置中啟用的那些對應的ILogger

    需要注意的是,由于配置文件更改后,會調用ApplyFilters()方法,并為MessageLogger[]賦新值,所以在遍歷之前,需要保存當前值,再進行處理。否則會出現修改異常。

    在系統中統一寫入日志的入口,通過日志等級作為參數調用其IsEnabled方法來確定當前日志是否執行對應具體日志的實現類,當符合條件執行具體日志輸出到對應的寫入途徑中會調用對應的Log方法(需要提供一個EventId來標識當前日志事件)

    ILogger默認的實現方式為多個,官方實現的由ConsoleLogger 、DebugLogger 、EventSourceLoggerEventLogLoggerTraceSourceLogger具體日志實現類代表不同的日志寫入途徑。

    關于“.Net Core日志記錄器如何實現”的內容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業相關的知識,可以關注億速云行業資訊頻道,小編每天都會為大家更新不同的知識點。

    向AI問一下細節

    免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

    AI

    定西市| 淮南市| 辽源市| 富民县| 天津市| 大方县| 九龙坡区| 凤台县| 宜兰市| 扶沟县| 建平县| 民丰县| 博爱县| 贵州省| 林西县| 汝阳县| 昌都县| 额敏县| 鹤岗市| 高邮市| 京山县| 剑阁县| 马尔康县| 离岛区| 睢宁县| 桐乡市| 静海县| 宁城县| 临夏县| 孟津县| 鄂托克旗| 定西市| 阳谷县| 历史| 平乐县| 逊克县| 镇坪县| 宁都县| 木里| 鲁山县| 德阳市|