asp.net core 入口程序

1.项目结构

分享图片

说明:

  1. .vscode 文件夹存放vscode配置文件,用于配置当前项目的启动调试配置项。
  2. bin 用于存放编译结果
  3. Controllers 用于存放控制器
  4. Models 用于存放视图模型
  5. obj 用于存放中间编译文件
  6. Properties 用于存放主机配置,如端口号,启动后在浏览器中打开
  7. Views 用于存放视图
  8. wwwroot 用于存放静态文件,如 HTML 文件、JavaScript 文件和 CSS 文件
  9. appsettings[.Environment].json 包含配置数据,如连接字符串
  10. Program.cs 程序的入口点
  11. Startup.cs 用于配置服务和应用的请求管道

2.流程分析

分享图片

 1 namespace HelloWorld
 2 {
 3     public class Program
 4     {
 5         public static void Main(string[] args)
 6         {
 7             CreateWebHostBuilder(args).Build().Run();
 8         }
 9 
10         public static IWebHostBuilder CreateWebHostBuilder(string[] args) =>
11             WebHost.CreateDefaultBuilder(args)
12                 .UseStartup<Startup>();
13     }
14 }

分享图片

 1  public static IWebHostBuilder CreateDefaultBuilder(string[] args)
 2 {
 3     var builder = new WebHostBuilder()
 4         // 使用 Kestrel 作为应用程序的托管 web 服务器
 5         .UseKestrel((builderContext,options) =>
 6         {
 7             options.Configure(builderContext.Configuration.GetSection("Kestrel"));
 8         })
 9         // 将内容根目录设置为应用程序的当前工作目录
10         .UseContentRoot(Directory.GetCurrentDirectory())
11         // 加载配置项
12         .ConfigureAppConfiguration((hostingContext,config) =>
13         {
14             var env = hostingContext.HostingEnvironment;
15             // 1. 加载appsettings.json
16             config.AddJsonFile("appsettings.json",optional: true,reloadOnChange: true)
17             // 2. 加载appsettings.{EnvironmentName}.json
18                   .AddJsonFile($"appsettings.{env.EnvironmentName}.json",reloadOnChange: true);
19 
20             if (env.IsDevelopment())
21             {
22                 var appAssembly = Assembly.Load(new AssemblyName(env.ApplicationName));
23                 if (appAssembly != null)
24                 {
25                     // 3. 加载应用在使用入口程序集的 Development 环境中运行时的机密管理器
26                     config.AddUserSecrets(appAssembly,optional: true);
27                 }
28             }
29 
30             // 4. 加载环境变量
31             config.AddEnvironmentVariables();
32 
33             if (args != null)
34             {
35                 // 5. 加载命令行参数
36                 config.AddCommandLine(args);
37             }
38         })
39         // 配置日志
40         .ConfigureLogging((hostingContext,logging) =>
41         {
42             // 从配置读取日志相关配置
43             logging.AddConfiguration(hostingContext.Configuration.GetSection("Logging"));
44             // 配置控制台和调试输出的日志记录(源码中使用的是 TryAddEnumerable,说明如果配置中已配置则不重复添加)
45             logging.AddConsole();
46             logging.AddDebug();
47         })
48         .ConfigureServices((hostingContext,services) =>
49         {
50             // Fallback
51             services.PostConfigure<HostFilteringOptions>(options =>
52             {
53                 if (options.AllowedHosts == null || options.AllowedHosts.Count == 0)
54                 {
55                     // "AllowedHosts": "localhost;127.0.0.1;[::1]"
56                     var hosts = hostingContext.Configuration["AllowedHosts"]?.Split(new[] { ; },StringSplitOptions.RemoveEmptyEntries);
57                     // Fall back to "*" to disable.
58                     options.AllowedHosts = (hosts?.Length > 0 ? hosts : new[] { "*" });
59                 }
60             });
61             // Change notification
62             services.AddSingleton<IOptionsChangeTokenSource<HostFilteringOptions>>(
63                 new ConfigurationChangeTokenSource<HostFilteringOptions>(hostingContext.Configuration));
64 
65             services.AddTransient<IStartupFilter,HostFilteringStartupFilter>();
66         })
67         // IIS集成
68         .UseIISIntegration()
69         // 作用域验证
70         .UseDefaultServiceProvider((context,options) =>
71         {
72             options.ValidateScopes = context.HostingEnvironment.IsDevelopment();
73         });
74 
75     if (args != null)
76     {
77         // 加载命令行参数,比ConfigureAppConfiguration中AddCommandLine先执行
78         builder.UseConfiguration(new ConfigurationBuilder().AddCommandLine(args).Build());
79     }
80 
81     return builder;
82 }
View Code

配置加载顺序

  • appsettings.json。
  • appsettings.{Environment}.json。
  • 应用在使用入口程序集的 Development 环境中运行时的机密管理器。
  • 环境变量。
  • 命令行参数。

 引用

WebHost.cs

WebHostBuilder.cs

Web主机

作用域验证

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


在开发中,有时候生成验证码的场景目前还是存在的,本篇演示不依赖第三方组件,生成随机验证码图片。 先添加验证码接口 public interface ICaptcha { /// &lt;summary&gt; /// 生成随机验证码 /// &lt;/summary&gt; /// &lt;para
后端技术 .net code 官方文档 https://docs.microsoft.com/zh-cn/aspnet/core/mvc/models/file-uploads?view=aspnetcore-3.1 上传方式:multipart/form-data 其他参数:Name,Versio
在上一篇文章中,我们比较出单表插入9999行数据,Freesql &gt;&#160;Dapper &gt; EfCore。在本文中,我们来看看级联插入 构建9999行数据 List&lt;Entity&gt; datas = new List&lt;Entity&gt;(); for (int i
需求:导入9999行数据时Dapper, Ef core, Freesql&#160;谁的性能更优,是如何执行的,级联增加谁性能更佳。 确认方法:sql server&#160;的 sys.dm_exec_query_stats SELECT TOP 1000 (select [text] from
资料整理 1.sp-api介绍:https://developer.amazonservices.com/ 2.github文档:https://github.com/amzn/selling-partner-api-docs 3.github代码:https://github.com/amzn/s
最近时间在整SM2算法,在网上看到不少代码,基本都是使用BouncyCastle库,现在这个版本算比较好的拿来分享给大家。 首先引入包&#160;Portable.BouncyCastle 完整代码见Gitee:https://gitee.com/Karl_Albright/CryptoHelper
在上文中,我介绍了事件驱动型架构的一种简单的实现,并演示了一个完整的事件派发、订阅和处理的流程。这种实现太简单了,百十行代码就展示了一个基本工作原理。然而,要将这样的解决方案运用到实际生产环境,还有很长的路要走。今天,我们就研究一下在事件处理器中,对象生命周期的管理问题。事实上,不仅仅是在事件处理器
上文已经介绍了Identity Service的实现过程。今天我们继续,实现一个简单的Weather API和一个基于Ocelot的API网关。 回顾 《Angular SPA基于Ocelot API网关与IdentityServer4的身份认证与授权(一)》 Weather API Weather
最近我为我自己的应用开发框架Apworks设计了一套案例应用程序,并以Apache 2.0开源,开源地址是:https://github.com/daxnet/apworks-examples,目的是为了让大家更为方便地学习和使用.NET Core、最新的前端开发框架Angular,以及Apwork
HAL(Hypertext Application Language,超文本应用语言)是一种RESTful API的数据格式风格,为RESTful API的设计提供了接口规范,同时也降低了客户端与服务端接口的耦合度。很多当今流行的RESTful API开发框架,包括Spring REST,也都默认支
在前面两篇文章中,我详细介绍了基本事件系统的实现,包括事件派发和订阅、通过事件处理器执行上下文来解决对象生命周期问题,以及一个基于RabbitMQ的事件总线的实现。接下来对于事件驱动型架构的讨论,就需要结合一个实际的架构案例来进行分析。在领域驱动设计的讨论范畴,CQRS架构本身就是事件驱动的,因此,
HAL,全称为Hypertext Application Language,它是一种简单的数据格式,它能以一种简单、统一的形式,在API中引入超链接特性,使得API的可发现性(discoverable)更强,并具有自描述的特点。使用了HAL的API会更容易地被第三方开源库所调用,并且使用起来也很方便
何时使用领域驱动设计?其实当你的应用程序架构设计是面向业务的时候,你已经开始使用领域驱动设计了。领域驱动设计既不是架构风格(Architecture Style),也不是架构模式(Architecture Pattern),它也不是一种软件开发方法论,所以,是否应该使用领域驱动设计,以及什么时候使用
《在ASP.NET Core中使用Apworks快速开发数据服务》一文中,我介绍了如何使用Apworks框架的数据服务来快速构建用于查询和管理数据模型的RESTful API,通过该文的介绍,你会看到,使用Apworks框架开发数据服务是何等简单快捷,提供的功能也非常多,比如对Hypermedia的
在上一讲中,我们已经完成了一个完整的案例,在这个案例中,我们可以通过Angular单页面应用(SPA)进行登录,然后通过后端的Ocelot API网关整合IdentityServer4完成身份认证。在本讲中,我们会讨论在当前这种架构的应用程序中,如何完成用户授权。 回顾 《Angular SPA基于
Keycloak是一个功能强大的开源身份和访问管理系统,提供了一整套解决方案,包括用户认证、单点登录(SSO)、身份联合、用户注册、用户管理、角色映射、多因素认证和访问控制等。它广泛应用于企业和云服务,可以简化和统一不同应用程序和服务的安全管理,支持自托管或云部署,适用于需要安全、灵活且易于扩展的用
3月7日,微软发布了Visual Studio 2017 RTM,与之一起发布的还有.NET Core Runtime 1.1.0以及.NET Core SDK 1.0.0,尽管这些并不是最新版,但也已经从preview版本升级到了正式版。所以,在安装Visual Studio 2017时如果启用了
在上文中,我介绍了如何在Ocelot中使用自定义的中间件来修改下游服务的response body。今天,我们再扩展一下设计,让我们自己设计的中间件变得更为通用,使其能够应用在不同的Route上。比如,我们可以设计一个通用的替换response body的中间件,然后将其应用在多个Route上。 O
不少关注我博客的朋友都知道我在2009年左右开发过一个名为Apworks的企业级应用程序开发框架,旨在为分布式企业系统软件开发提供面向领域驱动(DDD)的框架级别的解决方案,并对多种系统架构风格提供支持。这个框架的开发和维护我坚持了很久,一直到2015年,我都一直在不停地重构这个项目。目前这个项目在
好吧,这个题目我也想了很久,不知道如何用最简单的几个字来概括这篇文章,原本打算取名《Angular单页面应用基于Ocelot API网关与IdentityServer4ʺSP.NET Identity实现身份认证与授权》,然而如你所见,这样的名字实在是太长了。所以,我不得不缩写“单页面应用”几个字