<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
回顧:紀錄檔記錄之紀錄檔設定揭祕
在上一篇中,我們已經瞭解了內建系統的預設設定和自定義設定的方式,在學習了設定的基礎上,我們進一步的對紀錄檔在程式中是如何使用的深入瞭解學習。所以在這一篇中,主要是對紀錄檔記錄的核心機制進行學習說明。
在上一篇中,我們留下了兩個問題
紀錄檔記錄的輸出可以在哪裡檢視?而又由什麼實現決定的呢?
如何管理輸出不同的紀錄檔呢?都有哪些方式呢?
第一個問題:在官方的實現有:Console 、Debug 、EventSource 、EventLog 、TraceSource 、Azure App Service,還有一些第三方實現,當然了我們自己也是可以實現的。 是由ILoggerProvider
介面來決定實現的。
第二個問題:由 log Level、EventId、Logger Provider、Log filtering、Log category、Log scopes 合作解決。
由上面的問題可以發現,我們可以實現多種不同的輸出目標方式來實現寫紀錄檔記錄,但是又如何控制在寫紀錄檔這個操作不變的情況下,實現不同的輸入目標,這個時候我們就會想到,可以通過抽象的方式,將寫紀錄檔這個操作動作抽象出來,而輸出目標依賴這個動作實現具體的操作。所以當我們呼叫寫紀錄檔操作方法的時候,由此依次呼叫對應的具體實現方法,把紀錄檔寫到具體的目標上。
這個過程具體是怎麼實現的呢?我們接著往下看。
其實在學習之前,我們應該都已經瞭解.net core框架有一個重要的特徵就是依賴注入,通過在應用啟動時候,將各種定義好的實現型別放入到一個集合容器中,通過在執行時,將從集合容器中取出放入對應的型別中。
紀錄檔記錄的的實現方式也離不開這個。下面讓我們一起來看看。
public interface ILoggerFactory : IDisposable { ILogger CreateLogger(string categoryName); void AddProvider(ILoggerProvider provider); }
ILoggerFactory
是紀錄檔記錄器的工廠介面類,用於設定紀錄檔記錄系統並建立Logger範例的類,預設實現兩個介面方法為,通過CreateLogger()
方法來建立ILogger
範例,(其中引數categoryName
是一個紀錄檔類別,用於呼叫Logger
所在類的全名,類別指明紀錄檔訊息是誰寫入的,一般我們將紀錄檔所屬的的元件、服務或者訊息型別名稱作為紀錄檔類別。) 而AddProvider()
新增紀錄檔記錄提供程式,向紀錄檔系統註冊新增一個ILoggerProvider
。工廠介面類的預設實現類為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()
檢視原始碼如下:
建立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
屬性集合(根據紀錄檔類別生成了對應實際的紀錄檔寫入類FileLogger
、ConsoleLogger
等),並通過字典集合的方式儲存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→CreateLoggers→LoggerInformation[]→ApplyFilters→MessageLogger[]→Log→ILoggerProvider ( 執行具體類中的Log方法 )
ILoggerFactory
來源:在上一篇中我們在對紀錄檔設定進行說明的時候,應用程式在啟動初始化的時候會通過注入的方式
CreateDefaultBuilder
→ConfigureLogging
→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
物件以依賴注入的方式託管到集合容器中,為程式呼叫提供使用。
建立ILogger
範例的型別,根據紀錄檔類別名稱建立一個新的ILogger
範例
public interface ILoggerProvider : IDisposable { ILogger CreateLogger(string categoryName); }
這個是具體的紀錄檔寫入類,在工廠記錄器中我們已經提到了這個,在LoggerInformation[]
中會根據紀錄檔類別註冊對應的ILoggerProvider
,在系統中我們就可以通過ILogger
同時向多個途經寫入紀錄檔資訊。(這也是對上一篇中留下的問題進行再次說明)
ILoogerProvider
繼承了IDisposable
介面,如果某個具體的ILoggerProvider
物件需要釋放資源,就可以將相關的操作實現在Dispose
方法中。
預設的實現方式為多個,官方實現的由ConsoleLoggerProvider
、DebugLoggerProvider
、EventSourceLoggerProvider
、EventLogLoggerProvider
、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
。
表示用於執行紀錄檔記錄的型別,是系統中寫入紀錄檔的統一入口。
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()
用於指紀錄檔作用域。
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
、EventSourceLogger
、EventLogLogger
、TraceSourceLogger
具體紀錄檔實現類代表不同的紀錄檔寫入途徑。
ILoggerFactory
和ILoggerProvider
中都會通過方法建立ILogger物件,但兩者是不相同的。在工廠預設實現LoggerFactory
型別中它建立的ILogger物件是由註冊到LoggerFactory
物件上的所有ILoggerProvider物件提供一組 ILogger物件組合而成。而紀錄檔提供器ILoggerProvider
建立的ILogger是紀錄檔實現輸出到對應的渠道目標,寫入紀錄檔。ILogger
中的Log()方法會記錄執行紀錄檔,在紀錄檔記錄器工廠ILoggerFactory
和紀錄檔記錄提供器ILoggerProvider
中兩種不同的ILogger
實現對應的Log()
方法實現的意思也是不同的。在ILoggerFactory
產生的是ILogger
型別(也就是我們最終使用的Logger
),其Log()方法是依次呼叫Logger
中包含的LoggerInformation[]
陣列中的ILogger
。而ILoggerProvider
產生的為各類不同的XxxLogger(也就是上面說的Logger
中的LoggerInformation
陣列包含的如ConsoleLogger、DebugLogger
),其Log()方法是把紀錄檔寫到具體的目標上去。ILoggerProvider
整合到Logger
中,實現自己需要的紀錄檔記錄輸出方式。到此這篇關於.Net Core紀錄檔記錄核心機制的文章就介紹到這了。希望對大家的學習有所幫助,也希望大家多多支援it145.com。
相關文章
<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
综合看Anker超能充系列的性价比很高,并且与不仅和iPhone12/苹果<em>Mac</em>Book很配,而且适合多设备充电需求的日常使用或差旅场景,不管是安卓还是Switch同样也能用得上它,希望这次分享能给准备购入充电器的小伙伴们有所
2021-06-01 09:31:42
除了L4WUDU与吴亦凡已经多次共事,成为了明面上的厂牌成员,吴亦凡还曾带领20XXCLUB全队参加2020年的一场音乐节,这也是20XXCLUB首次全员合照,王嗣尧Turbo、陈彦希Regi、<em>Mac</em> Ova Seas、林渝植等人全部出场。然而让
2021-06-01 09:31:34
目前应用IPFS的机构:1 谷歌<em>浏览器</em>支持IPFS分布式协议 2 万维网 (历史档案博物馆)数据库 3 火狐<em>浏览器</em>支持 IPFS分布式协议 4 EOS 等数字货币数据存储 5 美国国会图书馆,历史资料永久保存在 IPFS 6 加
2021-06-01 09:31:24
开拓者的车机是兼容苹果和<em>安卓</em>,虽然我不怎么用,但确实兼顾了我家人的很多需求:副驾的门板还配有解锁开关,有的时候老婆开车,下车的时候偶尔会忘记解锁,我在副驾驶可以自己开门:第二排设计很好,不仅配置了一个很大的
2021-06-01 09:30:48
不仅是<em>安卓</em>手机,苹果手机的降价力度也是前所未有了,iPhone12也“跳水价”了,发布价是6799元,如今已经跌至5308元,降价幅度超过1400元,最新定价确认了。iPhone12是苹果首款5G手机,同时也是全球首款5nm芯片的智能机,它
2021-06-01 09:30:45