[asp.net mvc 奇淫巧技] 06 - 也许你的项目同一个用户的请求都是同步的

一、感慨

很久前看到一篇博客中有句话大致的意思是:“asp.net 程序性能低下的主要原因是开发人员技术参差不齐”,当时看到这句话不以为然,然而时间过的越久接触的.net 开发人员越多就越认同这句话;特别最近发现非常一个成熟的项目中有些问题非常非常影响性能,最终影响的是用户体验,借此给大家分享一下关于asp.net中一个小小的点,但对项目有很大的性能提升;以前觉得自己接触的项目少小,然后接触的项目越多,越大就会越发现,同样的问题依旧存在;

二、先从最简单的asp.net mvc例子说起

1、Controller

public class HomeController : Controller
    {
        public ActionResult About()
        {
            return View();
        }
        public ActionResult TestAjax()
        {
            Thread.Sleep(1500);
            return Json(new { Code = 1, LoginId = 0 });
        }
    }

Controller里有两个Action,一个是显示的页面,一个是需要请求的ajax,其中Thread.Sleep(1500)简单模拟业务处理时间;

2、View

@{
    ViewBag.Title = "About";
}
@section scripts{
    <script type="text/javascript">
        $(function () {
            $("#btn-request").click(function () {
                $("#result").empty();
                for (var i = 0; i < 5; i++) {
                    $.ajax({
                        url: "/Home/TestAjax",
                        type: "post",
                        success: function (x) {
                            $("#result").append("<div></div>")
                        }
                    });
                }
            });
        });
    </script>
}
<h3>
    模拟操作
</h3>
<div>
    <input type="button" value="模拟请求" id="btn-request" />
</div>
<h3>
    结果
</h3>
<div id="result">
</div>

页面中更加简单,一个按钮同时请求5次Ajax,用于模拟页面的多次异步请求;

3、点击按钮显示效果

结果和我们想象的一样,5个异步请求同时在1500毫秒左右响应;这个时候没什么问题;

4、添加一个简单的登录

1、添加登录Controller

public class LoginController : Controller
    {
        public ActionResult Index()
        {
            Session["LoginId"] = 1;
            return Json(new { Code = 1 }, JsonRequestBehavior.AllowGet);
        }
    }

2、修改View,添加一个按钮模拟Ajax登录

@section scripts{
    <script type="text/javascript">
        $(function () {
            //...其他代码...
            $("#btn-login").click(function () {
                $.ajax({
                    url: "/login",
                    type: "post",
                    success: function (x) {
                        alert(‘登录成功!‘);
                    }
                });
             });
         });
    </script>
}

...其他代码...
<div>
    <input type="button" value="模拟登录" id="btn-login" />
</div>
...其他代码...

3、 登录以后的模拟结果

结果很明显了,登录以后,虽然前端是同时发送了5个Ajax请求,结果却每隔1500毫秒返回一个请求;或者可以说后端做了同步的处理;

对于这个问题,写到这已经很明显了;Session的锁默认对同一个Session做了同步处理;不管前端如何异步请求,怎乃后端一直同步

在开发中很多人遇到过,一个请求卡住了,导致其他所有ajax包括页面请求全部卡主没有任何反应。解决的方式要么重启浏览器,要么清掉cookie;这个演示也同时解释了遇到的这个问题;

4、使用Session后同步的原因

其实原因很简单,试想一下,如果1个以上请求同时修改Session的值,那么Session的值就会有不确定性,这个应该很好理解;为了保证Session值的准确性,只能同一个Session加上锁同步操作;

三、解决Session同步执行方法

1、使用 Attribute [SessionState(SessionStateBehavior.ReadOnly)]

既然导致同步执行的原因是修改Session导致的,那么在不需要修改Session的请求中使用只读Session就可以解决问题了;

    [SessionState(SessionStateBehavior.ReadOnly)]
    public class HomeController : Controller
    {
        public ActionResult About()
        {
            return View();
        }
        public ActionResult TestAjax()
        {
            Thread.Sleep(1500);
            return Json(new { Code = 1, LoginId = Session["LoginId"] });
        }
    }

只需要在Controller上加上特性SessionState,设置值为ReadOnly;这样同一个用户的请求也可以是异步的;

在一般的项目开发中,Session修改添加操作会在用户登录的时候使用,那么在除了登录的请求中加入Session ReadOnly,就会根本解决问题;或者笼统的说,在不需要修改Session的请求中加入Session ReadOnly,你的项目性能会有很大的提升;

这种方式对于老项目存在同样的问题,或者习惯使用Session的用户是不错的选择;

2、 使用JWT等方式

既然Session存在这个问题,那么可以使用其他方式替代Session,JWT就是一个很不错的方式,特别现在前后端分离的主流下,JWT更是绝佳的选择。详情可以参考 https://jwt.io/

四、总结

1、如果你的项目正在使用Session,而且并未设置Session ReadOnly,简单的设置会对你项目性能有很大的提升;

2、asp.net webform存在同样的问题;

3、新项目选择非Session也是不错的选择;

4、从asp.net core 2.1开始,使用Session并不会有此问题,同时兼顾了Session的使用习惯也不会导致同步的性能问题;asp.net 2.1的解决方式类jwt方式;

5、源码 点击下载

记得推荐 ^_^

系列课程

原文地址:https://www.cnblogs.com/emrys5/p/aspnet-session-readonly.html

时间: 2024-10-25 18:16:57

[asp.net mvc 奇淫巧技] 06 - 也许你的项目同一个用户的请求都是同步的的相关文章

[asp.net mvc 奇淫巧技] 04 - 你真的会用Action的模型绑定吗?

在QQ群或者一些程序的交流平台,经常会有人问:我怎么传一个数组在Action中接收.我传的数组为什么Action的model中接收不到.或者我在ajax的data中设置了一些数组,为什么后台还是接收不了.还有一些怎么传送一个复杂的对象或者Action怎么接收一个复杂的对象等等这些问题.或者有些人遇到复杂的对象或者数组直接就传送个json字符串,然后在Action中把json字符串转成model对象,当然这也是一种做法,但也许不是最优的做法. 一.需求 按照如图的数据格式,传入到Action,用一

[asp.net mvc 奇淫巧技] 02 - 巧用Razor引擎在Action内生成Html代码

在web开发中经常会遇到在内部代码中获取Html,这些Html是需要和数据进行一起渲染.并不是直接把Html代码返回给客户端.这样的做法有很多应用场景,例如分页.Ajax一次性获取几段Html片段.生成邮件发送模板.生成Html静态页面等等.比较简单的或者容易想到的做法就是直接拼接Html,当然这肯定不是最合适的做法. 应用场景 1.在分页中,有一种做法是用ajax获取table的html代码和一些分页信息的Json var json = { "table": "<ta

[asp.net mvc 奇淫巧技] 03 - 枚举特性扩展解决枚举命名问题和支持HtmlHelper

一.需求 我们在开发中经常会遇到一些枚举,而且这些枚举类型可能会在表单中的下拉中,或者单选按钮中会用到等. 这样用是没问题的,但是用过的人都知道一个问题,就是枚举的命名问题,当然有很多人枚举直接中文命名,我是不推荐这种命名规则,因为实在不够友好. 那有没有可以不用中文命名,而且可以显示中文的方法呢.答案是肯定的. 二.特性解决枚举命名问题 那就是用特性解决命名问题,这样的话既可以枚举用英文命名,显示又可以是中文的,岂不两全其美. /// <summary> /// 性别 /// </su

[javascript 实践篇]——那些你不知道的“奇淫巧技”

1. 空(null, undefined)验证 刚开始,我是比较蠢的验证(我还真是这样子验证的) if (variable1 !== null || variable1 !== undefined || variable1 !== '') { let variable2 = variable1; } 大哥教会了我这样子验证,你会惊叹一下的 let variable2 = variable1 || ''; 如果你不信,在谷歌浏览器开发者面板的控制台下试试! //值为null的例子 let vari

【C#冷知识系列】(一)那些你知道或者不知道的奇淫巧技

引子 正如我在个人介绍中所写,我是一个仍然坚持.NET的头铁高级软件工程师,研究C#,.NET已经六年多,一直坚持认为自己的能力不足以教授别人,所以一直没有想法写博客.工作几年,内容涵盖了.NET框架下的各种软件的开发,WPF,WinForm,WebForm,ASP.NET,MVC5,开发的软件几乎涵盖了.NET家族的各个成员.让我下定决心写一系列C#高级开发文章的原因并不是因为某天早上起床突如其来的兴致勃勃的决定要将自己这些年积累的经验分享给大家,而是是因为公司前端的一句"快脱坑吧,你们做WP

ASP.NET Core 奇淫技巧之伪属性注入

原文:ASP.NET Core 奇淫技巧之伪属性注入 一.前言 开局先唠嗑一下,许久未曾更新博客,一直在调整自己的状态,去年是我的本命年,或许是应验了本命年的多灾多难,过得十分不顺,不论是生活上还是工作上.还好当我度过了所谓的本命年后,许多事情都在慢慢变好,我将会开始恢复更新博客,争取恢复到以前的速度上(因为工作比较忙,所以这个过程可能需要一段时间). 二.关于属性注入 说到属性注入,我们就不得不提一下 DI(Dependency Injection),即依赖注入,用过 ASP.NET Core

源码解析中看到的奇淫巧技

源码解析中看到的奇淫巧技 一. 数组重置 let arr = [123,123] arr.length // 2 arr.length = 0 arr // [] 当我们给数组的length 属性设置成 0 .那么数组就会被重置为空. (很神奇有木有 二. 数据类型判断 1. 判断是否为 undefined let isUndef = function(v) { return v === undefined || v === null } 2. 判断是否 不为空 let isDef = func

asp.net mvc下使用Html.Partial嵌套页面(功能同等用户控件)

return View()相关简介 在asp.net mvc中返回View时使用的是ViewResult,它继承自ViewResultBase 同时它还有个兄弟PartialViewResult.一个用于返回整体,另一个返回局部(HTML) 使用相关 1.控制器书写:除最后改为 "return PartialView()" 其余无差别 2.视图书写:@Html.Partial() 具有四个重载.根据情况选用 @Html.Partial("CustomerListControl

ASP.NET MVC Area使用-将Area设置成独立项目

环境说明:Vistual Studio 2013 MVC 4.0 其实关于ASP.NET MVC Area使用的基础知识可以参考 http://www.cnblogs.com/willick/p/3331519.html 这篇软文. Global.asax 中的 Application_Start 方法里面有这样一句代码 AreaRegistration.RegisterAllAreas(); 估计: 它的作用会到当前MVC 所在的bin中动态注入继承了 AreaRegistration的类,