Java自学者论坛

 找回密码
 立即注册

手机号码,快捷登录

恭喜Java自学者论坛(https://www.javazxz.com)已经为数万Java学习者服务超过8年了!积累会员资料超过10000G+
成为本站VIP会员,下载本站10000G+会员资源,会员资料板块,购买链接:点击进入购买VIP会员

JAVA高级面试进阶训练营视频教程

Java架构师系统进阶VIP课程

分布式高可用全栈开发微服务教程Go语言视频零基础入门到精通Java架构师3期(课件+源码)
Java开发全终端实战租房项目视频教程SpringBoot2.X入门到高级使用教程大数据培训第六期全套视频教程深度学习(CNN RNN GAN)算法原理Java亿级流量电商系统视频教程
互联网架构师视频教程年薪50万Spark2.0从入门到精通年薪50万!人工智能学习路线教程年薪50万大数据入门到精通学习路线年薪50万机器学习入门到精通教程
仿小米商城类app和小程序视频教程深度学习数据分析基础到实战最新黑马javaEE2.1就业课程从 0到JVM实战高手教程MySQL入门到精通教程
查看: 541|回复: 0

ASP.NET MVC自定义异常处理

[复制链接]
  • TA的每日心情
    奋斗
    昨天 11:25
  • 签到天数: 792 天

    [LV.10]以坛为家III

    2049

    主题

    2107

    帖子

    72万

    积分

    管理员

    Rank: 9Rank: 9Rank: 9

    积分
    723136
    发表于 2021-6-7 11:21:14 | 显示全部楼层 |阅读模式

    1.自定义异常处理过滤器类文件

    新建MyExceptionAttribute.cs异常处理类文件

    MyExceptionAttribute.cs代码如下:

    using System;
    using System.Collections.Generic;
    using System.Linq;
    using System.Web;
    using System.Web.Mvc;
    
    namespace WebApp.Models
    {
        public class MyExceptionAttribute : HandleErrorAttribute
        {
            public static Queue<Exception> exceptionQueue = new Queue<Exception>(); //异常处理队列对象 /// <summary>
            /// 控制器方法中出现异常,会调用该方法捕获异常
            /// </summary>
            /// <param name="filterContext"></param>
            public override void OnException(ExceptionContext filterContext)
            {
                exceptionQueue.Enqueue(filterContext.Exception);//将捕获的异常信息写到队列中
    
                filterContext.HttpContext.Response.Redirect("/Error.html");//跳转到错误页面
    
                base.OnException(filterContext);
            }
    
        }
    }

     

    2.将错误处理过滤器修改为自定义的异常处理过滤器

    修改如下:

    using System.Web;
    using System.Web.Mvc;
    using WebApp.Models;
    
    namespace WebApp
    {
        public class FilterConfig
        {
            public static void RegisterGlobalFilters(GlobalFilterCollection filters)
            {
                //filters.Add(new HandleErrorAttribute());
                filters.Add(new MyExceptionAttribute());//使用自定义异常处理的过滤器
            }
        }
    }

     

    3.修改Global.asax文件,在程序首次加载的时候,开启线程扫描异常队列,处理异常信息

    using Spring.Web.Mvc;
    using System;
    using System.Collections.Generic;
    using System.IO;
    using System.Linq;
    using System.Text;
    using System.Threading;
    using System.Web;
    using System.Web.Http;
    using System.Web.Mvc;
    using System.Web.Optimization;
    using System.Web.Routing;
    using WebApp.Models;
    
    namespace WebApp
    {
        // 注意: 有关启用 IIS6 或 IIS7 经典模式的说明,
        // 请访问 http://go.microsoft.com/?LinkId=9394801
    
        public class MvcApplication : SpringMvcApplication //System.Web.HttpApplication
        {
            protected void Application_Start()
            {
                AreaRegistration.RegisterAllAreas();
    
                WebApiConfig.Register(GlobalConfiguration.Configuration);
                FilterConfig.RegisterGlobalFilters(GlobalFilters.Filters);
                RouteConfig.RegisterRoutes(RouteTable.Routes);
                BundleConfig.RegisterBundles(BundleTable.Bundles);
    
                #region 开启线程扫描异队列,处理异常信息
                string fileLogPath = Server.MapPath("/Log/");//知道用来保存错误日志文件的文件夹路径
                //开启一个线程扫描日志队列
                ThreadPool.QueueUserWorkItem((a) =>
                {
                    while (true)
                    {
                        if (MyExceptionAttribute.exceptionQueue.Count > 0) //判断队列中是否有数据
                        {
                            Exception ex = MyExceptionAttribute.exceptionQueue.Dequeue();//出队
                            if (ex != null)
                            {
                                string fileName = DateTime.Now.ToString("yyyy-MM-dd") + ".txt";
                                File.AppendAllText(fileLogPath + fileName, ex.ToString(), Encoding.Default);//将异常追加写入到文件中
                            }
                            else
                            {
                                Thread.Sleep(3000);
                            }
                        }
                        else
                        {
                            Thread.Sleep(3000);//如果队列中没有数据,让当前线程休息3秒钟,避免造成CPU空转,避免CPU的浪费
                        }
                    }
                }, fileLogPath);
                #endregion
    
    
            }
        }
    }

     

    4.测试:

       public ActionResult Index()
            {
                int a = 2;
                int b = 0;
                int c = a / b;
                return Content(c.ToString());
            }

     

    5.效果

     

    6.其他说明:

    1.解决高并发错误日志问题:
       1.1用传统lock方式,如果并发量很大的话,用户会出现卡顿的情况。
       1.2.在服务器开辟一个队列,存储用户错误信息,然后单独开启一个线程来处理,就不会有卡顿情况。(队列是在内存中,写入速度很快)
       这就是生产者消费者模式

    2.Global下Application_Start()方法只在程序首次执行时调用

    3.ThreadPool线程池线程性能很好,比我们自己创建线程的效率要高很多。

    4.池用来解决对象频繁创建的问题,频繁创建消耗性能。

    5.Queue本身不是线程安全的,做下面的处理后就可以保证是线程内安全的了,就不会出现并发问题。

    Queue queue = Queue.Synchronized(myQueue); 

     

    7.源码下载:

    点击下载源码>>

    点击下载数据库文件>>

     

    哎...今天够累的,签到来了1...
    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    QQ|手机版|小黑屋|Java自学者论坛 ( 声明:本站文章及资料整理自互联网,用于Java自学者交流学习使用,对资料版权不负任何法律责任,若有侵权请及时联系客服屏蔽删除 )

    GMT+8, 2024-9-15 12:12 , Processed in 1.080332 second(s), 29 queries .

    Powered by Discuz! X3.4

    Copyright © 2001-2021, Tencent Cloud.

    快速回复 返回顶部 返回列表