.NET 项目默认情况下 日志是使用的 ILogger 接口,默认提供一下四种日志记录程序:

  • 控制台
  • 调试
  • EventSource
  • EventLog

这四种记录程序都是默认包含在 .NET 运行时库中。关于这四种记录程序的详细介绍可以直接查看微软的官方文档 https://docs.microsoft.com/zh-cn/dotnet/core/extensions/logging-providers

今天给大家分享自己实现一个日志记录程序,继承自  ILogger 接口,实现将日志记录到本地的 txt 文件中,并包含一个自动清理过期日志的功能任务。

类库的整体代码结构如下:

Models 文件夹中存放 LoggerSetting.cs 是 该模块注入服务时需要的配置参数

namespace Logger.LocalFile.Models
{
public class LoggerSetting
{
/// <summary>
/// 保存天数
/// </summary>
public int SaveDays { get; set; } = 7;
}
}

Tasks 文件夹中存放的 LogClearTask.cs 是用于自动清理过期日志的任务,会在日志服务注入的同时启动,会通过配置的保存天数参数,定期删除超过实现的日志文件

using Common;
using Logger.LocalFile.Models;
using Microsoft.Extensions.Hosting;
using Microsoft.Extensions.Options; namespace Logger.LocalFile.Tasks
{
public class LogClearTask : BackgroundService
{ private readonly int saveDays; public LogClearTask(IOptionsMonitor<LoggerSetting> config)
{
saveDays = config.CurrentValue.SaveDays;
} protected override async Task ExecuteAsync(CancellationToken stoppingToken)
{
while (!stoppingToken.IsCancellationRequested)
{
try
{ string basePath = Directory.GetCurrentDirectory().Replace("\\", "/") + "/Logs/"; if (Directory.Exists(basePath))
{
List<string> logPaths = IOHelper.GetFolderAllFiles(basePath).ToList(); var deleteTime = DateTime.UtcNow.AddDays(-1 * saveDays); if (logPaths.Count != 0)
{
foreach (var logPath in logPaths)
{
var fileInfo = new FileInfo(logPath); if (fileInfo.CreationTimeUtc < deleteTime)
{
File.Delete(logPath);
} }
}
} }
catch
{
} await Task.Delay(1000 * 60 * 60 * 24, stoppingToken);
}
} }
}

ILoggingBuilderExtensions 是应用注入服务的扩展方法,内容如下

using Logger.LocalFile.Models;
using Logger.LocalFile.Tasks;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Hosting;
using Microsoft.Extensions.Logging; namespace Logger.LocalFile
{ public static class ILoggingBuilderExtensions
{ public static void AddLocalFileLogger(this ILoggingBuilder builder, Action<LoggerSetting> action)
{
builder.Services.Configure(action);
builder.Services.AddSingleton<ILoggerProvider, LocalFileLoggerProvider>();
builder.Services.AddSingleton<IHostedService, LogClearTask>();
}
}
}

LocalFileLogger 是日志的保存执行方法,内容如下

using Common;
using Microsoft.Extensions.Logging;
using System.Text; namespace Logger.LocalFile
{
public class LocalFileLogger : ILogger
{
private readonly string categoryName;
private readonly string basePath; public LocalFileLogger(string categoryName)
{
this.categoryName = categoryName; basePath = Directory.GetCurrentDirectory().Replace("\\", "/") + "/Logs/"; if (Directory.Exists(basePath) == false)
{
Directory.CreateDirectory(basePath);
}
} public IDisposable BeginScope<TState>(TState state)
{
return default!;
} public bool IsEnabled(LogLevel logLevel)
{
if (logLevel != LogLevel.None)
{
return true;
}
else
{
return false;
}
} public void Log<TState>(LogLevel logLevel, EventId eventId, TState state, Exception? exception, Func<TState, Exception?, string> formatter)
{
if (IsEnabled(logLevel))
{
if (state != null && state.ToString() != null)
{
var logContent = state.ToString(); if (logContent != null)
{
if (exception != null)
{
var logMsg = new
{
message = logContent,
error = new
{
exception?.Source,
exception?.Message,
exception?.StackTrace
}
}; logContent = JsonHelper.ObjectToJson(logMsg);
} var log = new
{
CreateTime = DateTime.UtcNow,
Category = categoryName,
Level = logLevel.ToString(),
Content = logContent
}; string logStr = JsonHelper.ObjectToJson(log); var logPath = basePath + DateTime.UtcNow.ToString("yyyyMMddHH") + ".log"; File.AppendAllText(logPath, logStr + Environment.NewLine, Encoding.UTF8); }
}
}
}
}
}

LocalFileLoggerProvider 是Logger执行方法向外部的供应者,内容如下:

using Microsoft.Extensions.Logging;
using System.Collections.Concurrent; namespace Logger.LocalFile
{
public class LocalFileLoggerProvider : ILoggerProvider
{
private readonly ConcurrentDictionary<string, LocalFileLogger> loggers = new(); public ILogger CreateLogger(string categoryName)
{
return loggers.GetOrAdd(categoryName, new LocalFileLogger(categoryName));
} public void Dispose()
{
loggers.Clear();
GC.SuppressFinalize(this);
}
}
}

当我们其他项目想要使用我们这个 Logger.LocalFile 类库时,只要添加该类库的引用,然后在启动服务时进行注入即可,注入方法如下:

Web 项目注入方式

//注册本地文件日志服务
builder.Logging.AddLocalFileLogger(options => { options.SaveDays = 7; });

控制台项目注入方式

.ConfigureLogging((hostContext, builder) =>
{
//注册本地文件日志服务
builder.AddLocalFileLogger(options => { options.SaveDays = 7; });
})
.Build();

Web 项目直接在 builder 后面编写注入就可以,控制台项目需要先 .ConfigureLogging 才可以,这是两者的区别。

这样就注入了我们自己编写的日志记录程序,项目运行时会在项目的 Logs 文件夹中产生日志文件,如下图

至此 .NET 扩展 官方 Logger 实现将日志保存到本地文件就讲解完了,有任何不明白的,可以在文章下面评论或者私信我,欢迎大家积极的讨论交流,有兴趣的朋友可以关注我目前在维护的一个 .net 基础框架项目,项目地址如下

.NET 扩展 官方 Logger 实现将日志保存到本地文件的更多相关文章

  1. C#中使用Log4net日志输出到本地文件、Textbox或Listview

    网上很多配置log4net的方法,但是排行靠前的 根本就没有说明清除,导致浪费了两个小时来搞清楚如何配置,真是无语,特写此文,给那些刚接触log4net的朋友 1.参考链接:http://blog.s ...

  2. Java读取oracle数据库中blob字段数据文件保存到本地文件(转载)

    转自:https://www.cnblogs.com/forever2698/p/4747349.html package com.bo.test; import java.io.FileOutput ...

  3. Python3.4 获取百度网页源码并保存在本地文件中

    最近学习python 版本 3.4 抓取网页源码并且保存在本地文件中 import urllib.request url='http://www.baidu.com' #上面的url一定要写明确,如果 ...

  4. 【Jmeter】jmeter提取response中的返回值,并保存到本地文件--BeanShell后置处理器

    有个需求,需要在压测环境中,创建几十万的账号数据,然后再根据创建结果,查询到某些账号信息. 按照之前我的做法,直接Python调用API,然后再数据库查询: 但是近期所有开发人员的数据库访问权限被限制 ...

  5. Android View转为图片保存为本地文件,异步监听回调操作结果;

    把手机上的一个View或ViewGroup转为Bitmap,再把Bitmap保存为.png格式的图片: 由于View转Bitmap.和Bitmap转图片都是耗时操作,(生成一个1M的图片大约500ms ...

  6. winform程序生成条形码并且并且保存到本地文件中。

    今天公司让做一个输入数字.字母生成条形码并且可以以图片格式保存到本地.当看到这个需求时候感觉很搞笑,明明可以用文本框搞定的东西非得做个程序.哎,寄人篱下,不多说了,这就是养兵千日用兵一时. 我在网上找 ...

  7. C#中将String类型保存到本地文件的方法

    今天刚刚遇到,要在WinForm中把一个xml保存到本地, 由于之前是学习java的,一时间还真不知道怎么写, 没想到C#居然那么方便,就3句代码就实现了我要的功能: StreamWriter sw ...

  8. 将nosetests的echo结果保存到本地文件

    nose是很好用的python 测试框架. 但是一直很纠结如何将结果保存到本地.采用nosetests -h查看相关的options,找到一个xunit的东西,似乎可以实现功能. 测试结果: 可见,已 ...

  9. C# 中从网络上下载文件保存到本地文件

    下面是C#中常用的从Internet上下载文件保存到本地的一些方法,没有太多的技巧. 1.通过  WebClient  类下载文件 WebClient webClient = new WebClien ...

随机推荐

  1. 每天一个 HTTP 状态码 205

    205 Reset Content 205 Reset Content 表示服务器成功地处理了客户端的请求,要求客户端重置它发送请求时的文档视图.这个响应码跟 204 No Content 类似,也不 ...

  2. Netty是什么,Netty为什么速度这么快,线程模型分析

    哈喽!大家好,我是小奇,一位热爱分享的程序员 小奇打算以轻松幽默的对话方式来分享一些技术,如果你觉得通过小奇的文章学到了东西,那就给小奇一个赞吧 文章持续更新 一.前言 书接上回,现在下着大雨看来是去 ...

  3. 图的连通性--Tarjan算法

    一些概念 无向图: 连通图:在无向图中,任意两点都直接或间接连通,则称该图为连通图.(或者说:任意两点之间都存在可到达的路径) 连通分量: G的 最大连通子图 称为G的连通分量. 有向图 (ps.区别 ...

  4. 【freertos】009-任务控制

    目录 前言 9.1 相对延时 9.1.1 函数原型 9.1.2 函数说明 9.1.3 参考例子 9.2 绝对延时 9.2.1 函数原型 9.2.2 函数说明 9.2.3 参考例子 9.3 获取任务优先 ...

  5. 关于spring整合mybatis

    第一步导入依赖 <dependencies> <dependency> <groupId>org.mybatis</groupId> <artif ...

  6. SPFA 最短路算法

    SPFA算法 1.什么是spfa算法? SPFA 算法是 Bellman-Ford算法 的队列优化算法的别称,通常用于求含负权边的单源最短路径,以及判负权环.SPFA一般情况复杂度是O(m)O(m) ...

  7. 论文解读(MGAE)《MGAE: Masked Autoencoders for Self-Supervised Learning on Graphs》

    论文信息 论文标题:MGAE: Masked Autoencoders for Self-Supervised Learning on Graphs论文作者:Qiaoyu Tan, Ninghao L ...

  8. 基于.NetCore开发博客项目 StarBlog - (12) Razor页面动态编译

    系列文章 基于.NetCore开发博客项目 StarBlog - (1) 为什么需要自己写一个博客? 基于.NetCore开发博客项目 StarBlog - (2) 环境准备和创建项目 基于.NetC ...

  9. v-if和v-for哪个优先级更高?

    首先在实际开发阶段,不应该把v-if和v-for在同一个标签中使用, 在vue2中,v-for的优先级是高于v-if的,如果同时出现,每次渲染都会先执行循环再判断条件,无论如何循环都不可避免,浪费了性 ...

  10. Windows下maven配置环境变量

    右键 "计算机",选择 "属性",之后点击 "高级系统设置",点击"环境变量",来设置环境变量,有以下系统变量需要配置: ...