在MVC或WEBAPI中记录每个Action的执行时间和记录下层方法调用时间
刚才在博客园看了篇文章,http://www.cnblogs.com/cmt/p/csharp_regex_timeout.html 突然联想到以前遇到的问题,w3wp进程吃光CPU都挂起IIS进程,即使网站或者服务出现了问题,导致CPU百分之百,但是有时候我们根本不知道具体哪里出了问题!公司曾经一个项目组遇到过WCF服务端莫名其妙就CPU占用百分之百的情况,但是具体有查不到原因。我总结了下,查不到原因是因为没有合理的方法,根本不知道从何开始查。如果当时对客户端或者服务端方法调用都加上日志,估计很快就查到了是哪个方法出了问题。
如果在在MVC或者WEBAPI中,记录每个action的执行时间,后期用来做一些分析,我们会发现哪些调用是执行耗时是比较久的,下面分享下我们在项目中是如何做的。
记录每个action的执行时间,这个需要点技巧。记录下层调用的时间也需要一些技巧的,最开始我想用动态代理来实现,后来发现这个东西不是那么的友好,后来换了种思路去实现。
首先介绍MVC和WEBAPI如何记录action的执行时间
MVC的实现
using System.Diagnostics;
using System.Globalization;
using System.Web.Mvc; namespace WebSeat.Web.Core
{
public class TimingActionFilter : IActionFilter
{
private const string Key = "__action_duration__"; public void OnActionExecuting(ActionExecutingContext filterContext)
{
var stopWatch = new Stopwatch();
stopWatch.Start();
//filterContext.HttpContext.Request.Params[Key] = stopWatch;
filterContext.HttpContext.Items[Key] = stopWatch;
} public void OnActionExecuted(ActionExecutedContext filterContext)
{
var items = filterContext.HttpContext.Items;
if (!items.Contains(Key))
{
return;
}
var stopWatch = items[Key] as Stopwatch;
if (stopWatch == null)
{
return;
}
stopWatch.Stop();
var actionName = filterContext.ActionDescriptor.ActionName;
var controllerName = filterContext.ActionDescriptor.ControllerDescriptor.ControllerName;
#if DEBUG
Debug.WriteLine("Execute {0}- {1} took {2}.]", controllerName, actionName, stopWatch.ElapsedMilliseconds);
#endif var response = filterContext.HttpContext.Response;
if (response != null)
{
response.AddHeader("TotalSpendTimes",
stopWatch.ElapsedMilliseconds.ToString(CultureInfo.InvariantCulture));
}
}
}
}
最后在Application_Start中注册一下
FilterConfig.RegisterGlobalFilters(GlobalFilters.Filters); public class FilterConfig
{
public static void RegisterGlobalFilters(GlobalFilterCollection filters)
{
filters.Add(new HandleErrorAttribute());
filters.Add(new TimingActionFilter());
}
}
WEBAPI中的实现
using System.Diagnostics;
using System.Globalization;
using System.Web.Http.Controllers;
using System.Web.Http.Filters; namespace WebSeat.FlipCourse.WebApi.Resolver.Filter
{
public class TimingActionFilter : ActionFilterAttribute
{
private const string Key = "__action_duration__";
public override void OnActionExecuting(HttpActionContext actionContext)
{
var stopWatch = new Stopwatch();
actionContext.Request.Properties[Key] = stopWatch;
stopWatch.Start();
} public override void OnActionExecuted(HttpActionExecutedContext actionExecutedContext)
{
if (!actionExecutedContext.Request.Properties.ContainsKey(Key))
{
return;
}
var stopWatch = actionExecutedContext.Request.Properties[Key] as Stopwatch;
if (stopWatch == null)
{
return;
}
stopWatch.Stop();
var actionDescriptor = actionExecutedContext.ActionContext.ActionDescriptor;
var actionName = actionDescriptor.ActionName;
var controllerName = actionDescriptor.ControllerDescriptor.ControllerName;
#if DEBUG
Debug.WriteLine("Execute {0}- {1} took {2}.]", controllerName, actionName, stopWatch.ElapsedMilliseconds);
//可直接记录到日志中
#endif
if (actionExecutedContext.Response != null)
{
actionExecutedContext.Response.Content.Headers.Add("TotalSpendTimes", stopWatch.ElapsedMilliseconds.ToString(CultureInfo.InvariantCulture));
}
}
}
}
上面记录Controller里面action的执行时间,action其实调用了底层方法,其实也就记录了整个调用过程的耗时。但是如果我们的程序没有被MVC或者WEBAPI呢?比如我们只是用WCF来提供服务,此时没有了Filter,这就麻烦了。但由于我们项目一直采用分层架构,上层肯定会调用下层,下层返回相应的数据,并且返回相关的调用状态。
底层我们统一返回InvokeResult<T>泛型对象,里面包含返回的数据以及此次操作的其他信息。
/// <summary>
/// 操作状结果处理--有返回值
/// </summary>
/// <typeparam name="T"></typeparam>
public class HandleResult<T>
{
public HandleResult()
{
Success = true;
InvokCode = InvokeCode.Success;
} public T Result { get; set; }
public string Message{ get; set; }
public InvokeCode InvokCode { get; set; }
public bool Success { get; set; }
}
上层调用下层并向客户端返回ClientResult<T>泛型对象
/// <summary>
/// 操作状结果处理--有返回值
/// </summary>
/// <typeparam name="T"></typeparam>
public class ClientResult<T>
{ public T Result { get; set; } public string Message { get; set; } public bool Success { get; set; }
}
为什么要统一返回一种类型呢,这个我不多说,用过的人自然之道他的好处。现在就要做一个转换,把下层返回的HandleResult<T>转换为ClientResult<T>。如果在项目中只用HandleResult<T>进行传输,那么就不需要转换。我们的项目需要转换也是有特殊的需求。不管转不转换,反正就是调用下层,下层返回一个HandleResult<T>对象,抓住这个特点,我们可以做很多事情。我们可以定义一个类,类里面Invoke有这个方法
//不需要转换的操作
/// <summary>
/// 调用方法 返回ClientResult<TResult/> 但是没有对Result进行映射转换
/// </summary>
/// <typeparam name="TResult"></typeparam>
/// <param name="fun"></param>
/// <returns></returns>
public static ClientResult<TResult> Invoke<TResult>(Func<HandleResult<TResult>> fun)
{
var result = new ClientResult<TResult> { Success = true };
var stopWatch = new Stopwatch();
stopWatch.Start();
try
{
HandleResult<TResult> data = fun();
result.Success = data.Success;
result.Message = data.Message;
result.Result = data.Result;
}
catch (WebSeatException ex)
{
result.Success = false;
result.Message = string.Format("{0}({1})", ex.Code.ToMessage(), ex.Description);
}
catch (Exception ex)
{
result.Success = false;
result.Message = ex.GetMessage();
LogHelper.Error(ex);
}
finally
{
stopWatch.Stop();
var time = stopWatch.ElapsedMilliseconds;
if (time > )
{
#if DEBUG
LogHelper.Error(new Exception(),
string.Format("调用方法:{0}.{1}, 执行时间:{2}毫秒", fun.Method.DeclaringType, fun.Method.Name, time));
#endif
}
else
{
LogHelper.Log(string.Format("调用方法:{0}.{1}, 执行时间:{2}毫秒", fun.Method.DeclaringType, fun.Method.Name, time));
}
}
return result;
}
//需要转换的 如果有需要用Mapper做数据映射,可以给这个方法加个重载,代码如下,具体实现和上面差不多。
/// <summary>
/// 调用方法读取数据 并且 把 HandleResult<TGResult/>转换成 ClientResult<TResult/>
/// </summary>
/// <typeparam name="TGResult"></typeparam>
/// <typeparam name="TResult"></typeparam>
/// <param name="fun"></param>
/// <returns></returns>
public static ClientResult<TResult> Invoke<TGResult, TResult>(Func<HandleResult<TGResult>> fun)
{
var result = new ClientResult<TResult> { Success = true };
var stopWatch = new Stopwatch();
stopWatch.Start();
try
{
HandleResult<TGResult> data = fun();
result.Success = data.Success;
result.Message = data.Message;
result.Result = Mapper.Map<TGResult, TResult>(data.Result);
}
catch (WebSeatException ex)
{
result.Success = false;
result.Message = string.Format("{0}({1})", ex.Code.ToMessage(), ex.Description);
}
catch (Exception ex)
{
result.Success = false;
result.Message = ex.GetMessage();
LogHelper.Error(ex);
}
finally
{
stopWatch.Stop();
var time = stopWatch.ElapsedMilliseconds;
if (time > )
{
#if DEBUG
LogHelper.Error(new Exception(),
string.Format("调用方法:{0}.{1}, 执行时间:{2}毫秒", fun.Method.DeclaringType, fun.Method.Name, time));
#endif
}
else
{
LogHelper.Log(string.Format("调用方法:{0}.{1}, 执行时间:{2}毫秒", fun.Method.DeclaringType, fun.Method.Name, time));
}
}
return result;
}
我们可以看到,在上面的方法中,我们对调用过程做了包装,并做了一些转换,上层调用下层我们规定必须使用Invoke这个方法,相当于他是上层调用下层唯一的通道,所有的调用都要从此进过,这样设计我们就能够对所有的调用相关的记录。
上层调用下层的分几种情况,如果是WCF做服务端,使用和以下一样,我这里还是用的Controller里面的action做示例,示例代码如下:
public JsonResult GetGradeList()
{
return Json(WebExtension.Invoke(() => GradeService.GetGradeList()));
} public JsonResult GetGradeBySchoolIdAndSchoolStageId(int schoolId = , int schoolStageId = )
{
return Json(WebExtension.Invoke(() => GradeService.GetGradeBySchoolIdAndSchoolStageId(schoolId, schoolStageId)));
} public JsonResult GetAccountById(int id = )
{
return Json(WebExtension.Invoke<AccountDTO, AccountModel>(() => AccountDetailService.GetAccountById(id)), JsonRequestBehavior.AllowGet);
} public JsonResult GetOneSchool()
{
return Json(WebExtension.Invoke<SchoolDTO, SchoolViewModel>(() => SchoolService.GetOneSchool()));
}
记录的日志如下,一般的调试信息记录在debug中,错误信息记录在error中,如果单个方法调用超过3秒直接记录到错误日志中。
在MVC或WEBAPI中记录每个Action的执行时间和记录下层方法调用时间的更多相关文章
- MVC与WebApi中的异常过滤器
一.MVC的异常过滤器 1.自定义MVC异常过滤器 创建一个类,继承HandleErrorAttribute即可,如果不需要作为特性使用直接实现IExceptionFilter接口即可, 注意,该 ...
- 关于mvc、webapi中get、post、put、delete的参数
webapi中post提交的数据必须是一个类,get提交的数量无所谓 多种参数get时,参数名不能相同: 在能通过c#的校验的前提下,参数名.参数数量不能全完相同 public string Get( ...
- MVC路由规则以及前后台获取Action、Controller、ID名方法
1.前后台获取Action.Controller.ID名方法 前台页面:ViewContext.RouteData.Values["Action"].ToString(); Vie ...
- 【C#小知识】C#中一些易混淆概念总结---------数据类型存储,方法调用,out和ref参数的使用
这几天一直在复习C#基础知识,过程中也发现了自己以前理解不清楚和混淆的概念.现在给大家分享出来我的笔记: 一,.NET平台的重要组成部分都是有哪些 1)FCL (所谓的.NET框架类库) 这些类是微软 ...
- WebApi官网学习记录---webapi中controller与action的选择
如果framework找到一个匹配的URI,创建一个包含占位符值的字典,key就是这些占位符(不包括大括号),value来自URI或者默认值,这个字典存储在IHttpRouteData对象中.默认值可 ...
- MVC与WebApi中的异常统一处理
1.简单例子 /// <summary> /// 全局页面控制器异常记录 MVC的异常处理 /// </summary> public class CustomErrorAtt ...
- MVC和WebApi中设置Area中的页为首页
拿WebApi为例,我们一般会生成一份帮助文档,帮助文档会在Area中 我们现在要讲帮助文档设为首页 只需在App_Start文件夹下添加 RouteConfig 类 public class Rou ...
- C#.Net Mvc运营监控,计算方法/接口/action/页面执行时间
1.建立一个TimingActionFilter过滤器 public class TimingActionFilter : ActionFilterAttribute { public overrid ...
- ASP.NET MVC 在控制器中接收视图表单POST过来的数据方法
方法一:通过Request.Form [HttpPost] public ActionResult Test() { string id=Reques ...
随机推荐
- [oracle 11g 新特性] virtual column虚拟列
总结:虚拟列可以使用于一些特殊场合,实质是类似于函数列(即以 表中已有的列 经过函数运算得来),“虚拟列不存储在数据库中,是在执行查询时由oracle后台计算出来返回给用户”,因此虚拟列不会增加存储空 ...
- Clean Code(二):函数
笔记2:函数1.短小.还要更短小 每个函数都一目了然,每个函数灰依序把你带到下一个函数 if.else.while语句等,其中的代码块应该只有一行,块内调用的函数名称应该较具有说明性2.只 ...
- HDU-1060(简单数学)
Leftmost Digit Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) P ...
- Struts,Spring,Hibernate的作用
Spring的作用: 在SSH框假中spring充当了管理容器的角色.我们都知道Hibernate用来做持久层,因 为它将JDBC做了一个良好的封装,程序员在与数据库进行交互时可以不用书写大量的SQL ...
- SQL Server 2008 Values 新用途
SQL Server 2008中新增功能:可以使用单个Insert命令插入多行. Create table Demo_Values (PKID int not null identity(1,1) p ...
- 20160416--javaweb之国际化
一:国际化1.国际化的概念:一款软件希望不同的国家和地区的使用者都可以使用,这个时候软件中的一些内容和数据需要根据用户地区信息不同而展示成不同的样子. 2.国际化的组成部分: (1)页面中固定文本元素 ...
- Hyper-V Windows 8.1 & Windows Server 2012 R2 Q&A
从Windows8开始,x64位系统自带Hyper-V功能,很多开发者和专业用户往往希望利用的Microsoft提供的这一免费功能,但是微软在这方面并不是最佳. 主要写几个大家经常遇到的问题. Win ...
- CDT 错误 Cannot run program "gcc"
Eclipse+CDT 编辑C/C++程序出错误: 出现编译错误: **** Rebuild of configuration Debug for project example **** **** ...
- 02_Jquery_02_元素选择器
[简述] 元素选择器就是通过元素名来查询元素 $("elementName")这里就可以通过元素名来获取jquery元素了. 但与id选择器不同的是,名称相同的元素有很多,所以获取 ...
- Codevs 3233 古道
3233 古道 时间限制: 1 s 空间限制: 8000 KB 题目等级:**白银 Silver** [传送门](http://codevs.cn/problem/3233/) 题目描述 Descri ...