在前后端分离的架构中,JWT (JSON Web Token) 是目前最流行的认证方案。它就像一张“电子门禁卡”。用户登录成功后,服务器发给他一张卡(Token),以后每次请求都要带着这张卡,服务器只需验证卡的真伪,而不需要每次都去查数据库。
刚子不喜欢背书,你只需要理解三个核心点:
它由三部分组成(用 . 分隔):

在项目中安装以下 NuGet 包:
dotnet add package Microsoft.AspNetCore.Authentication.JwtBearer
这会引入处理 JWT 验证的核心中间件。
用户登录时,我们需要核实账号密码,然后生成 Token 发给他。为了演示方便,我们假设有一个模拟的用户库。
在 Program.cs 顶部定义模型:
// 定义用户模型
public class User
{
public string Username { get; set; }
public string Password { get; set; } // 实际项目中应存储哈希值
public string Role { get; set; } // 角色:Admin, User
}// 模拟数据库用户
public static class UserStore
{
public static List<User> Users = new List<User>
{
new User { Username = "admin", Password = "123456", Role = "Admin" },
new User { Username = "gangzi", Password = "123456", Role = "User" }
};
}
这个接口负责“发卡”。
using System.IdentityModel.Tokens.Jwt;
using System.Security.Claims;
using System.Text;
using Microsoft.IdentityModel.Tokens;// ... builder 配置 ...app.MapPost("/login", (User login) =>
{
// 1. 验证账号密码
var user = UserStore.Users.FirstOrDefault(u => u.Username == login.Username && u.Password == login.Password);
if (user == null)
{
return Results.Unauthorized(); // 401 未授权
} // 2. 创建 Claims(声明) - 也就是 Payload 里要存的数据
var claims = new List<Claim>
{
new Claim(ClaimTypes.Name, user.Username),
new Claim(ClaimTypes.Role, user.Role), // 存入角色,用于授权
new Claim("MyCustomClaim", "SomeData") // 也可以存自定义数据
}; // 3. 生成签名密钥(实际项目中应从 appsettings.json 读取)
// 密钥至少要 16 个字符
var secretKey = "This_Is_A_Very_Secret_Key_For_JWT_2026!";
var key = new SymmetricSecurityKey(Encoding.UTF8.GetBytes(secretKey));
var creds = new SigningCredentials(key, SecurityAlgorithms.HmacSha256); // 4. 生成 Token
var token = new JwtSecurityToken(
issuer: "MyTodoApp", // 签发者
audience: "MyTodoAppUsers", // 接收者
claims: claims,
expires: DateTime.Now.AddHours(1), // 过期时间:1小时
signingCredentials: creds
); var tokenString = new JwtSecurityTokenHandler().WriteToken(token); return Results.Ok(new { Token = tokenString });
});
刚子敲黑板: 这里的 secretKey 是服务器的“底牌”。绝对不能泄露!如果黑客拿到了这个 Key,他就能伪造任意用户的 Token。在生产环境中,一定要放在 appsettings.json 或环境变量里,并且长度要足够长。
有了发卡的逻辑,还需要配置中间件,让系统自动验证每个请求带来的 Token。
在 Program.cs 的 builder.Services 部分添加:
// 注册认证服务
builder.Services.AddAuthentication(options =>
{
options.DefaultAuthenticateScheme = JwtBearerDefaults.AuthenticationScheme;
options.DefaultChallengeScheme = JwtBearerDefaults.AuthenticationScheme;
})
.AddJwtBearer(options =>
{
// 配置 Token 验证参数
options.TokenValidationParameters = new TokenValidationParameters
{
ValidateIssuer = true,
ValidateAudience = true,
ValidateLifetime = true,
ValidateIssuerSigningKey = true,
ValidIssuer = "MyTodoApp",
ValidAudience = "MyTodoAppUsers",
IssuerSigningKey = new SymmetricSecurityKey(Encoding.UTF8.GetBytes("This_Is_A_Very_Secret_Key_For_JWT_2026!"))
};
});// 注册授权服务
builder.Services.AddAuthorization();
在 app.Build() 之后,确保顺序正确:UseAuthentication 必须在 UseAuthorization 之前。
var app = builder.Build();app.UseAuthentication(); // 开启认证:识别身份
app.UseAuthorization(); // 开启授权:检查权限// ... 其他中间件 ...
现在安检门装好了,我们给之前的 Todo 接口加上“锁”。
在 Minimal API 中,使用 RequireAuthorization() 扩展方法。
// 只有带了有效 Token 才能访问
app.MapGet("/todos", async (AppDbContext db) =>
{
return await db.Todos.ToListAsync();
}).RequireAuthorization();
假设删除操作只有 "Admin" 角色才能执行。
app.MapDelete("/todos/{id}", async (int id, AppDbContext db) =>
{
// ... 删除逻辑 ...
return Results.NoContent();
}).RequireAuthorization("Admin"); // 这是错误的写法!
修正:Minimal API 的角色授权需要配合策略,或者简写为:
// 正确写法:定义一个策略,或者使用 Claims
app.MapDelete("/todos/{id}", async (int id, AppDbContext db) =>
{
// ... 删除逻辑 ...
}).RequireAuthorization(policy => policy.RequireRole("Admin"));
让我们用 Postman 或 Swagger 演示整个流程。
场景一:未登录直接访问
GET /todos。
场景二:登录获取 Token
POST /login,Body 传入 {"username": "admin", "password": "123456"}。
场景三:携带 Token 访问
Authorization: Bearer <你的Token>。(注意 Bearer 后有个空格)GET /todos。
场景四:权限不足
gangzi 用户登录(角色是 User)。DELETE /todos/1。
appsettings.json 的 Jwt:Key 配置项中,并使用 IConfiguration 读取。在这篇文章中,我们完成了系统的安全闭环:
/login 接口颁发 Token。RequireAuthorization 和基于角色的权限控制。