1**、环境准备**
- Visual Studio 2022
- .NET6 平台支持
- Internet Information Services 服务器( IIS )
- Linux 服务器 【 CentOS 系统】 ( 跨平台部署使用 )
- Linux 服务器下的 Docker 容器( Docker 部署使用)
2**、上手**NET6WebAPI
- 创建 .NET6 WebAPI 项目
- 项目结构解读
- 多种 API 定义
- 多种启动方式 3 、 RESTful 风格
3.1****什么是RESTful ?
REST 全称是 Representational State Transfer ,中文意思是表述性状态转移。
REST 本身并没有创造新的技术、组件或服务
REST 指的是一组架构约束条件和原则。
如果一个架构符合 REST 的约束条件和原则,我们就称它为 RESTful 架构。
理论上 REST 架构风格并不是绑定在 HTTP 上,只不过目前 HTTP 是唯一与 REST 相关的实例。 所以我们这
里描述的 REST 也是通过 HTTP 实现的 REST 。
3.2RESTfull 主要规则是什么?
3.2.1 资源与 URI ,所有的资源都有一个资源标志符
Core WebApi - -- 访问一块资源
3.2.2 统一资源接口
3.2.3 资源的表述
服务端可以通过 Content-Type 告诉客户端资源的表述形式。
资源的表述形式有:文本资源可以采用 html 、 xml 、 json 等格式,图片可以使用 PNG 或 JPG 展现出来
3.2.4 资源的链接
dotnet run -- 参数 在项目根目录下执行 ----- 先编译 --- 再启动
dotnet xxx.dll -- 参数 直接启动 dll 文件
https://localhost:7289/Company ---- 代表 Company 的一块资源
https://localhost:7289/User ---- 代表 User 的一块资源
幂等性:对同一 REST 接口的多次访问,得到的资源状态是相同的。
安全性:对该 REST 接口访问,不会使服务器端资源的状态发生改变。
传统 URL 请求格式:
http://127.0.0.1/user/query/1 GET 根据用户 id 查询用户数据
http://127.0.0.1/user/save POST 新增用户
http://127.0.0.1/user/update POST 修改用户信息
http://127.0.0.1/user/delete GET/POST 删除用户信息
RESTful 请求格式:
http://127.0.0.1/user/1 GET 根据用户 id 查询用户数据
http://127.0.0.1/user POST 新增用户
http://127.0.0.1/user PUT 修改用户信息
http://127.0.0.1/user DELETE 删除用户信息 很多人在设计 RESTful 架构时,使用很多时间来寻找漂亮的 URI ,而忽略了超媒体。所以,应该多花一
些时间来给资源的表述提供链接,而不是专注于 " 资源的 CRUD" 。
比如下一页的链接地址、资源的链接地址都应该符合 RESTful 的设计思路。
不同的动作都可以有一个唯一的 URl 地址;
3.2.5 状态的转移
原则上要求无状态通信原则。
4**、多方式调用**CoreAPI
- 浏览器访问 -- 只能是 Get 请求
- swagger 访问
- 后台模拟 Http 请求访问 ----- HttpClient / HttpWebRequest
- Core WebAPI 返回结果中文乱码问题解决
- Ajax 请求访问 -- 异步请求【后面讲解 -- 跨域请求】
5**、Swagger配置**
- 基本配置
项目创建之初: - 版本控制
- 注释展示
- 扩展 Token 值传入
- 扩展文件上传按钮
6**、路由解读**
源码项目: Zhaoxi.NET6Demo.WebApi
源码案例: AttributeRouteController 控制器
6.1****什么是路由?
路由器 ---- 转发网络信号
客户端在访问 API 时, API 服务器接收到请求之后,请求的 URL 地址,来通过一定的规则来进行匹配,
然后转发到确定的某一个 API 中,由某一个 API 来做业务处理,这个转发的行为就是路由来完成,这个
匹配的规则就是路由规则。
6.2特性路由Route
在 Core WebAPI 中,每一个 API 必须指定特性路由,即在 API 或者控制器上标记
Route("api/[Controller]/Api") ;访问 API ,就按照这个格式访问;
6.3RESTFul **约束,如果有多个Get、****Post...**操作怎么办?
在一个控制器中可以定义多个 API 方法
通过路由规则来区分
6.3****路由约束
6.4****全局路由扩展
如果在开发中,需要把 API 的请求地址来一个修改,按照目前的做法只能一个一个的修改;
7**、Swagger补充**
源码项目: Zhaoxi.NET6Demo.WebApi
源码案例: VersionControlController 控制器 +SwaggerExtension 类
- 框架支持 API 版本控制
- Swagger 支持 API 版本控制
- 遵循 RESTful 的多版本控制
7.1****框架支持API 版本
8**、内置IOC容器使用**
源码项目: Zhaoxi.NET6Demo.WebApi
源码案例: IOCContainerController 控制器
- 分层架构,上层调用下层
- 具备抽象【接口和抽象类】和实现【普通类】
- 注册抽象和具体之间的关系
- 通过构造函数注入【内置容器仅支持构造函数注入】
9**、API参数修饰特性**
源码项目: Zhaoxi.NET6Demo.WebApi
源码案例: ParameterFromController 控制器
9.1 FromServices
表示来自于 IOC 容器创建
必然需要 IOC 容器先注册
如果没有标记 [FromServices] ,默认会认定这个参数是要通过调用方传递
客户端向服务器发起请求,组装一个固定格式的,符合 http 协议标准的数据格式;发给服务器;
以下几种特性,作用,就是明确指定,参数的数据是来源于 Http 请求的参数中,某一个渠道,可能是头 信息,body ,表单;
9.2 FromBody
api 搜集来自于客户端请求的参数中,从 HTTP Body 中去搜集这个参数的数据,通常用于取 JSON,
XML , 收集到以后,绑定到当前的参数 / 对象中;
FromBody 修饰 -Get 请求 --- 不能访问
FromBody 修饰 -Post 请求 --- 可以访问
FromBody 修饰 -Put 请求 --- 可以访问
FromBody 修饰 -Delete 请求 --- 可以访问
请求 API ,客户端携带的参数为 JSON 格式。 content - type: application/json
9.3 FromForm
api 搜集来自于客户端请求的参数中,到 Form 表单中去搜集这个参数的数据,收集到以后,绑定到当 前的参数/ 对象中;
FromForm 修饰 -Get 请求 --- 不能访问
FromForm 修饰 -Post 请求 --- 可以访问
FromForm 修饰 -Put 请求 --- 可以访问
FromForm 修饰 -Delete 请求 --- 可以访问
9.4 FromHeader
api 搜集来自于客户端请求的参数中,到 Header 头信息中去搜集这个参数的数据,收集到以后,绑定 到当前的参数/ 对象中;
FromHeader 修饰 -Get 请求 --- 可以访问
FromHeader 修饰 -Post 请求 --- 可以访问
FromHeader 修饰 -Put 请求 --- 可以访问
FromHeader 修饰 -Delete 请求 --- 可以访问
9.5 FromQuery
如果客户端通过查询字符串方式传递参数, FromQuery 就是在 Url 地址中去获取值 api 搜集来自于客户端请求的参数中,通过 URL Query 中去搜集这个参数的数据,收集到以后,绑定到 当前的参数/ 对象中;
FromQuery 修饰 -Get 请求
FromQuery 修饰 -Post 请求
FromQuery 修饰 -Put 请求
FromQuery 修饰 -Delete 请求
9.6 FromRouteapi
搜集来自于客户端请求的参数中,在路由中去搜集这个参数的数据,收集到以后,绑定到当前的参
数 / 对象中;
FromRoute 修饰 --Get 请求
FromRoute 修饰 --Post 请求
FromRoute 修饰 --Put 请求
FromRoute 修饰 --Delete 请求
10**、返回结果**
Zhaoxi.NET6Demo.WebApi
ReturnResultController
10.1****返回指定类型(Specific type)
返回指定类型,如果是对象、 int ,默认会返回 Json 格式 --- 经过序列化处理的;
如果是字符串:直接返回字符串;
最简单的 API 会返回原生的或者复杂的数据类型(比如, string 或者自定义对象类型)。考虑如下的
Action 方法,其返回了一个自定义的 User 对象的集合
cs
[HttpGet]
[Route("GetStudentById/{id:int}")]
public Student GetStudentById(int id) => new Student()
{
Id = 1,
Age = 36,
Name = "张三"
};
[HttpGet]
[Route("GetStudentList")]
public IEnumerable<Student> GetStudentList() => new List<Student>()
{
new Student()
{
Id = 1,
Age = 36,
Name = "张三"
}, new Student()
{
Id = 1,
Age = 36,
Name = "李四"
}
};
[HttpGet]
[Route("GetAsyncUserList")]
public async IAsyncEnumerable<Student> GetAsyncUserList([FromServices]
IStudentService studentService)
{
var studentlist = studentService.GetUserListAsync();
// 使用GetAuthorsAsync异步方法,不用authors查询完毕,就会进入下一步迭代返回
authors
await foreach (var student in studentlist)
{
yield return student;
}
}
10.2****返回IActionResult 实例
只要是实现了 IActionResult 的接口的,都可以作为返回值;
cs
[Route("GetStudentJson")]
[HttpGet]
public IActionResult GetStudentJson()
{
//返回Json对象
return new JsonResult(new Student()
{
Id = 1,
Age = 36,
Name = "张三"
});
响应Nofound
//return NotFound("No records");
响应Ok
return Ok(); // 必须有Ok等方法包装
}
10.3****返回ActionResult<T> 实例
cs
[HttpGet]
[Route("GetStudentNew")]
public ActionResult<Student> GetStudentNew()
{
//返回Json对象
return new Student()
{
Id = 1,
Age = 36,
Name = "张三"
};
}
10.4****建议返回类型
WebApi 作为服务存在,需要和第三方对接;
最好能够约束一个标准;固定的返回数据的格式;
固定格式
cs
public class ApiResult<T> where T : class
{
/// <summary>
/// Api执行是否成功
/// </summary>
public bool Success { get; set; }
/// <summary>
/// 错误消息
/// </summary>
public string? Message { get; set; }
/// <summary>
/// 结果集
/// </summary>
public T? Data { get; set; }
}
11**、日志记录**
Zhaoxi.NET6Demo.WebApi
LoggingController
注意:不能允许没有任何监控的系统上线;
如何监控 --- 日志记录
需要日志信息的持久化 -- 保存到文件中,保存到数据库中;
11.1log4net 日志记录
支持文本日志,数据库日志
1 、 Nuget 引入程序包 log4net + Microsoft.Extensions.Logging.Log4Net.AspNetCore
2 、准备配置文件【设置为始终复制】
3 、植入 log4net
4 、注入 log 对象,写日志,写入 txt 文件
11.2****数据库日志
要支持数据库记录日志,日志信息写入到数据库中保存;
1 、修改配置文件,支持数据库
2 、 nuget 引入:写入数据库需要的程序集
3 、执行初始化数据库库脚本
11.2Nlog 日志记录
1 、 nuget 引入: NLog.Web.AspNetCore
2 、准备配置文件
3 、读取配置文件,植入 Nlog
12**、Core WebAPI-**Filter
12.1****什么是AOP ?
AOP(Aspect Orient Programming) ,面向切面编程 , 作为面向对象编程的一种补充 , 可以在不破坏 之前的封装为基础动态增加一些功能;从而让系统更具备扩展性;
增加一个缓存功能
增加日志的功能
既希望不要违背开闭原则,也希望能够增加新的工能;
在之前的业务逻辑之前增加了逻辑;
在之前的业务逻辑之后增加了逻辑;
12.2Core WebAPI 中的AOP 支持有哪些?
Zhaoxi.NET6Demo.WebApi
FiltersController
授权 --- Authorize
资源 -- Resource
异常 -- Exception
方法前后 --- Action
AlwayRunResult
结果前后 --- Result
12.3ResourceFilter 扩展定制
a 定义类、实现 IResourceFilter / IAsyncIResourceFilter 接口,继承 Attribute 父类
b 实现方法
c 标记在 API 方法上
注意:只要是框架提供的是接口,抽象类;我们是不能直接使用的,需要通过扩展;
接口:实现接口,实现方法,把实现类植入;
抽象类:自定类来继承抽象类,覆写方法,把实现类注入;
CustomResourceFilterAttribute 特点:
使用 OnResourceExecuting 方法 + OnResourceExecuted 方法 把要控制器构造函数的执行 + API 调用 给包裹起来了;
12.4ResourceFilter 扩展缓存
ResouceFilter 的特点,适合什么场景应用呢? ---- ResouceFilter 天生就是为了缓存而生的 。
缓存:就是一个临时存储区域,以一个 Key-value 格式保存数据;
key--- 保存数据的标识,也需要这个标识 key 才能获取缓存。
请求来了 --- 在还没有做业务逻辑计算之前 --- 判断缓存是否存在,如果存在,就直接返回缓存的值。
如果不存在,就应该去做计算,计算完毕,把结果保存到缓存中去;
缓存:如果缓存区域中的值没有变化,且 key 不变的,获取的值就应该是之前的值;
url 作为 key---- url 不变,缓存就应该不变;
12.5ActionFilter 扩展定制
a 定义类、实现或者实现 IActionFilter / IAsyncAction / ActionFilterAttribute 抽象类
b 实现方法 / 覆写方法
c 标记在 API 方法上
特点:
和 ResourceFilter 比较:
ResouceFilter 使用两个方法包裹了控制器的构造实例 + API 执行;
ActionFilter 使用两个办法只包裹了 API 逻辑部分;
12.6ActionFilter 场景应用
ActionFilter 适合什么场景应用呢? 缓存? --- 也可以扩展缓存; 为什么说 ResourceFilter 更适合做缓存? --- ResouceFilter 做缓存性能更高; 请求处理的环节会更
少,所以 ResouceFilter 更适合做缓存;
ActionFilter 究竟适合做什么呢? ActionFilter -- 靠近 API 方法; 传入到 API 的参数, API 执行
结束后,执行结果;都是 ActionFilter 最先获取到;记录日志的时候;需要记录下来;
12.7ExceptionFilter 扩展
专门用来统一进行异常的处理;
异常处理在开发中,是一个必须要做的事儿 ~
a 定义类、实现或者实现 IExceptionFilter / IAsyncExceptionFilter 接口
b 实现方法
c 标记在 API 方法上
发生异常就会触发扩展类中的实现方法;
12.8 Filter****的生效范围和依赖注入
前面的 Filter 都是标记在方法上的,如果要大批量的生效怎么办呢?
- 方法注册 ---- 当前方法生效
- 控制器注册 --- 控制器下所有方法生效
- 全局注册 --- 全局生效
TypeFilter 和 ServiceFilter 的解读
12.9ResultFilter/AlwaysRunResultFilter 特点总结
a 定义类、实现或者实现接口
b 实现方法
c 标记在 API 方法上
13**、鉴权授权**
13.1****授权核心价值
1 、保证数据安全,让不同的人做各自的事儿,让管理更便捷
2 、保证系统的安全
Http 协议: 多进程相互通信的一种协议标准,统一这个一个标准后,可以相互通信了;
1 、无状态
Sesssion/Cookes 授权
- 把多个请求和响应建立了联系;
13.2JWT授权/token令牌授权
通过一个令牌,验证这个令牌来确定权限;
https://jwt.io/
13.3****加密解密技术
加密: 可以把一段明文 ( 人类能够识别的信息 )===== 加密技术加密 ====== 得到密文(人类无法识别的 信息);
解密: 可以把一段密文(人类无法识别的信息 ==== 解密技术解密 ====== 得到明文 ( 人类能够识别的信 息)
13.4****可逆加密:
可以加密也可以解密
对称可逆加密
有一个公开的加密解密算法 ( 你知我知大家都知道 ) ,
有一个加密 key 和一个解密 key,
且两个 key 完全相 同;
加密算+ 加密 key 可以加密。
解密算法 + 解密 key 可以解密;
非对称可逆加密
有一个公开的加密解密算法 ( 你知我知大家都知道 ) ,有一个加密 key 和一个解密 key, 且两个 key 不相同,
且两个 key 是成对的,知道加密 key 无法推导出解密 key, 知道解密 key 也不能推导出加密 key;
加密 key+ 加密算法 --- 加密 必须又解密算法 + 和加密 key 成套的解密 key 才能解密;
公钥私钥:
加密 key 做私钥,解密 key 做公钥;
13.5****鉴权授权角色授权策略授权
角色授权,客户端带上 token 来请求 Api , Api 要鉴权,要求角色授权,在带上的 token 中,必须包含 Api授权指定的觉得;
启用角色授权
如果标记多个 Authorize ,指定的角色,是并且的关系;
角色授权的场景:
场景一:如果要验证多个同时具备,同时具备多个 role, 就标记多个 Authorize ,分别把角色写上 , 多个 角色是并且的关系
cs
[Authorize(AuthenticationSchemes= JwtBearerDefaults.AuthenticationScheme,Roles =
"teacher")]
[Authorize(AuthenticationSchemes = JwtBearerDefaults.AuthenticationScheme, Roles =
"admin")]
[Authorize(AuthenticationSchemes = JwtBearerDefaults.AuthenticationScheme, Roles =
"student")]
场景二:多个角色,只要有一个角色匹配即可,多个角色为或者的关系 , 只需要标记一个 Authorize 特 性,roles=" 角色名称以逗号分割 " ,逗号分割的角色名称是或者的关系;
策略:方式,定义多个策略;
启用策略
配置策略生效
13.6鉴权授权扩展Requirement
token 是认证服务器颁发, token 也没有被篡改;
如果我希望去调用第三方的服务来做一些数据验证呢?
Requirement :还是基于策略的授权, Requirement 可以把验证逻辑给封装出来;
1 、定义 Requirement
2 、实现 IAuthorizationHandler --- 直接使用 AuthorizationHandler<> 泛型类
3 、实现 IAuthorizationHandler 接口,
4 、使用 Requirement ,执行的时候,通过 Requirement 找到 IAuthorizationHandler 的实现类,
AuthorizationHandler 实现类可以支持构造函数注入;
5 、标记生效
作用:把验证逻辑另外封装了,可以定义更多的验证逻辑;
14**、发布部署**
- Internet Information Services
- Linux 系统部署
- Docker 部署
14.1Internet Information Services 部署
- 环境安装
cs
安装IIS
安装ASP.NET Core运行时环境:https://dotnet.microsoft.com/zh
cn/download/dotnet/6.0
- 发布项目
- 部署
14.2 Linux****系统发布Core WebAPI
cs
Linux系统-CentOS7 ---基于虚拟机来安装
IP:192.168.1.97
cs
安装教程
链接:https://pan.baidu.com/s/1gI93YwptrWdPKHBh5fLNqQ?pwd=6666
提取码:6666
我们的 Core WebAPI 应用程序是需要跑在 .NET Core 环境上面的。所以我们就需要先安装 .NET Core 环境。
根据微软官方的文档说明:如果要开发 .NET 应用,请安装 SDK (包括运行时)。 或者,如果只需运行 应用程序,请安装运行时。 如果要安装该运行时,建议安装 ASP.NET Core 运行时,因为它同时包括 .NET Core 和 ASP.NET Core 运行时。
命令执行:
1 、签名 密钥 添加到受信任密钥列表,并添加 Microsoft 包存储库
cs
sudo rpm -Uvh https://packages.microsoft.com/config/centos/7/packages-microsoft
prod.rpm
2 、查看可以安装的 dotnet - SDK
cs
yum search dotnet-sdk
3 、安装适合的版本
cs
sudo yum install dotnet-sdk-6.0.3.2
4 、查看安装的环境
cs
dotnet --list-sdks
dotnet --list-runtimes
5 、发布 Core WebAPI 项目 , 上传到 Linux 服务器
6 、命令启动