.net 日志系统解析

一.   写在前面

日志系统对于任何项目都是必不可少的,无论对于测试阶段的debug,性能测试,执行时间,操作记录还是线上的问题排查,访问记录等,日志系统都扮演着重要的角色。本篇分享的目的是能帮助需要的人快速搭建自己的LogSystem.,仅供参考。 先上个图呗,自认为页面还算清爽吧:

我的LogSystem使用Log4net入库的方式,网上特别多的分享,但是能完整运行下来的真是很少,所以现在需要和以后用得上的小伙伴抓紧收藏咯。

二.  Log4Net自定义内容入库

Log4Net存日志的方式,给人的感觉实在是不实用,IT行业不都求一个自动化吗?废话不说了,先上Log4net入库系统的代码。

LogSystem数据库结构,我的建议是一个项目一个表。

在Log组件中,你需要这样几个类。下面分别给出代码:

LogContent.cs,这里定义了Log实体,在实体化实体的时候,通过给构造函数传参创建好这个对象。注释很详细了

using System;
namespace LogComponent
{
 public class LogContent
 {
 public LogContent(string logLevel, string logMsg, string logModule, string description, string userName)
 {
  LogLevel = logLevel;
  UserName = userName;
  Description = description;
  LogMsg = logMsg;
  LogModule = logModule;
 }
 /// <summary>
 /// 日志级别
 /// </summary>
 public string LogLevel { get; set; }
 /// <summary>
 /// 日志消息
 /// </summary>
 public string LogMsg { get; set; }
 /// <summary>
 /// 系统登陆用户
 /// </summary>
 public string UserName { get; set; }
 /// <summary>
 /// 日志描述信息
 /// </summary>
 public string Description { get; set; }
 /// <summary>
 /// 记录时间
 /// </summary>
 public DateTime LogDate { get; set; }
 /// <summary>
 /// 模块名称
 /// </summary>
 public string LogModule { get; set; }
 }
}

LogHelper.cs,定义了日志级别,和写入方法

[assembly: log4net.Config.XmlConfigurator(Watch = true,ConfigFile = "log4net.config")]
namespace LogComponent
{
 public class LogHelper
 {
 static log4net.ILog log = log4net.LogManager.GetLogger("myLogger");
 /// <summary>
 /// 异常日志
 /// </summary>
 /// <param name="logMsg">日志信息</param>
 /// <param name="logModule">代码模块</param>
 /// <param name="description">其他描述</param>
 /// <param name="userName">用户名</param>
 public static void LogError(string logMsg, string logModule, string description = "", string userName = "")
 {
  log.Error(new LogContent("Error", SubLogString(logMsg), logModule, SubLogString(description), userName));
 }
 public static void LogInfo(string logMsg, string logModule, string description = "", string userName = "")
 {
  log.Info(new LogContent("Info", SubLogString(logMsg), logModule, SubLogString(description), userName));
 }
 public static void LogWarn(string logMsg, string logModule, string description = "", string userName = "")
 {
  log.Warn(new LogContent("Warn", SubLogString(logMsg), logModule, SubLogString(description), userName));
 }
 public static void LogDebug(string logMsg, string logModule, string description = "", string userName = "")
 {
  log.Debug(new LogContent("Debug", SubLogString(logMsg), logModule, SubLogString(description), userName));
 }
 private static string SubLogString(string str)
 {
  if (str.Length > 1500)
  {
  return str.Substring(0, 1500);
  }
  return str;
 }
 }
}

MessagePartternConverter.cs

using log4net.Core;
using log4net.Layout.Pattern;
using System.IO;
using System.Reflection;
namespace LogComponent
{
 class MessagePatternConverter : PatternLayoutConverter
 {
 protected override void Convert(TextWriter writer, LoggingEvent loggingEvent)
 {
  if (Option != null)
  {
  // Write the value for the specified key
  WriteObject(writer, loggingEvent.Repository, LookupProperty(Option, loggingEvent));
  }
  else
  {
  // Write all the key value pairs
  WriteDictionary(writer, loggingEvent.Repository, loggingEvent.GetProperties());
  }
 }
 /// <summary>
 /// 通过反射获取传入的日志对象的某个属性的值
 /// </summary>
 /// <param name="property"></param>
 /// <returns></returns>
 private object LookupProperty(string property, log4net.Core.LoggingEvent loggingEvent)
 {
  object propertyValue = string.Empty;
  PropertyInfo propertyInfo = loggingEvent.MessageObject.GetType().GetProperty(property);
  if (propertyInfo != null)
  propertyValue = propertyInfo.GetValue(loggingEvent.MessageObject, null);
  return propertyValue;
 }
 }
}

MyLayout.cs

using log4net.Layout;
namespace LogComponent
{
 class MyLayout : PatternLayout
 {
 public MyLayout()
 {
  this.AddConverter("property", typeof(MessagePatternConverter));
 }
 }
}

其实看到这里,最重要的并不是代码了,核心部分Log4net都帮我们写好了,关键在于你的配置,下面是log4net.config的内容。拿到你的web项目里是一样用的。但是不要忘了在你的项目中引用nuget:log4net哟。

log4net.config如下:在其中主要配置了log入库的参数和sql语句,当然还有sql连接。注释已经很详细了

<?xml version="1.0" encoding="utf-8" ?>
<configuration>
 <configSections>
 <section name="log4net" type="log4net.Config.Log4NetConfigurationSectionHandler, log4net"/>
 </configSections>
 <log4net>
 <root >
 <level value="Debug"/>
 <appender-ref ref="ADONetAppender"/>
 </root>
 <logger name="myLogger">
 <level value="Debug"/>
 <appender-ref ref="ADONetAppender"/>
 </logger>
 <appender name="ADONetAppender" type="log4net.Appender.ADONetAppender,log4net">
 <!--BufferSize为缓冲区大小,只有日志记录超value条才会一块写入到数据库-->
 <bufferSize value="1"/>
 <!--或写为<param name="BufferSize" value="1" />-->
 <!--引用-->
 <connectionType value="System.Data.SqlClient.SqlConnection, System.Data, Version=1.0.3300.0, Culture=neutral, PublicKeyToken=b77a5c561934e089"/>
 <!--连接数据库字符串-->
 <connectionString value="Data Source=115.29.54.31;Initial Catalog=LogSystem;uid=sa;pwd=sa.;MultipleActiveResultSets=True"/>
 <!--插入到表Log-->
 <commandText value="INSERT INTO HdPubLog ([LogDate],[LogMsg],[UserName],[Description],[LogLevel],[LogModule]) VALUES (@log_date,@LogMsg,@UserName,@Description,@LogLevel,@LogModule)"/>
 <parameter>
 <parameterName value="@log_date"/>
 <dbType value="DateTime"/>
 <layout type="log4net.Layout.RawTimeStampLayout"/>
 <!--获取log4net中提供的日志时间RawTimeStampLayout为默认的时间输出格式-->
 </parameter>
 <parameter>
 <parameterName value="@LogMsg"/>
 <dbType value="String"/>
 <size value="1510"/>
 <layout type="LogComponent.MyLayout, LogComponent">
  <param name="ConversionPattern" value="%property{LogMsg}"/>
 </layout>
 </parameter>
 <parameter>
 <parameterName value="@UserName"/>
 <dbType value="String"/>
 <size value="50"/>
 <layout type="LogComponent.MyLayout, LogComponent">
  <param name="ConversionPattern" value="%property{UserName}"/>
 </layout>
 </parameter>
 <parameter>
 <parameterName value="@Description"/>
 <dbType value="String"/>
 <size value="1510"/>
 <layout type="LogComponent.MyLayout, LogComponent">
  <param name="ConversionPattern" value="%property{Description}"/>
 </layout>
 </parameter>
 <parameter>
 <parameterName value="@LogLevel"/>
 <dbType value="String"/>
 <size value="50"/>
 <layout type="LogComponent.MyLayout, LogComponent">
  <param name="ConversionPattern" value="%property{LogLevel}"/>
 </layout>
 </parameter>
 <parameter>
 <parameterName value="@LogModule"/>
 <dbType value="String"/>
 <size value="50"/>
 <layout type="LogComponent.MyLayout, LogComponent">
  <param name="ConversionPattern" value="%property{LogModule}"/>
 </layout>
 </parameter>
 </appender>
 </log4net>
</configuration>

这样一来,你的配置就完成了,你可以直接测试插入的情况:

三.   把Log信息可视化

我的UI使用的是Datatables.js,弹出框是layer,日期组件好像是layDate,下拉框是修改样式后的select2。UI代码是我自己的一个框架里的,内容太多就不贴出来了,你只需要和以前一样,把数据从库里查出来,绑定给任意你喜欢的数据表格上。由于单页面的日志系统没有什么复杂操作,就用个sqlHelper查一下就算了,代码和条件拼接如下

 public class xxxDal
 {
 private SqlHelper _sqlHelper = new SqlHelper();
 /// <summary>
 /// 获取xxx的日志
 /// </summary>
 /// <param name="model"></param>
 /// <returns></returns>
 public List<LogModel> GetxxxLog(SM_LogModel model)
 {
  StringBuilder sql = new StringBuilder();
  List<SqlParameter> sqlParameters = new List<SqlParameter>();
  StringBuilder sqlWhere = new StringBuilder();
  if (!string.IsNullOrWhiteSpace(model.LogStartTime))
  {
  sqlParameters.Add(new SqlParameter("@LogStartTime", model.LogStartTime));
  sqlWhere.Append(@" AND h.LogDate > @LogStartTime");
  }
  if (!string.IsNullOrWhiteSpace(model.LogEndTime))
  {
  sqlParameters.Add(new SqlParameter("@LogEndTime", model.LogEndTime));
  sqlWhere.Append(@" AND h.LogDate < @LogEndTime");
  }
  if (!string.IsNullOrWhiteSpace(model.LogLevel))
  {
  sqlParameters.Add(new SqlParameter("@LogLevel", model.LogLevel));
  sqlWhere.Append(@" AND h.LogLevel = @LogLevel");
  }
  if (!string.IsNullOrWhiteSpace(model.LogModule))
  {
  sqlParameters.Add(new SqlParameter("@LogModule", model.LogModule));
  sqlWhere.Append(@" AND h.LogModule = @LogModule");
  }
  sql.AppendFormat(@"
   WITH t AS ( SELECT ROW_NUMBER() OVER ( ORDER BY id DESC ) AS IndexNum ,
   [Id] ,
   CONVERT(VARCHAR, [LogDate], 21) AS [LogDate] ,
   [UserName] ,
   SUBSTRING([Description], 0, 150) AS [Description] ,
   SUBSTRING([LogMsg], 0, 200) AS [LogMsg] ,
   [LogLevel] ,
   [LogModule]
  FROM [LogSystem].[dbo].[xxxLog] h
  WHERE 1 = 1
   {0}
  )
 SELECT *
 FROM t
 WHERE IndexNum > @startIndex
  AND indexnum < @endIndex", sqlWhere);
  sqlParameters.Add(new SqlParameter("@startIndex", model.Start));
  sqlParameters.Add(new SqlParameter("@endIndex", model.Start + model.Length));
  DataTable dt = _sqlHelper.ExecuteDataTable(sql.ToString(), sqlParameters.ToArray());
  return DataTableTools<LogModel>.DataTableToList(dt);
 }
 public int GetxxxLogTotalCount(SM_LogModel model)
 {
  StringBuilder sql = new StringBuilder(); List<SqlParameter> sqlParameters = new List<SqlParameter>();
  sql.Append(@"
   SELECT COUNT(*)
   FROM [HdPubLog] h where 1=1 ");
  if (!string.IsNullOrWhiteSpace(model.LogStartTime))
  {
  sqlParameters.Add(new SqlParameter("@LogStartTime", model.LogStartTime));
  sql.Append(@" AND h.LogDate > @LogStartTime");
  }
  if (!string.IsNullOrWhiteSpace(model.LogEndTime))
  {
  sqlParameters.Add(new SqlParameter("@LogEndTime", model.LogEndTime));
  sql.Append(@" AND h.LogDate < @LogEndTime");
  }
  if (!string.IsNullOrWhiteSpace(model.LogLevel))
  {
  sqlParameters.Add(new SqlParameter("@LogLevel", model.LogLevel));
  sql.Append(@" AND h.LogLevel = @LogLevel");
  }
  if (!string.IsNullOrWhiteSpace(model.LogModule))
  {
  sqlParameters.Add(new SqlParameter("@LogModule", model.LogModule));
  sql.Append(@" AND h.LogModule = @LogModule");
  }
  return _sqlHelper.ExecuteScalar<int>(sql.ToString(), sqlParameters.ToArray());
 }
 [HttpPost]
 public LogModel GetxxxxSignelLog(int id)
 {
  string sql = @"
   SELECT [Id] ,
    CONVERT(VARCHAR(30), [LogDate], 21) AS [LogDate] ,
    [UserName] ,
    [Description] ,
    [LogMsg] ,
    [LogLevel] ,
    [LogModule] ,
    [Id] IndexNum
   FROM [LogSystem].[dbo].[xxxxLog] h
   WHERE h.id = @Id";
  var row = _sqlHelper.ExecuteDataRow(sql, new SqlParameter("@Id", id));
  return DataTableTools<LogModel>.DataRowToModel(row);
 }
 }

以上就是本文的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,同时也希望多多支持我们!

时间: 2017-01-18

.NET 日志系统设计思路及实现代码

日志很明显是帮助大家定位到问题的一个很重要的手段,本来是想直接使用的NLog 来做系统的日志工具,哎伤不起,一变态非要说这个有很多不可控制的因素,这里我给大家讲一下我是怎么实现日志模块的,欢迎拍砖 总体架构图 •    在这里我把日子的等级分为 跟踪,BUG 和错误 3种  定义枚举如下 复制代码 代码如下: /// <summary>    /// 日志等级    /// </summary>    public enum Loglevel    {        Track=1

使用shell脚本对Nginx日志进行切分的示例代码

本文介绍了使用shell脚本对Nginx日志进行切分的示例代码,分享给大家,具体如下: 1.日志格式 默认的日志格式: main log_format main '$remote_addr - $remote_user [$time_local] "$request" ' $status $body_bytes_sent "$http_referer" ' '"$http_user_agent" "$http_x_forwarded_f

Python中实现最小二乘法思路及实现代码

之所以说"使用"而不是"实现",是因为python的相关类库已经帮我们实现了具体算法,而我们只要学会使用就可以了.随着对技术的逐渐掌握及积累,当类库中的算法已经无法满足自身需求的时候,我们也可以尝试通过自己的方式实现各种算法. 言归正传,什么是"最小二乘法"呢? 定义:最小二乘法(又称最小平方法)是一种数学优化技术,它通过最小化误差的平方和寻找数据的最佳函数匹配. 作用:利用最小二乘法可以简便地求得未知的数据,并使得这些求得的数据与实际数据之间误

tomcat应用文件备份脚本及日志备份脚本的示例代码

下面先给大家介绍tomcat应用文件备份脚本 #!/bin/bash #Back up the entire tomcat8080-dist to the /backup directory backuppath=/mnt/tomcat8080-dist/webapps/ backpath=/backup/backup-tomcat8080-dist/ action=/backup/backup-tomcat8080-dist/ APPNAME=dist BACKUPNAME1=dist BAC

python实现word文档批量转成自定义格式的excel文档的思路及实例代码

支持按照文件夹去批量处理,也可以单独一个文件进行处理,并且可以自定义标识符 最近在开发一个答题类的小程序,到了录入试题进行测试的时候了,发现一个问题,试题都是word文档格式的,每份有100题左右,拿到的第一份试题,光是段落数目就有800个.而且可能有几十份这样的试题. 而word文档是没有固定格式的,想批量录入关系型数据库mysql,必须先转成excel文档.这个如果是手动一个个粘贴到excel表格,那就头大了. 我最终需要的excel文档结构是这样的:每道题独立占1行,每1列是这道题的一项内

用PHP代替JS玩转DOM的思路及示例代码

事情的起源比较简单,我需要把一个导航页的数据整理好写入数据库.一个比较直观的方法是对html文件进行分析,通用的方法是用php的正则表达式来匹配.但是这样做开发和维护都很困难,代码可读性非常差. 导航页的数据都是规则的排列在DOM树当中的,用JS可以用几个循环轻松的对其进行操作,而且JS需要依赖浏览器,操作数据库很困难.其实PHP就有现成的类库对DOM树种的节点进行增删改查操作,在此做一些笔记. 这里涉及到2个类 DOMDocument 和 DOMXPath. 其实思路比较明确,就是通过DOMD

Javascript图像处理思路及实现代码

思路 HTML5的canvas提供了getImageData接口来获取canvas中的数据,所以我们能够先用drawImage接口将图片画在canvas上然后再通过getImageData得到图片数据矩阵. 需要注意,虽然IE9开始支持了canvas接口,但是其getImageData获取的数据并不是以标准的TypedArray方式存储的,或者说IE9没有提供对WebGL Native binary data的支持,所以如果需要对IE9支持,下面的矩阵需要用Array的方式保存.虽然IE9以下版

将DataTable转换成List&amp;lt;T&amp;gt;实现思路及示例代码

前几天在工作中,遇到一个问题:需要将查询出来的DataTable数据源,转换成List<T>的泛型集合(已知T类型).第一反应,我想肯定要用到"泛型"(这不是废话吗?都说了要转换成List<T>泛型集合了),而且还要用到"反射"相关的.呵呵.很快,我就做出了一个小实例,测试通过.下面我将代码贴出来,分享给大家.代码都有详细的注释,读者朋友可以很清晰的看懂我的思路. 首先,这是我写的一个通用转换类,完成此类操作.也是实现这个功能最核心的部分:

可替代log4j日志的c#简单日志类队列实现类代码分享

复制代码 代码如下: using System;using System.Collections.Generic;using System.Globalization;using System.IO;using System.Linq;using System.Text;using System.Threading; namespace LogisTrac{    /// <summary>    /// 日志类     /// 队列 可年/月/周/日/大小分割    /// 调用方法:   

java异步写日志到文件中实现代码

java异步写日志到文件中详解 实现代码: package com.tydic.ESUtil; import java.io.File; import java.io.FileWriter; import java.io.IOException; import java.io.InputStream; import java.io.PrintWriter; import java.util.Properties; public class LogWriter { // 日志的配置文件 publi