1 基本概念

Dotnet core 一个重要的特征是 Dependency injection ,中文一般是依赖注入,可以简单理解为一个集合,在应用程序启动时,定义各种具体的实现类型并将其放到集合中;在应用程序运行时,从集合中取出之前放入的类型。

Logging 的实现就采用这种方式,写日志分为两步:创建写日志的对象;用创建的对象写日志。ILoggerProvider 创建写日志的对象 ILogger ,即在应用程序启动时,把实现了 ILoggerProvider 接口的类型放到集合中,在应用程序运行期间,需要写日志,先去集合中取 ILoggerProvider ,使用其创建 ILogger 对象,然后就可以写日志了。

1.1 ILogger

写日志的定义:

void Log<TState>(LogLevel logLevel, EventId eventId, TState state, Exception exception, Func<TState, Exception, string> formatter);

除了 logLevel 有点熟,其它都感觉无用,只是我们并没有直接使用,并不是其真的无用。

1.2 ILoggerProvider

创建写日志对接定义:

ILogger CreateLogger(string categoryName);

通过定义可以知道,所有的 ILogger 对象都有 categoryName 属性。属性值不能为 Null。

到这里其实可以写日志了,也仅只是写日志而已,还有两个比较重要的问题:1 写的日志在哪里可以看?2 如何管理写日志?

问题1 是由 ILoggerProvider 接口的实现来决定,官方的实现有:Console 、Debug 、EventSource 、EventLog 、TraceSource 、Azure App Service,还有一些第三方实现,如此我们自己也是可以实现的。

例如 ConsoleLoggerProvider 是将日志在命令行中打印出来,如果是控制台程序或者是 self host 启动 web 程序便可以在命令行中看到打印的日志信息。

问题2由 log Level、EventId、Logger Provider、Log filtering、Log category、Log scopes 合作解决。好像是很复杂,其实比较简单。

1.3 Log Level

定义如下,数值越大级别越高。:

Trace = 0

Debug = 1

Information = 2

Warning = 3

Error = 4

Critical = 5

None = 6

例如:

_logger.LogWarning(LoggingEvents.GetItemNotFound, "GetById({ID}) NOT FOUND", id);

日志的 logLevel 是 Warning ,eventId 是的 GetItemNotFound,内容是“GetById({ID}) NOT FOUND”,这个对应接口定义中 formatter 的返回值 string。

2 log filtering 说明

为了说明日志过滤,先说明下日志的整体框架,在应用程序启动时,可以注册多个不同的 ILoggerProvider 实现,而在写日志时,只调用一次写,但所有注册的 ILoggerProvider 都能收到写的日志,这背后是由框架实现的。具体的简单点说,我们调用 log()后,它再遍历所有注册的 ILoggerProvider ,再分别调用它们的 log()方法。

这样在一个正常的系统中会有多个 ILoggerProvider 实现,而每个 ILoggerProvider 创建的 ILogger 又有一个类别 category;也就是说在调用log()写日志时,ILogger 对象具有 Provider 和 Category,这点很重要!

而过滤便是针对 Provider 和 Category 设置的。

2.1 过滤类型

可以为某个 Provider 的某个 Category 设置一条过滤规则,也可以针对所有的Provider 和所有的Category 设置一条过滤规则。

这样过滤规则的类型也就可以如下所示:

表1:

Provider 限定

Provider 不限定

Category 限定

1

3

Category 不限定

2

4

2.2 过滤规则

(1)minimum level

最低级别的意思是框架只会将高于(包括等于)此级别的日志传递给 Provider,进而 Provider 调用 log()方法处理日志。

(2)Filter functions

Func<string, LogLevel, bool> categoryLevelFilter

这是委托签名,入参是类别 Category 名称和写日志的级别,返回是 bool 类型,也就是说可以根据类别和消息级别来决定是否写日志。

2.3 最佳过滤规则

至此,我可以可以添加一条过滤规则,其签名如下:

LoggerFilterRule(string providerName, string categoryName, LogLevel? logLevel, Func<string, string, LogLevel, bool> filter)

比如:.AddFilter<ConsoleLoggerProvider>("System", LogLevel.Critical) 便等同于:

LoggerFilterRule("ConsoleLoggerProvider", "System", LogLevel.Critical,null)

如果我们添加了多条过滤规则,但每个 Provider 只能对应一条过滤规则,这时应当使用以下规则:

(1)1>2>3>4 (参考表1)

(2)同一等级后注册高于先注册

(3)无可用过滤规则,则不过滤

补充,在设置过滤规则时可以同时设置 minimum level 和 Filter functions,先执行minimum level,如有必要再执行 Filter functions。

在匹配类别 Category 时,不是字符串相同比较,是字符串前缀包含即可。

正常的系统,对象间关系简单示意如下:

图1

3 配置示例

设置过滤规则有两种方式:代码和配置文件

3.1 代码

ILoggingBuilder AddFilter<T>(this ILoggingBuilder builder, string category, LogLevel level)

还有一些类似的签名,T 是 Provider 类型,可选, category 必选。可以实现表1中:1、2、3类型设置,4类型没有找到(应该是没有提供)

3.2 配置文件

{

"Logging": {

"LogLevel": {

"Default": "Trace",

"Microsoft": "Critical"

},

"Console": {

"LogLevel": {

"Default": "Debug",

"System": "Warning"

}

}

}

}

这是一实例配置,从上往下定义了4个过来规则,其对应的表1中的级别分别是:4、3、2、1

4 实践一下

(1)注册 Provider

(2)注册 Filter

(3)获取 ILogger

(4)写日志

我没有贴代码,请参考下面的git中的SampleApp。

5 参考

https://docs.microsoft.com/en-us/aspnet/core/fundamentals/logging/?view=aspnetcore-2.1

https://github.com/aspnet/Logging

Asp.net core logging 日志的更多相关文章

  1. 如何利用Serilog的RequestLogging来精简ASP.NET Core的日志输出

    这是该系列的第一篇文章:在ASP.NET Core 3.0中使用Serilog.AspNetCore. 第1部分-使用Serilog RequestLogging来简化ASP.NET Core的日志输 ...

  2. Elasticsearch,Kibana,Logstash,NLog实现ASP.NET Core 分布式日志系统

    Elasticsearch - 简介 Elasticsearch 作为核心的部分,是一个具有强大索引功能的文档存储库,并且可以通过 REST API 来搜索数据.它使用 Java 编写,基于 Apac ...

  3. ASP.NET Core Logging in Elasticsearch with Kibana

    在微服务化盛行的今天,日志的收集.分析越来越重要.ASP.NET Core 提供了一个统一的,轻量级的Logining系统,并可以很方便的与第三方日志框架集成.我们也可以根据不同的场景进行扩展,因为A ...

  4. ASP.NET Core学习日志1

    1.ASP.NET进行了结构化的优化,使框架更为精简,模块化更加明显. 2.ASP.NET Core不再基于System.Web.dll,而是基于细粒度.分解的NuGet包. 3.基础特性: 1.We ...

  5. ASP.NET Core Logging Solution

    Serilog.Extensions.Logging.File This package makes it a one-liner - loggerFactory.AddFile() - to con ...

  6. ASP.NET Core分布式日志系统ELK实战演练

    一.ELK简介  ELK是Elasticsearch.Logstash和Kibana首字母的缩写.这三者均是开源软件,这三套开源工具组合起来形成了一套强大的集中式日志管理平台. •  Elastics ...

  7. ASP.NET Core可视化日志组件使用

    前言 今天站长推荐一款日志可视化组件LogDashboard,可以不用安装第三方进程,只需要在项目中安装相应的Nuget包,添加数行代码,就可以实现拥有带Web页面的日志管理面板,十分nice哦. 下 ...

  8. ASP.NET Core 添加日志NLog

    1.在Nuget上搜索 NLog.Extensions.Logging 安装最新版 2.添加日志配置文件,在项目指定目录下添加配置文件nlog.config,内容添加如下: <?xml vers ...

  9. Asp.net Core 异常日志与API返回值处理

    需求: 1.对异常进行捕获记录日志 并且修改返回值给前端 解释: ILogger4是自定义的一个日志,更改它就好 解决方案1: 使用中间件进行异常捕获并且修改其返回值 public class Err ...

随机推荐

  1. C++类学习(2)

    Ⅰ:类概念 一:类的构成 class 类名 { public: 公有数据成员和成员函数:类的接口 protected: 保护数据成员和成员函数: private: 私有数据成员和成员函数: }://注 ...

  2. 网络流--最大流--Dinic模板矩阵版(当前弧优化+非当前弧优化)

    //非当前弧优化版 #include <iostream> #include <cstdio> #include <math.h> #include <cst ...

  3. 虚拟 IP 设为静态 IP

    一:虚拟机设置桥接模式 1.进入虚拟机设置中将网络适配器设置成桥接模式 2.编辑--虚拟网络编辑器--选择桥接 二:将虚拟IP设置成静态IP (1)方案一:进入虚拟机系统 System 设置 (2)方 ...

  4. 一只简单的网络爬虫(基于linux C/C++)————开篇

    最近学习开发linux下的爬虫,主要是参考了该博客及其他一些网上的资料.网络爬虫(又被称为网页蜘蛛,网络机器人,在FOAF社区中间,更经常的称为网页追逐者),是一种按照一定的规则,自动地抓取万维网信息 ...

  5. 【Java8新特性】你知道Java8为什么要引入Lambda表达式吗?

    写在前面 这是一道真实的面试题,一个读者朋友出去面试,面试官竟然问他这样一个问题:你说说Java8中为什么引入Lambda表达式?引入Lambda表达式后有哪些好处呢?还好这个朋友对Java8早有准备 ...

  6. redis系列之3----redis高级应用(主从、事务与锁、持久化)

    文章主目录 安全性设置 主从复制 事务与锁 持久化机制 发布以及订阅消息 上文<详细讲解redis数据结构(内存模型)以及常用命令>介绍了redis的数据类型以及常用命令,本文我们来学习下 ...

  7. Hadoop入门学习笔记-第二天 (HDFS:NodeName高可用集群配置)

    说明:hdfs:nn单点故障,压力过大,内存受限,扩展受阻.hdfs ha :主备切换方式解决单点故障hdfs Federation联邦:解决鸭梨过大.支持水平扩展,每个nn分管一部分目录,所有nn共 ...

  8. dp cf 20190615

    A. Timofey and a tree 这个不算是dp,就是一个思维题,好难想的思维题,看了题解才写出来的, 把点和边分开,如果一条边的两个点颜色不同就是特殊边,特殊边两边连的点就叫特殊点, 如果 ...

  9. Python+Typora博客图片上传

    简介 用Typora 写Markdown 1年多了,这个编辑器的确很好用,但就是写完博客复制到博客园时要一个个手动插替换图片非常麻烦.后来发现最新版的Typora 已经支持图片上传功能,在 设置-图像 ...

  10. 简单谈谈Spring的IoC

    一.前言   这几天正在复习Spring的相关内容,同时想要对Spring的实现原理做一些深入的研究.今天看了看Spring中IoC的实现,找到了一篇非常详细的博客,研究了一个下午,看完之后唯一的感受 ...