麻豆小视频在线观看_中文黄色一级片_久久久成人精品_成片免费观看视频大全_午夜精品久久久久久久99热浪潮_成人一区二区三区四区

首頁 > 學院 > 開發設計 > 正文

[Asp.net5]ApplicationBuilder詳解

2019-11-14 14:32:08
字體:
來源:轉載
供稿:網友

applicationBuilder(IApplicationBuilder接口),是OWIN的基礎,而且里面都是代理、代理的代理,各種lambda表達式,估計要看這部分代碼,很多人得頭昏腦漲。今天就對個類以及幾個擴展方法進行講解。

按慣例先貼代碼(這是我修改后的,將接口繼承去掉了、HttpContext類修改成自己的MyHttpContext類)

public class ApplicationBuilder    {        PRivate readonly IList<Func<RequestDelegate, RequestDelegate>> _components = new List<Func<RequestDelegate, RequestDelegate>>();        public ApplicationBuilder() { }        private ApplicationBuilder(ApplicationBuilder builder)        {        }        public ApplicationBuilder Use(Func<RequestDelegate, RequestDelegate> middleware)        {            _components.Add(middleware);            return this;        }        public ApplicationBuilder New()        {            return new ApplicationBuilder(this);        }        public RequestDelegate Build()        {            RequestDelegate app = context =>            {                context.StatusCode = "404";                System.Console.WriteLine("404");                return Task.FromResult(0);            };            foreach (var component in _components.Reverse())            {                app = component(app);            }            return app;        }    }
View Code

 RequestDelegate的定義如下:

public delegate Task RequestDelegate(MyHttpContext context);

從ApplicationBuilder的源代碼中我們可以關注3個點:_components、Use方法、Build方法。

  • _components是也一個列表(IList)對象,不過里面類型有點特殊——是以代理RequestDelegate為參數、代理RequestDelegate為返回值的一個代理。這里用代理說有點別嘴,可以把代理叫做函數,就是里面的類型是一個函數,這個函數的參數也是函數,返回值也是函數。
  • Use方法,就是在上面的列表對象后面添加一條新記錄。
  • Build方法就是將_components數組按照反向順序,制作成一個鏈式結構(有點類似鏈表的感覺)。下面用倆幅圖說明下:

  Build之前

 

  Build之后

  我們還可以從代碼中看到Item1的參數給的是“404”,而返回結果是RequestDelegate類型。也就是說這個返回類似于void RequestDelegate(MyHttpContext context)。如果系統給我們一個context變量,那么這個管道就可以從頭到尾的跑下去了。而事實上在asp.net5中,這個管道就是用于替代傳統的IHttpModule的(可能不準確),那現在問題就來了,Item1的參數是這個管道的第一環還是最后一環呢?從圖形來看應該是第一環,但是事實上這是一個誤解。因為箭頭兩面一個是參數,一個是執行體(參數是一個方法,會在執行體內調用執行)。在執行體內,可能在開始就執行參數的內容,之后執行具體的內容;也可以是先執行具體內容,之后執行參數,最后在執行一部分具體內容;還可以先執行具體內容,之后參數;還可能無視參數,直接直接自己的內容,那么之前的參數就會被忽略。也就是說無所謂順序,404可能是管道的第一環,也可能是最后一環,也可能是中間環節,還可能壓根就不執行。這個和Item1、Item2等內容具體的寫法有關系。(雖然也是鏈式結構是不是和鏈表感覺不一樣

  是不是感覺太零活了,源碼還對ApplicationBuilder做了倆個擴展方法,代碼整理如下:

   public static class RunExtensions    {        public static ApplicationBuilder Use(this ApplicationBuilder app, Func<MyHttpContext, Func<Task>, Task> middleware)        {            return app.Use(next =>            {                return context =>                {                    Func<Task> simpleNext = () => next(context);                    return middleware(context, simpleNext);                };            });        }        public static void Run(this ApplicationBuilder app, RequestDelegate handler)        {            if (app == null)            {                throw new ArgumentNullException("why?");            }            if (handler == null)            {                throw new ArgumentNullException("How?");            }            app.Use(_ => handler);        }    }
View Code

   首先說Use方法,改方法是對之前Use方法的一個更改。將傳入的參數更改為 Func<MyHttpContext, Func<Task>, Task>。這樣做有什么好處?之前的Func<RequestDelegate, RequestDelegate>對象并不能給人清楚的明了的感覺,而Func<MyHttpContext, Func<Task>, Task>就非常明確了。傳入的參數:MyHttpContext就是Context對象,Func<Task>就是next的執行體。返回值是一個Task(類似于void)。一目了然。

  再說Run方法,顯而易見,Run方法只執行自己的內容,并沒有執行參數體。所以鏈式結構的在其前的都會被舍棄,不會被執行。

 

  最后把自己的測試例子貼出來,供大家參考

示例1:

static void Main(string[] args)        {            MyHttpContext context = new MyHttpContext() { StatusCode = "A" };            Func<MyHttpContext, Func<Task>, Task> middleware =                (x, y) => { context.StatusCode += "C"; System.Console.WriteLine(context.StatusCode); return y(); };            Func<MyHttpContext, Func<Task>, Task> middleware2 =              (x, y) => { context.StatusCode += "End1"; System.Console.WriteLine(context.StatusCode); return Task.FromResult(0); };            ApplicationBuilder builder = new ApplicationBuilder();            builder.Use(                next =>                {                    return (MyHttpContext o) =>                    {                        o.StatusCode += "B";                        System.Console.WriteLine(context.StatusCode);                        next(o);                        return Task.FromResult(0);                    };                }            );            builder.Use(middleware);            //builder.Use(middleware2);            //builder.Use(middleware);            //builder.Run(o => { o.StatusCode += "End2"; return Task.FromResult(0); });            builder.Build().Invoke(context);            System.Console.ReadLine();        }
View Code

執行結果:

示例2:

static void Main(string[] args)        {            MyHttpContext context = new MyHttpContext() { StatusCode = "A" };            Func<MyHttpContext, Func<Task>, Task> middleware =                (x, y) => { context.StatusCode += "C"; System.Console.WriteLine(context.StatusCode); return y(); };            Func<MyHttpContext, Func<Task>, Task> middleware2 =              (x, y) => { context.StatusCode += "End1"; System.Console.WriteLine(context.StatusCode); return Task.FromResult(0); };            ApplicationBuilder builder = new ApplicationBuilder();            builder.Use(                next =>                {                    return (MyHttpContext o) =>                    {                        o.StatusCode += "B";                        System.Console.WriteLine(context.StatusCode);                        next(o);                        return Task.FromResult(0);                    };                }            );            builder.Use(middleware);            builder.Use(middleware2);            //builder.Use(middleware);            //builder.Run(o => { o.StatusCode += "End2"; System.Console.WriteLine(context.StatusCode); return Task.FromResult(0); });            builder.Build().Invoke(context);            System.Console.ReadLine();        }
View Code

執行結果:

示例3:

static void Main(string[] args)        {            MyHttpContext context = new MyHttpContext() { StatusCode = "A" };            Func<MyHttpContext, Func<Task>, Task> middleware =                (x, y) => { context.StatusCode += "C"; System.Console.WriteLine(context.StatusCode); return y(); };            Func<MyHttpContext, Func<Task>, Task> middleware2 =              (x, y) => { context.StatusCode += "End1"; System.Console.WriteLine(context.StatusCode); return Task.FromResult(0); };            ApplicationBuilder builder = new ApplicationBuilder();            builder.Use(                next =>                {                    return (MyHttpContext o) =>                    {                        o.StatusCode += "B";                        System.Console.WriteLine(context.StatusCode);                        next(o);                        return Task.FromResult(0);                    };                }            );            builder.Use(middleware);            //builder.Use(middleware2);            //builder.Use(middleware);            builder.Run(o => { o.StatusCode += "End2"; System.Console.WriteLine(context.StatusCode); return Task.FromResult(0); });            builder.Build().Invoke(context);            System.Console.ReadLine();        }
View Code

執行結果:

 


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 久久精品观看 | h色网站免费观看 | 欧美日韩一 | 日韩精品中文字幕在线播放 | 一区二区三区国产视频 | 免费观看一级黄色片 | 国产精品久久久久久238 | av在线播放亚洲 | 欧美淫交 | 伊人成人免费视频 | 成人 日韩 | 在线观看免费视频麻豆 | 国产免费黄色 | 国产精品久久久久久一区二区三区 | 亚洲亚色 | 亚洲欧美日韩久久精品第一区 | 羞羞视频2023 | 成人午夜在线观看视频 | 中文字幕22页 | 精品国产91一区二区三区 | 99riav视频一区二区 | 欧美一区二区黄色 | 91精品久久久久久久 | 中文字幕在线网 | 欧美成人精品一区二区 | 91九色视频 | 国产精品久久二区 | 护士hd欧美free性xxxx | 欧美日韩手机在线观看 | 福利在线免费 | 国产91片 | 久久久久久久免费看 | 91九色网址 | 国产高潮失禁喷水爽到抽搐视频 | 亚洲第一成人久久网站 | 国产成人精品免费视频大全办公室 | 日韩精品中文字幕一区二区 | 毛片视频网站 | 视频一区国产精品 | 性明星video另类hd | 欧美1—12sexvideos |