欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

asp.net通過消息隊(duì)列處理高并發(fā)請求(以搶小米手機(jī)為例)

 更新時間:2020年03月22日 14:24:42   作者:chenxizhaolu  
這篇文章主要介紹了asp.net通過消息隊(duì)列處理高并發(fā)請求(以搶小米手機(jī)為例),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧

網(wǎng)站面對高并發(fā)的情況下,除了增加硬件, 優(yōu)化程序提高以響應(yīng)速度外,還可以通過并行改串行的思路來解決。這種思想常見的實(shí)踐方式就是數(shù)據(jù)庫鎖和消息隊(duì)列的方式。這種方式的缺點(diǎn)是需要排隊(duì),響應(yīng)速度慢,優(yōu)點(diǎn)是節(jié)省成本。

演示一下現(xiàn)象

創(chuàng)建一個在售產(chǎn)品表

CREATE TABLE [dbo].[product](
  [id] [int] NOT NULL,--唯一主鍵
  [name] [nvarchar](50) NULL,--產(chǎn)品名稱
  [status] [int] NULL ,--0未售出 1 售出 默認(rèn)為0
  [username] [nvarchar](50) NULL--下單用戶
 )

添加一條記錄

insert into product(id,name,status,username) values(1,'小米手機(jī)',0,null)

創(chuàng)建一個搶票程序

public ContentResult PlaceOrder(string userName)
    {
      using (RuanMou2020Entities db = new RuanMou2020Entities())
      {
          var product = db.product.Where<product>(p => p.status== 0).FirstOrDefault();
          if (product.status == 1)
          {
            return Content("失敗,產(chǎn)品已經(jīng)被賣光");
          }
          else
          {
            //模擬數(shù)據(jù)庫慢造成并發(fā)問題
            Thread.Sleep(5000);
            product.status = 1;
            product.username= userName;
              db.SaveChanges(); 
              return Content("成功購買");
             } 
      } 
    }

如果我們在5秒內(nèi)一次訪問以下兩個地址,那么返回的結(jié)果都是成功購買且數(shù)據(jù)表中的username是lisi。

/controller/PlaceOrder?username=zhangsan

/controller/PlaceOrder?username=lisi

這就是并發(fā)帶來的問題。

第一階段,利用線程鎖簡單粗暴

Web程序是多線程的,那我們把他在容易出現(xiàn)并發(fā)的地方加一把鎖就可以了,如下圖處理方式。

    private static object _lock = new object();
    public ContentResult PlaceOrder(string userName)
    {
      using (RuanMou2020Entities db = new RuanMou2020Entities())
      {
        lock (_lock)
        {
          var product = db.product.Where<product>(p => p.status == 0).FirstOrDefault();
          if (product.status == 1)
          {
            return Content("失敗,產(chǎn)品已經(jīng)被賣光");
          }
          else
          {
            //模擬數(shù)據(jù)庫慢造成并發(fā)問題
            Thread.Sleep(5000);
            product.status = 1;
            product.username = userName;
            db.SaveChanges();
            return Content("成功購買");
          }
        }
      }
    }

這樣每一個請求都是依次執(zhí)行,不會出現(xiàn)并發(fā)問題了。

優(yōu)點(diǎn):解決了并發(fā)的問題。

缺點(diǎn):效率太慢,用戶體驗(yàn)性太差,不適合大數(shù)據(jù)量場景。

第二階段,拉消息隊(duì)列,通過生產(chǎn)者,消費(fèi)者的模式

1,創(chuàng)建訂單提交入口(生產(chǎn)者)

public class HomeController : Controller
  {

    /// <summary>
    /// 接受訂單提交(生產(chǎn)者)
    /// </summary>
    /// <returns></returns>
    public ContentResult PlaceOrderQueen(string userName)
    {
      //直接將請求寫入到訂單隊(duì)列
      OrderConsumer.TicketOrders.Enqueue(userName);
      return Content("wait");
    }

    /// <summary>
    /// 查詢訂單結(jié)果
    /// </summary>
    /// <returns></returns>
    public ContentResult PlaceOrderQueenResult(string userName)
    {
      var rel = OrderConsumer.OrderResults.Where(p => p.userName == userName).FirstOrDefault();
      if (rel == null)
      {
        return Content("還在排隊(duì)中");
      }
      else
      {
        return Content(rel.Result.ToString());
      }
    }
}

2,創(chuàng)建訂單處理者(消費(fèi)者)

/// <summary>
  /// 訂單的處理者(消費(fèi)者)
  /// </summary>
  public class OrderConsumer
  {
    /// <summary>
    /// 訂票的消息隊(duì)列
    /// </summary>
    public static ConcurrentQueue<string> TicketOrders = new ConcurrentQueue<string>();
    /// <summary>
    /// 訂單結(jié)果消息隊(duì)列
    /// </summary>
    public static List<OrderResult> OrderResults = new List<OrderResult>();
    /// <summary>
    /// 訂單處理
    /// </summary>
    public static void StartTicketTask()
    {
      string userName = null;
      while (true)
      {
        //如果沒有訂單任務(wù)就休息1秒鐘
        if (!TicketOrders.TryDequeue(out userName))
        {
          Thread.Sleep(1000);
          continue;
        }
        //執(zhí)行真實(shí)的業(yè)務(wù)邏輯(如插入數(shù)據(jù)庫)
        bool rel = new TicketHelper().PlaceOrderDataBase(userName);
        //將執(zhí)行結(jié)果寫入結(jié)果集合
        OrderResults.Add(new OrderResult() { Result = rel, userName = userName });
      }
    }
  }

3,創(chuàng)建訂單業(yè)務(wù)的實(shí)際執(zhí)行者

/// <summary>
  /// 訂單業(yè)務(wù)的實(shí)際處理者
  /// </summary>
  public class TicketHelper
  {
    /// <summary>
    /// 實(shí)際庫存標(biāo)識
    /// </summary>
    private bool hasStock = true;
    /// <summary>
    /// 執(zhí)行一個訂單到數(shù)據(jù)庫
    /// </summary>
    /// <returns></returns>
    public bool PlaceOrderDataBase(string userName)
    {
      //如果沒有了庫存,則直接返回false,防止頻繁讀庫
      if (!hasStock)
      {
        return hasStock;
      }
      using (RuanMou2020Entities db = new RuanMou2020Entities())
      {
        var product = db.product.Where(p => p.status == 0).FirstOrDefault();
        if (product == null)
        {
          hasStock = false;
          return false;
        }
        else
        {
          Thread.Sleep(10000);//模擬數(shù)據(jù)庫的效率比較慢,執(zhí)行插入時間比較久
          product.status = 1;
          product.username = userName;
          db.SaveChanges();
          return true;
        }
      }
    }
  }
  /// <summary>
  /// 訂單處理結(jié)果實(shí)體
  /// </summary>
  public class OrderResult
  {
    public string userName { get; set; }
    public bool Result { get; set; }
  }

4,在程序啟動前,啟動消費(fèi)者線程

protected void Application_Start()
    {
      AreaRegistration.RegisterAllAreas();
      GlobalConfiguration.Configure(WebApiConfig.Register);
      FilterConfig.RegisterGlobalFilters(GlobalFilters.Filters);
      RouteConfig.RegisterRoutes(RouteTable.Routes);
      BundleConfig.RegisterBundles(BundleTable.Bundles);

      //在Global的Application_Start事件里單獨(dú)開啟一個消費(fèi)者線程
      Task.Run(OrderConsumer.StartTicketTask);
    }

這樣程序的運(yùn)行模式是:用戶提交的需求里都會添加到消息隊(duì)列里去排隊(duì)處理,程序會依次處理該隊(duì)列里的內(nèi)容(當(dāng)然可以一次取出多條來進(jìn)行處理,提高效率)。

優(yōu)點(diǎn):比上一步快了。

缺點(diǎn):不夠快,而且下單后需要輪詢另外一個接口判斷是否成功。

第三階段 反轉(zhuǎn)生產(chǎn)者消費(fèi)者的角色,把可售產(chǎn)品提前放到隊(duì)列里,然后讓提交的訂單來消費(fèi)隊(duì)列里的內(nèi)容

1,創(chuàng)建生產(chǎn)者并且在程序啟動前調(diào)用其初始化程序

public class ProductForSaleManager
  {
    /// <summary>
    /// 待售商品隊(duì)列
    /// </summary>
    public static ConcurrentQueue<int> ProductsForSale = new ConcurrentQueue<int>();
    /// <summary>
    /// 初始化待售商品隊(duì)列
    /// </summary>
    public static void Init()
    {
      using (RuanMou2020Entities db = new RuanMou2020Entities())
      {
        db.product.Where(p => p.status == 0).Select(p => p.id).ToList().ForEach(p =>
        {
          ProductsForSale.Enqueue(p);
        });
      }
    }
  }


 public class MvcApplication : System.Web.HttpApplication
  {
    protected void Application_Start()
    {
      AreaRegistration.RegisterAllAreas();
      GlobalConfiguration.Configure(WebApiConfig.Register);
      FilterConfig.RegisterGlobalFilters(GlobalFilters.Filters);
      RouteConfig.RegisterRoutes(RouteTable.Routes);
      BundleConfig.RegisterBundles(BundleTable.Bundles);

      //程序啟動前,先初始化待售產(chǎn)品消息隊(duì)列
      ProductForSaleManager.Init();
    }
  }

2,創(chuàng)建消費(fèi)者

public class OrderController : Controller
  {
    /// <summary>
    /// 下訂單
    /// </summary>
    /// <param name="userName">訂單提交者</param>
    /// <returns></returns>
    public async Task<ContentResult> PlaceOrder(string userName)
    {
      if (ProductForSaleManager.ProductsForSale.TryDequeue(out int pid))
      {
        await new TicketHelper2().PlaceOrderDataBase(userName, pid);
        return Content($"下單成功,對應(yīng)產(chǎn)品id為:{pid}");
      }
      else
      {
        await Task.CompletedTask;
        return Content($"商品已經(jīng)被搶光");
      }
    }
  }

3,當(dāng)然還需要一個業(yè)務(wù)的實(shí)際執(zhí)行者

/// <summary>
  /// 訂單業(yè)務(wù)的實(shí)際處理者
  /// </summary>
  public class TicketHelper2
  {
    /// <summary>
    /// 執(zhí)行復(fù)雜的訂單操作(如數(shù)據(jù)庫)
    /// </summary>
    /// <param name="userName">下單用戶</param>
    /// <param name="pid">產(chǎn)品id</param>
    /// <returns></returns>
    public async Task PlaceOrderDataBase(string userName, int pid)
    {
      using (RuanMou2020Entities db = new RuanMou2020Entities())
      {
        var product = db.product.Where(p => p.id == pid).FirstOrDefault();
        if (product != null)
        {
          product.status = 1;
          product.username = userName;
          await db.SaveChangesAsync();
        }
      }
    }
  }

這樣我們同時訪問下面三個地址,如果數(shù)據(jù)庫里只有兩個商品的話,會有一個請求結(jié)果為:商品已經(jīng)被搶光。

http://localhost:88/Order/PlaceOrder?userName=zhangsan

http://localhost:88/Order/PlaceOrder?userName=lisi

http://localhost:88/Order/PlaceOrder?userName=wangwu

這種處理方式的優(yōu)點(diǎn)為:執(zhí)行效率快,相比第二種方式不需要第二個接口來返回查詢結(jié)果。

缺點(diǎn):暫時沒想到,歡迎大家補(bǔ)充。

說明:該方式只是個人猜想,并非實(shí)際項(xiàng)目經(jīng)驗(yàn),大家只能作為參考,慎重用于項(xiàng)目。歡迎大家批評指正。

到此這篇關(guān)于asp.net通過消息隊(duì)列處理高并發(fā)請求(以搶小米手機(jī)為例)的文章就介紹到這了,更多相關(guān)asp.net 消息隊(duì)列處理高并發(fā) 內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 詳解ASP.NET Core中配置監(jiān)聽URLs的五種方式

    詳解ASP.NET Core中配置監(jiān)聽URLs的五種方式

    這篇文章主要介紹了詳解ASP.NET Core中配置監(jiān)聽URLs的五種方式,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-04-04
  • ASP.Net2.0 GridView 多列排序,顯示排序圖標(biāo),分頁

    ASP.Net2.0 GridView 多列排序,顯示排序圖標(biāo),分頁

    ASP.Net2.0 GridView 多列排序,顯示排序圖標(biāo),分頁...
    2006-09-09
  • 詳解如何在ASP.NET Core中使用IHttpClientFactory

    詳解如何在ASP.NET Core中使用IHttpClientFactory

    這篇文章主要介紹了詳解如何在ASP.NET Core中使用IHttpClientFactory,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2021-02-02
  • asp.net中各種類型的JSON格式化

    asp.net中各種類型的JSON格式化

    asp.net中各種類型的JSON格式化實(shí)現(xiàn)代碼,需要的朋友可以參考下。
    2011-12-12
  • 在?Net7.0?環(huán)境下如何使用?RestSharp?發(fā)送?Http(FromBody和FromForm)請求

    在?Net7.0?環(huán)境下如何使用?RestSharp?發(fā)送?Http(FromBody和FromForm)請求

    這篇文章主要介紹了在?Net7.0?環(huán)境下使用?RestSharp?發(fā)送?Http(FromBody和FromForm)請求,今天,我就兩個小的知識點(diǎn),就是通過使用?RestSharp?訪問?WebAPI,提交?FromBody?和?FromForm?兩種方式的數(shù)據(jù),還是有些區(qū)別的,本文結(jié)合實(shí)例代碼介紹的非常詳細(xì),需要的朋友參考下吧
    2023-09-09
  • asp.net slickupload 使用方法(文件上傳)

    asp.net slickupload 使用方法(文件上傳)

    asp.net下使用slickupload上傳文件的代碼
    2009-05-05
  • .Net Core簡單使用Mvc內(nèi)置的Ioc

    .Net Core簡單使用Mvc內(nèi)置的Ioc

    這篇文章主要為大家詳細(xì)介紹了.Net Core簡單使用Mvc內(nèi)置的Ioc,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2018-03-03
  • 詳解.net mvc session失效問題

    詳解.net mvc session失效問題

    這篇文章主要介紹了.net mvc session失效問題,本文通過問題分析,解決過程分步驟給大家介紹的非常詳細(xì),需要的朋友可以參考下
    2016-09-09
  • ASP.NET Core MVC獲取請求的參數(shù)方法示例

    ASP.NET Core MVC獲取請求的參數(shù)方法示例

    這篇文章主要給大家介紹了關(guān)于ASP.NET Core MVC是如何獲取請求的參數(shù),文中通過示例代碼介紹的非常詳細(xì),對大家學(xué)習(xí)或者使用ASP.NET Core MVC具有一定的參考學(xué)習(xí)價值,需要的朋友們下面來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-05-05
  • 在.NET中使用DiagnosticSource的方法

    在.NET中使用DiagnosticSource的方法

    這篇文章主要介紹了在.NET中使用DiagnosticSource的方法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-10-10

最新評論