从 ASP.NET 迁移到 ASP.NET Core
liuian 2024-12-02 22:22 29 浏览
目标框架
ASP.NET Core 项目为开发人员提供了面向 .NET Core 和/或 .NET Framework 的灵活性。 若要确定最合适的目标框架,请参阅为服务器应用选择 .NET Core 或 .NET Framework。
面向 .NET Framework 时,项目需要引用单个 NuGet 包。
得益于有 ASP.NET Core 元包,面向 .NET Core 时可以避免进行大量的显式包引用。 在项目中安装 Microsoft.AspNetCore.App 元包:
XML
<ItemGroup>
<PackageReference Include="Microsoft.AspNetCore.App" />
</ItemGroup>
使用此元包时,应用不会部署元包中引用的任何包。 .NET Core 运行时存储中包含这些资产,并已预编译,旨在提升性能。 如需了解更多详情,请参阅用于 ASP.NET Core 的 Microsoft.AspNetCore.App 元包。
项目结构差异
ASP.NET Core 中简化了 .csproj 文件格式。 下面是一些显著的更改:
- 无需显式添加,即可将文件视作项目的一部分。 服务于大型团队时,这可减少出现 XML 合并冲突的风险。
- 没有对其他项目的基于 GUID 的引用,这可以提高文件的可读性。
- 无需在 Visual Studio 中卸载文件即可对它进行编辑:
Global.asax 文件替换
ASP.NET Core 引入了启动应用的新机制。 ASP.NET 应用程序的入口点是 Global.asax 文件。 路由配置及筛选器和区域注册等任务在 Global.asax 文件中进行处理。
C#
public class MvcApplication : System.Web.HttpApplication
{
protected void Application_Start()
{
AreaRegistration.RegisterAllAreas();
FilterConfig.RegisterGlobalFilters(GlobalFilters.Filters);
RouteConfig.RegisterRoutes(RouteTable.Routes);
BundleConfig.RegisterBundles(BundleTable.Bundles);
}
}
此方法会将应用程序和应用程序要部署到的服务器耦合在一起,并且它们的耦合方式会干扰实现。 为了将它们分离,引入了 OWIN 来提供一种更为简便的同时使用多个框架的方法。 OWIN 提供了一个管道,可以只添加所需的模块。 托管环境使用 Startup 函数配置服务和应用的请求管道。 Startup 在应用程序中注册一组中间件。 对于每个请求,应用程序都使用现有处理程序集的链接列表的头指针调用各个中间件组件。 每个中间件组件可以向请求处理管道添加一个或多个处理程序。 为此,需要返回对成为列表新头的处理程序的引用。 每个处理程序负责记住并调用列表中的下一个处理程序。 使用 ASP.NET Core 时,应用程序的入口点是 Startup,不再具有 Global.asax 的依赖关系。 结合使用 OWIN 和 .NET Framework 时,使用的管道应如下所示:
C#
using Owin;
using System.Web.Http;
namespace WebApi
{
// Note: By default all requests go through this OWIN pipeline. Alternatively you can turn this off by adding an appSetting owin:AutomaticAppStartup with value “false”.
// With this turned off you can still have OWIN apps listening on specific routes by adding routes in global.asax file using MapOwinPath or MapOwinRoute extensions on RouteTable.Routes
public class Startup
{
// Invoked once at startup to configure your application.
public void Configuration(IAppBuilder builder)
{
HttpConfiguration config = new HttpConfiguration();
config.Routes.MapHttpRoute("Default", "{controller}/{customerID}", new { controller = "Customer", customerID = RouteParameter.Optional });
config.Formatters.XmlFormatter.UseXmlSerializer = true;
config.Formatters.Remove(config.Formatters.JsonFormatter);
// config.Formatters.JsonFormatter.UseDataContractJsonSerializer = true;
builder.UseWebApi(config);
}
}
}
这会配置默认路由,默认为 XmlSerialization 而不是 Json。 根据需要向此管道添加其他中间件(加载服务、配置设置、静态文件等)。
ASP.NET Core 使用相似的方法,但是不依赖 OWIN 处理条目。 而是通过 Program.cs Main 方法(类似于控制台应用程序)来完成,并且 Startup 会通过该处进行加载。
C#
using Microsoft.AspNetCore;
using Microsoft.AspNetCore.Hosting;
namespace WebApplication2
{
public class Program
{
public static void Main(string[] args)
{
CreateWebHostBuilder(args).Build().Run();
}
public static IWebHostBuilder CreateWebHostBuilder(string[] args) =>
WebHost.CreateDefaultBuilder(args)
.UseStartup<Startup>();
}
}
Startup 必须包含 Configure 方法。 在 Configure 中,向管道添加必要的中间件。 在下面的示例(来自默认网站模板)中,扩展方法为管道配置以下支持:
- 错误页
- HTTP 严格传输安全
- 从 HTTP 重定向到 HTTPS
- ASP.NET Core MVC
C#
public void Configure(IApplicationBuilder app, IHostingEnvironment env)
{
if (env.IsDevelopment())
{
app.UseDeveloperExceptionPage();
}
else
{
app.UseHsts();
}
app.UseHttpsRedirection();
app.UseMvc();
}
现在主机和应用程序已分离,这样将来就可以灵活地迁移到其他平台。
备注
若要获取 ASP.NET Core Startup 和中间件的更深入的参考信息,请参阅 ASP.NET Core 中的 Startup
存储配置
ASP.NET 支持存储设置。 这些设置可用于支持应用程序已部署到的环境(以此用途为例)。 常见做法是将所有的自定义键值对存储在 Web.config 文件的 <appSettings> 部分中:
XML
<appSettings>
<add key="UserName" value="User" />
<add key="Password" value="Password" />
</appSettings>
应用程序使用 System.Configuration 命名空间中的 ConfigurationManager.AppSettings 集合读取这些设置:
C#
string userName = System.Web.Configuration.ConfigurationManager.AppSettings["UserName"];
string password = System.Web.Configuration.ConfigurationManager.AppSettings["Password"];
ASP.NET Core 可以将应用程序的配置数据存储在任何文件中,并可在启动中间件的过程中加载它们。 项目模板中使用的默认文件是 appsettings.json:
JSON复制
{
"Logging": {
"IncludeScopes": false,
"LogLevel": {
"Default": "Debug",
"System": "Information",
"Microsoft": "Information"
}
},
"AppConfiguration": {
"UserName": "UserName",
"Password": "Password"
}
}
将此文件加载到应用程序内的 IConfiguration 的实例的过程在 Startup.cs 中完成:
C#
public Startup(IConfiguration configuration)
{
Configuration = configuration;
}
public IConfiguration Configuration { get; }
应用读取 Configuration 来获得这些设置:
C#
string userName = Configuration.GetSection("AppConfiguration")["UserName"];
string password = Configuration.GetSection("AppConfiguration")["Password"];
此方法有扩展项,它们可使此过程更加可靠,例如使用依存关系注入 (DI) 来加载使用这些值的服务。 DI 方法提供了一组强类型的配置对象。
C#
// Assume AppConfiguration is a class representing a strongly-typed version of AppConfiguration section
services.Configure<AppConfiguration>(Configuration.GetSection("AppConfiguration"));
备注
若要获取 ASP.NET Core 配置的更深入的参考信息,请参阅 ASP.NET Core 中的配置。
本机依存关系注入
生成大型可缩放应用程序时,一个重要的目标是将组件和服务松散耦合。 依赖项注入不仅是可实现此目标的常用技术,还是 ASP.NET Core 的本机组件。
在 ASP.NET 应用中,开发人员依赖第三方库实现依存关系注入。 其中的一个库是 Microsoft 模式和做法提供的 Unity。
实现打包 UnityContainer 的 IDependencyResolver 是使用 Unity 设置依存关系注入的一个示例:
C#
using Microsoft.Practices.Unity;
using System;
using System.Collections.Generic;
using System.Web.Http.Dependencies;
public class UnityResolver : IDependencyResolver
{
protected IUnityContainer container;
public UnityResolver(IUnityContainer container)
{
if (container == null)
{
throw new ArgumentNullException("container");
}
this.container = container;
}
public object GetService(Type serviceType)
{
try
{
return container.Resolve(serviceType);
}
catch (ResolutionFailedException)
{
return null;
}
}
public IEnumerable<object> GetServices(Type serviceType)
{
try
{
return container.ResolveAll(serviceType);
}
catch (ResolutionFailedException)
{
return new List<object>();
}
}
public IDependencyScope BeginScope()
{
var child = container.CreateChildContainer();
return new UnityResolver(child);
}
public void Dispose()
{
Dispose(true);
}
protected virtual void Dispose(bool disposing)
{
container.Dispose();
}
}
创建 UnityContainer 的实例,注册服务,然后将 HttpConfiguration 的依赖关系解析程序设置为容器的 UnityResolver 新实例:
C#
public static void Register(HttpConfiguration config)
{
var container = new UnityContainer();
container.RegisterType<IProductRepository, ProductRepository>(new HierarchicalLifetimeManager());
config.DependencyResolver = new UnityResolver(container);
// Other Web API configuration not shown.
}
在必要时注入 IProductRepository:
C#
public class ProductsController : ApiController
{
private IProductRepository _repository;
public ProductsController(IProductRepository repository)
{
_repository = repository;
}
// Other controller methods not shown.
}
由于依存关系注入是 ASP.NET Core 的组成部分,因此可以在 Startup.cs 的 ConfigureServices 方法中添加你的服务:
C#
public void ConfigureServices(IServiceCollection services)
{
// Add application services.
services.AddTransient<IProductRepository, ProductRepository>();
}
可在任意位置注入存储库,Unity 亦是如此。
提供静态文件
Web 开发的一个重要环节是提供客户端静态资产的功能。 HTML、CSS、Javascript 和图像是最常见的静态文件示例。 这些文件需要保存在应用(或 CDN)的发布位置中,并且需要引用它们,以便请求可以加载这些文件。 在 ASP.NET Core 中,此过程发生了变化。
在 ASP.NET 中,静态文件存储在各种目录中,并在视图中进行引用。
在 ASP.NET Core 中,静态文件存储在“Web 根”(<内容根>/wwwroot)中,除非另有配置。 通过从 Startup.Configure 调用 UseStaticFiles 扩展方法将这些文件加载到请求管道中:
C#
public void Configure(IApplicationBuilder app)
{
app.UseStaticFiles();
}
备注
如果面向 .NET Framework,请安装 NuGet 包 Microsoft.AspNetCore.StaticFiles。
例如,可以通过浏览器从类似 http://<app>/images/<imageFileName> 的位置访问 wwwroot/images 文件夹中的图像资产。
多值 cookie
ASP.NET Core 不支持多值 cookie。 为每个值创建一个 cookie。
ASP.NET Core 中不压缩身份验证 cookie
出于安全原因,ASP.NET Core 中不压缩身份验证 cookie。 使用身份验证 cookie 时,开发人员应将声明信息数量减少到所需的量。
部分应用迁移
部分应用迁移的一种方法是创建 IIS 子应用程序,只将特定的路由从 ASP.NET 4.x 迁移到 ASP.NET Core,同时保留应用的 URL 结构。 例如,applicationHost.config 文件中应用的 URL 结构:
XML
<sites>
<site name="Default Web Site" id="1" serverAutoStart="true">
<application path="/">
<virtualDirectory path="/" physicalPath="D:\sites\MainSite\" />
</application>
<application path="/api" applicationPool="DefaultAppPool">
<virtualDirectory path="/" physicalPath="D:\sites\netcoreapi" />
</application>
<bindings>
<binding protocol="http" bindingInformation="*:80:" />
<binding protocol="https" bindingInformation="*:443:" sslFlags="0" />
</bindings>
</site>
...
</sites>
目录结构:
复制
.
├── MainSite
│ ├── ...
│ └── Web.config
└── NetCoreApi
├── ...
└── web.config
[BIND] 和输入格式化程序
ASP.NET 早期版本使用 [Bind] 属性防止“过多发布”攻击。 在 ASP.NET Core 中,输入格式化程序的工作方式有所不同。 与输入格式化程序一起用于分析 JSON 或 XML 时,[Bind] 属性不再专用于防止过多发布。 数据源是使用 x-www-form-urlencoded 内容类型发布的表单数据时,这些属性会影响模型绑定。
对于将 JSON 信息发布到控制器并使用 JSON 输入格式化程序分析数据的应用程序,我们建议将 [Bind] 属性替换为与 [Bind] 属性定义的属性相匹配的视图模型。
相关推荐
- Python生态下的微服务框架FastAPI
-
FastAPI是什么FastAPI是一个用于构建API的web框架,使用Python并基于标准的Python类型提示。与flask相比有什么优势高性能:得益于uvloop,可达到与...
- SpringBoot:如何解决跨域问题,详细方案和示例代码
-
跨域问题在前端开发中经常会遇到,特别是在使用SpringBoot框架进行后端开发时。解决跨域问题的方法有很多,我将为你提供一种详细的方案,包含示例代码。首先,让我们了解一下什么是跨域问题。跨域是指在...
- 使用Nginx轻松搞定跨域问题_使用nginx轻松搞定跨域问题的方法
-
跨域问题(Cross-OriginResourceSharing,简称CORS)是由浏览器的同源策略引起的。同源策略指的是浏览器限制来自不同源(协议、域名、端口)的JavaScript对资源的...
- spring boot过滤器与拦截器的区别
-
有小伙伴使用springboot开发多年,但是对于过滤器和拦截器的主要区别依然傻傻分不清。今天就对这两个概念做一个全面的盘点。定义与作用范围过滤器(Filter):过滤器是一种可以动态地拦截、处理和...
- nginx如何配置跨域_nginx配置跨域访问
-
要在Nginx中配置跨域,可以使用add_header指令来添加Access-Control-Allow-*头信息,如下所示:location/api{if($reques...
- 解决跨域问题的8种方法,含网关、Nginx和SpringBoot~
-
跨域问题是浏览器为了保护用户的信息安全,实施了同源策略(Same-OriginPolicy),即只允许页面请求同源(相同协议、域名和端口)的资源,当JavaScript发起的请求跨越了同源策略,...
- 图解CORS_图解数学
-
CORS的全称是Cross-originresourcesharing,中文名称是跨域资源共享,是一种让受限资源能够被其他域名的页面访问的一种机制。下图描述了CORS机制。一、源(Orig...
- CORS 幕后实际工作原理_cors的工作原理
-
跨域资源共享(CORS)是Web浏览器实施的一项重要安全机制,用于保护用户免受潜在恶意脚本的攻击。然而,这也是开发人员(尤其是Web开发新手)感到沮丧的常见原因。小编在此将向大家解释它存在...
- 群晖无法拉取Docker镜像?最稳定的方法:搭建自己的加速服务!
-
因为未知的原因,国内的各大DockerHub镜像服务器无法使用,导致在使用群晖时无法拉取镜像构建容器。网上大部分的镜像加速服务都是通过Cloudflare(CF)搭建的,为什么都选它呢?因为...
- Sa-Token v1.42.0 发布,新增 API Key、TOTP 验证码等能力
-
Sa-Token是一款免费、开源的轻量级Java权限认证框架,主要解决:登录认证、权限认证、单点登录、OAuth2.0、微服务网关鉴权等一系列权限相关问题。目前最新版本v1.42.0已...
- NGINX常规CORS错误解决方案_nginx配置cors
-
CORS错误CORS(Cross-OriginResourceSharing,跨源资源共享)是一种机制,它使用额外的HTTP头部来告诉浏览器允许一个网页运行的脚本从不同于它自身来源的服务器上请求资...
- Spring Boot跨域问题终极解决方案:3种方案彻底告别CORS错误
-
引言"接口调不通?前端同事又双叒叕在吼跨域了!""明明Postman能通,浏览器却报OPTIONS403?""生产环境跨域配置突然失效,凌晨3点被夺命连环Ca...
- SpringBoot 项目处理跨域的四种技巧
-
上周帮一家公司优化代码时,顺手把跨域的问题解决了,这篇文章,我们聊聊SpringBoot项目处理跨域的四种技巧。1什么是跨域我们先看下一个典型的网站的地址:同源是指:协议、域名、端口号完全相...
- Spring Cloud入门看这一篇就够了_spring cloud使用教程
-
SpringCloud微服务架构演进单体架构垂直拆分分布式SOA面向服务架构微服务架构服务调用方式:RPC,早期的webservice,现在热门的dubbo,都是RPC的典型代表HTTP,HttpCl...
- 前端程序员:如何用javascript开发一款在线IDE?
-
前言3年前在AWSre:Invent大会上AWS宣布推出Cloud9,用于在云端编写、运行和调试代码,它可以直接运行在浏览器中,也就是传说中的WebIDE。3年后的今天随着国内云计算的发...
- 一周热门
-
-
【验证码逆向专栏】vaptcha 手势验证码逆向分析
-
Python实现人事自动打卡,再也不会被批评
-
Psutil + Flask + Pyecharts + Bootstrap 开发动态可视化系统监控
-
一个解决支持HTML/CSS/JS网页转PDF(高质量)的终极解决方案
-
再见Swagger UI 国人开源了一款超好用的 API 文档生成框架,真香
-
网页转成pdf文件的经验分享 网页转成pdf文件的经验分享怎么弄
-
C++ std::vector 简介
-
飞牛OS入门安装遇到问题,如何解决?
-
系统C盘清理:微信PC端文件清理,扩大C盘可用空间步骤
-
10款高性能NAS丨双十一必看,轻松搞定虚拟机、Docker、软路由
-
- 最近发表
- 标签列表
-
- python判断字典是否为空 (50)
- crontab每周一执行 (48)
- aes和des区别 (43)
- bash脚本和shell脚本的区别 (35)
- canvas库 (33)
- dataframe筛选满足条件的行 (35)
- gitlab日志 (33)
- lua xpcall (36)
- blob转json (33)
- python判断是否在列表中 (34)
- python html转pdf (36)
- 安装指定版本npm (37)
- idea搜索jar包内容 (33)
- css鼠标悬停出现隐藏的文字 (34)
- linux nacos启动命令 (33)
- gitlab 日志 (36)
- adb pull (37)
- python判断元素在不在列表里 (34)
- python 字典删除元素 (34)
- vscode切换git分支 (35)
- python bytes转16进制 (35)
- grep前后几行 (34)
- hashmap转list (35)
- c++ 字符串查找 (35)
- mysql刷新权限 (34)