一、感慨

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

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

1、Controller

  1. public class HomeController : Controller
  2. {
  3. public ActionResult About()
  4. {
  5. return View();
  6. }
  7. public ActionResult TestAjax()
  8. {
  9. Thread.Sleep();
  10. return Json(new { Code = , LoginId = 0 });
  11. }
  12. }

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

2、View

  1. @{
  2. ViewBag.Title = "About";
  3. }
  4. @section scripts{
  5. <script type="text/javascript">
  6. $(function () {
  7. $("#btn-request").click(function () {
  8. $("#result").empty();
  9. for (var i = 0; i < 5; i++) {
  10. $.ajax({
  11. url: "/Home/TestAjax",
  12. type: "post",
  13. success: function (x) {
  14. $("#result").append("<div></div>")
  15. }
  16. });
  17. }
  18. });
  19. });
  20. </script>
  21. }
  22. <h3>
  23. 模拟操作
  24. </h3>
  25. <div>
  26. <input type="button" value="模拟请求" id="btn-request" />
  27. </div>
  28. <h3>
  29. 结果
  30. </h3>
  31. <div id="result">
  32. </div>

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

3、点击按钮显示效果

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

4、添加一个简单的登录

1、添加登录Controller

  1. public class LoginController : Controller
  2. {
  3. public ActionResult Index()
  4. {
  5. Session["LoginId"] = ;
  6. return Json(new { Code = }, JsonRequestBehavior.AllowGet);
  7. }
  8. }

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

  1. @section scripts{
  2. <script type="text/javascript">
  3. $(function () {
  4. //...其他代码...
  5. $("#btn-login").click(function () {
  6. $.ajax({
  7. url: "/login",
  8. type: "post",
  9. success: function (x) {
  10. alert('登录成功!');
  11. }
  12. });
  13. });
  14. });
  15. </script>
  16. }
  17.  
  18. ...其他代码...
  19. <div>
  20. <input type="button" value="模拟登录" id="btn-login" />
  21. </div>
  22. ...其他代码...

3、 登录以后的模拟结果

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

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

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

4、使用Session后同步的原因

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

三、解决Session同步执行方法

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

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

  1. [SessionState(SessionStateBehavior.ReadOnly)]
  2. public class HomeController : Controller
  3. {
  4. public ActionResult About()
  5. {
  6. return View();
  7. }
  8. public ActionResult TestAjax()
  9. {
  10. Thread.Sleep();
  11. return Json(new { Code = , LoginId = Session["LoginId"] });
  12. }
  13. }

只需要在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、为了简化会话状态,ASP.NET 对一个给定的会话在某一时刻只处理一个查询。如果客户端形成了多个重叠的请求,他们将被排成队列,并由服务器依序处理。其好处是不需要担忧多个请求对同一数据进行修改的情况,缺点是得不到所希望的请求吞吐量。
---引用自 《精通ASP.NET MVC5》 19.6.1

6、源码 点击下载

记得推荐 ^_^

系列课程

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

  1. [asp.net mvc 奇淫巧技] 01 - 封装上下文 - 在View中获取自定义的上下文

    我们在asp.net 开发中已经封装了最强大的HttpContext,我们可以在HttpContext中可以获取到几乎任何想获取的东西,也可以在HttpContext写入需要返回客户端的信息.但是这些 ...

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

    在web开发中经常会遇到在内部代码中获取Html,这些Html是需要和数据进行一起渲染.并不是直接把Html代码返回给客户端.这样的做法有很多应用场景,例如分页.Ajax一次性获取几段Html片段.生 ...

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

    一.需求 我们在开发中经常会遇到一些枚举,而且这些枚举类型可能会在表单中的下拉中,或者单选按钮中会用到等. 这样用是没问题的,但是用过的人都知道一个问题,就是枚举的命名问题,当然有很多人枚举直接中文命 ...

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

    在QQ群或者一些程序的交流平台,经常会有人问:我怎么传一个数组在Action中接收.我传的数组为什么Action的model中接收不到.或者我在ajax的data中设置了一些数组,为什么后台还是接收不 ...

  5. [asp.net mvc 奇淫巧技] 05 - 扩展ScriptBundle,支持混淆加密javascript

    一.需求: 在web开发中,经常会处理javascript的一些问题,其中就包括js的压缩,合并,发布版本以及混淆加密等等问题.在asp.net 开发中我们使用ScriptBundle已经可以解决ja ...

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

    1. 空(null, undefined)验证 刚开始,我是比较蠢的验证(我还真是这样子验证的) if (variable1 !== null || variable1 !== undefined | ...

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

    一.前言 开局先唠嗑一下,许久未曾更新博客,一直在调整自己的状态,去年是我的本命年,或许是应验了本命年的多灾多难,过得十分不顺,不论是生活上还是工作上.还好当我度过了所谓的本命年后,许多事情都在慢慢变 ...

  8. ASP.NET MVC+EF框架+EasyUI实现权限管理系列(17)-注册用户功能的细节处理(各种验证)

    原文:ASP.NET MVC+EF框架+EasyUI实现权限管理系列(17)-注册用户功能的细节处理(各种验证) ASP.NET MVC+EF框架+EasyUI实现权限管系列 (开篇)   (1):框 ...

  9. ASP.NET MVC+EF框架+EasyUI实现权限管理系列(12)-实现用户异步登录和T4模板

    原文:ASP.NET MVC+EF框架+EasyUI实现权限管理系列(12)-实现用户异步登录和T4模板 ASP.NET MVC+EF框架+EasyUI实现权限管系列 (开篇)   (1):框架搭建  ...

随机推荐

  1. QQ connect client request's parameters are invalid, invalid openid 问题的解决

    很多人的这个问题是POST的时候发生,我的也恰好在POST的时候发生.后来我发现可能是因为QQ的这个后端是采用类PHP的语言开发,在动态语言的获取参数时POST参数和GET参数是可以分开读取的,也就是 ...

  2. 如何用plugman编辑和添加cordova插件

    1.安装工具 进入nodejs, 安装工具plugman,管理插件,输入命令npm install -g plugman 等待下载安装 2.使用plugman命令生成插件框架 cmd 进入用于生成插件 ...

  3. js对象属性值为对象形式取值方式

    console.log(rowData);//取带点的属性值 console.log(rowData['layoutPipegallery.pipegallerycode']);//取带点的属性值

  4. 批处理(Batch)---批处理脚本。

    批处理(Batch),也称为批处理脚本.顾名思义,批处理就是对某对象进行批量的处理,通常被认为是一种简化的脚本语言,它应用于DOS和Windows系统中.批处理文件的扩展名为bat .目前比较常见的批 ...

  5. Total Command使用笔记

    一.快键键(基于水晶2右)以下数字小键盘无效 Tab 左右窗口切换 Ctrl+d 进入工作目录ctrl+d+数字 指定目录alt+←/→ 后退/前进目录ctrl+\ 跳转到根目录Ctrl+b,不分层级 ...

  6. jieba库分词

    (1)团队简介的词频统计 import jieba import collections s="制作一个购票小程序,这个购票小程序可以根据客户曾经的购票历史" s+="和 ...

  7. ADC0832的应用

    ADC0832是美国国家半导体公司生产的一种8位逐次比较型CMOS双通道A-D转换器,采用5V电源电压供电,模拟电压输入范围为0~5V,内部时钟250KHz时转换速度为32微秒. 仿真图为: 程序为: ...

  8. Django中数据库表的关联与创建(语言:python)

    首先选择选用的数据库,(本人选用django(1.11.8版本)) 在主项目settings中操作如下: DATABASES = { 'default': { 'ENGINE': 'django.db ...

  9. JVM GC-----垃圾回收算法

    说到Java,一定绕不开GC,尽管不是Java首创的,但Java一定是使用GC的代表.GC就是垃圾回收,更直接点说就是内存回收.是对内存进行整理,从而使内存的使用尽可能大的被复用. 一直想好好写一篇关 ...

  10. QT5:C++实现基于multimedia的音乐播放器(二)

    今天接着上一篇来实现播放器的槽函数. 先来实现播放模式,槽函数如下: //播放模式 void Music::musicPlayPattern() { //z=++z%3; ) { //顺序播放 pla ...