使用 EF Core 和 Blazor 实现设备类别关联修改 使用efcore
和blazor
连接数据库,数据库有两个表,一个设备,一个类别,设备通过一个整数关联到类别表。
类别表有类别id、类别名称两个字段,设备有设备id、设备名、类别id字段。
在前端展示所有的设备,其类别显示为一个下拉框,我可以通过下拉框改变设备的类别,并通过按钮保存。
创建数据模型 在Models
文件夹下创建Category.cs
和Device.cs
文件。
Category.cs
1 2 3 4 5 6 7 8 namespace YourNamespace.Models { public class Category { public int CategoryId { get ; set ; } public string CategoryName { get ; set ; } } }
Device.cs
1 2 3 4 5 6 7 8 9 10 11 namespace YourNamespace.Models { public class Device { public int DeviceId { get ; set ; } public string DeviceName { get ; set ; } public int CategoryId { get ; set ; } public Category Category { get ; set ; } } }
Blazor 组件 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 <table class ="table" > <thead> <tr> <th>设备名</th> <th>类别</th> <th>操作</th> </tr> </thead> <tbody> @foreach (var device in devices) { <tr> <td>@device.DeviceName</td> <td> <select @bind-value ="device.CategoryId" > @foreach (var category in categories) { <option value ="@category.CategoryId" >@category.CategoryName</option> } </select > </td> <td> <button class ="btn btn-primary" @onclick="@(async () => await UpdateExistingEntry(device))" >保存</button> </td> </tr> } </tbody> </table> @code { private List<DbContext.Category> categories = new (); private List<DbContext.Device> devices = new (); protected override void OnInitialized () { UpdateDeviceList(); } private void UpdateDeviceList () { using (var db = new ChargeDb()) { categories = db.Categories.ToList(); devices = db.Devices.Include(d => d.Category) .ToList(); } } private async Task UpdateExistingEntry (DbContext.Device device ) { using (var db = new ChargeDb()) { var existingDevice = db.Devices.Include(d => d.Category).FirstOrDefault(d => d.Id == device.Id); existingDevice.Description = device.Description; existingDevice.ImageUrl = device.ImageUrl; existingDevice.Name = device.Name; existingDevice.CategoryId = device.CategoryId; db.SaveChanges(); Log.Information("Device updated: dev {Id} - cat {CategoryId}" , device.Id, device.CategoryId); } UpdateDeviceList(); }
重点
ef core
实体模型在设置 Category
导航属性时,同时设置一个叫 CategoryId
的 int
属性,这样在前端做数据绑定时,绑定这个属性。
select
绑定 device
的 CategoryId
,select
内部的 option
绑定 category.CategoryId
附:EF Core 命名约定 考虑
1 2 3 4 5 6 7 8 9 10 11 12 public class Blog { public int Id { get ; set ; } public ICollection<Post> Posts { get ; } = new List<Post>(); }public class Post { public int Id { get ; set ; } public int ? BlogId { get ; set ; } public Blog? Blog { get ; set ; } }
Blog
类包含一个 ICollection<Post>
类型的 Posts
属性,这表明一个 Blog
可以包含多个 Post
,是一对多关系中的 “一” 方。
Post
类包含一个可空的 int? BlogId
属性和一个可空的 Blog? Blog
属性。BlogId
属性名遵循 EF Core 的外键命名约定,即 [导航属性名][主表主键名]
,在这种情况下,导航属性名是 Blog
,主表(Blog
类)的主键名是 Id
,所以 BlogId
会被 EF Core 自动识别为外键。
完整示例如下
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 using Microsoft.EntityFrameworkCore;using System.Collections.Generic;using System.Linq;using System.Threading.Tasks;public class Blog { public int Id { get ; set ; } public ICollection<Post> Posts { get ; } = new List<Post>(); }public class Post { public int Id { get ; set ; } public int ? BlogId { get ; set ; } public Blog? Blog { get ; set ; } }public class BloggingContext : DbContext { public DbSet<Blog> Blogs { get ; set ; } public DbSet<Post> Posts { get ; set ; } protected override void OnConfiguring (DbContextOptionsBuilder optionsBuilder ) { optionsBuilder.UseSqlite("Data Source=blogging.db" ); } }class Program { static async Task Main () { using (var context = new BloggingContext()) { await context.Database.EnsureCreatedAsync(); var blog = new Blog(); context.Blogs.Add(blog); var post = new Post { Blog = blog }; context.Posts.Add(post); await context.SaveChangesAsync(); var retrievedPost = await context.Posts .Include(p => p.Blog) .FirstOrDefaultAsync(); if (retrievedPost != null && retrievedPost.Blog != null ) { System.Console.WriteLine($"Post {retrievedPost.Id} is associated with Blog {retrievedPost.Blog.Id} " ); } } } }