当用户仅提供部分字段时发布实体 class
Posting entity class when only some fields are provided by user
我很好奇人们是如何处理这个问题的。
我已经使用 Visual Studio 将带有 Entity Framework 的 CRUD 页面添加到我的 Razor Pages 应用程序。
但是,Create 页面无法验证。那是因为该实体具有诸如用户 ID、具有默认值的创建日期以及我想要设置默认值的几个字段等字段。所以 ModelState.IsValid
显然是要 return false
。而且似乎很少有用户会为实体的每个字段提供完全正确的数据。
处理这个问题的一个选择是创建一个新的 class,它只包含我需要的用户字段,然后将它们复制到实体中,然后再将其保存到数据库中。
另一种选择是编写代码,在调用 ModelState.IsValid
之前填充不依赖于用户输入的字段。
这对我来说是一个新情况。我很想听听其他人是如何处理这种情况的。
我个人总是通过在前端使用一个单独的 class 来抽象 EF class 来处理这种情况。这提供了以下好处:
- 仅允许用户在保存前填写某些字段(例如您的案例),但不允许用户填写 ID、时间戳等其他字段。
- 减少耦合 - 允许更轻松地重构前端或后端代码,而无需也重构另一个(前提是前端 class 不变)。
- [编辑] 这可以防止任何用户编辑 ID 等不应由用户更新的字段。
这样做的一个缺点是它会导致相似 class 之间的许多映射,但是自动映射器工具可以在一定程度上缓解这种情况。
我使用隐藏输入。
<hidden asp-for="UserId" value=@GetUser() />
<hidden asp-for="CreatedDate" value=@DateTime.Now.ToString("mm/dd/yyyy") />
它们将作为表单的一部分与模型的其余部分一起提交。这是最简单的解决方案,imo。
在我看来,如果您可以使用 AutoMapper
而不是复制值,那么您的第一个选择会更加简单和安全。
参考下面使用asp.net core 3.0 Razor Pages 的演示:
1.Install-Package AutoMapper.Extensions.Microsoft.DependencyInjection
2.Create模型(保存到数据库)和ViewModel(显示在用户页面)
public class Movie
{
public Movie()
{
Genre = "Action";
}
public int ID { get; set; }
public string Title { get; set; }
[DataType(DataType.Date)]
public DateTime ReleaseDate { get; set; }
public string Genre { get; set; }
}
public class MovieViewModel
{
public string Title { get; set; }
[DataType(DataType.Date)]
public DateTime ReleaseDate { get; set; }
}
3.Create AutoMapper Profile
class
public class MovieProfile: Profile
{
public MovieProfile()
{
CreateMap<MovieViewModel, Movie>().ReverseMap();
}
}
4.Create.cshtml.cs
public class CreateModel : PageModel
{
private readonly ApplicationContext _context;
private readonly IMapper _mapper;
public CreateModel(ApplicationContext context, IMapper mapper)
{
_context = context;
_mapper = mapper;
}
public IActionResult OnGet()
{
return Page();
}
[BindProperty]
public MovieViewModel MovieVM { get; set; }
public async Task<IActionResult> OnPostAsync()
{
if (!ModelState.IsValid)
{
return Page();
}
Movie movie = _mapper.Map<Movie>(MovieVM);
_context.Movie.Add(movie);
await _context.SaveChangesAsync();
return RedirectToPage("./Index");
}
}
Create.cshtml
@page
@model RazorpagesCore.Pages.Movies.CreateModel
<div class="row">
<div class="col-md-4">
<form method="post">
<div asp-validation-summary="ModelOnly" class="text-danger"></div>
<div class="form-group">
<label asp-for="MovieVM.Title" class="control-label"></label>
<input asp-for="MovieVM.Title" class="form-control" />
<span asp-validation-for="MovieVM.Title" class="text-danger"></span>
</div>
<div class="form-group">
<label asp-for="MovieVM.ReleaseDate" class="control-label"></label>
<input asp-for="MovieVM.ReleaseDate" class="form-control" />
<span asp-validation-for="MovieVM.ReleaseDate" class="text-danger"></span>
</div>
<div class="form-group">
<input type="submit" value="Create" class="btn btn-primary" />
</div>
</form>
</div>
</div>
@section Scripts {
@{await Html.RenderPartialAsync("_ValidationScriptsPartial");}
}
我很好奇人们是如何处理这个问题的。
我已经使用 Visual Studio 将带有 Entity Framework 的 CRUD 页面添加到我的 Razor Pages 应用程序。
但是,Create 页面无法验证。那是因为该实体具有诸如用户 ID、具有默认值的创建日期以及我想要设置默认值的几个字段等字段。所以 ModelState.IsValid
显然是要 return false
。而且似乎很少有用户会为实体的每个字段提供完全正确的数据。
处理这个问题的一个选择是创建一个新的 class,它只包含我需要的用户字段,然后将它们复制到实体中,然后再将其保存到数据库中。
另一种选择是编写代码,在调用 ModelState.IsValid
之前填充不依赖于用户输入的字段。
这对我来说是一个新情况。我很想听听其他人是如何处理这种情况的。
我个人总是通过在前端使用一个单独的 class 来抽象 EF class 来处理这种情况。这提供了以下好处:
- 仅允许用户在保存前填写某些字段(例如您的案例),但不允许用户填写 ID、时间戳等其他字段。
- 减少耦合 - 允许更轻松地重构前端或后端代码,而无需也重构另一个(前提是前端 class 不变)。
- [编辑] 这可以防止任何用户编辑 ID 等不应由用户更新的字段。
这样做的一个缺点是它会导致相似 class 之间的许多映射,但是自动映射器工具可以在一定程度上缓解这种情况。
我使用隐藏输入。
<hidden asp-for="UserId" value=@GetUser() />
<hidden asp-for="CreatedDate" value=@DateTime.Now.ToString("mm/dd/yyyy") />
它们将作为表单的一部分与模型的其余部分一起提交。这是最简单的解决方案,imo。
在我看来,如果您可以使用 AutoMapper
而不是复制值,那么您的第一个选择会更加简单和安全。
参考下面使用asp.net core 3.0 Razor Pages 的演示:
1.Install-Package AutoMapper.Extensions.Microsoft.DependencyInjection
2.Create模型(保存到数据库)和ViewModel(显示在用户页面)
public class Movie
{
public Movie()
{
Genre = "Action";
}
public int ID { get; set; }
public string Title { get; set; }
[DataType(DataType.Date)]
public DateTime ReleaseDate { get; set; }
public string Genre { get; set; }
}
public class MovieViewModel
{
public string Title { get; set; }
[DataType(DataType.Date)]
public DateTime ReleaseDate { get; set; }
}
3.Create AutoMapper Profile
class
public class MovieProfile: Profile
{
public MovieProfile()
{
CreateMap<MovieViewModel, Movie>().ReverseMap();
}
}
4.Create.cshtml.cs
public class CreateModel : PageModel
{
private readonly ApplicationContext _context;
private readonly IMapper _mapper;
public CreateModel(ApplicationContext context, IMapper mapper)
{
_context = context;
_mapper = mapper;
}
public IActionResult OnGet()
{
return Page();
}
[BindProperty]
public MovieViewModel MovieVM { get; set; }
public async Task<IActionResult> OnPostAsync()
{
if (!ModelState.IsValid)
{
return Page();
}
Movie movie = _mapper.Map<Movie>(MovieVM);
_context.Movie.Add(movie);
await _context.SaveChangesAsync();
return RedirectToPage("./Index");
}
}
Create.cshtml
@page
@model RazorpagesCore.Pages.Movies.CreateModel
<div class="row">
<div class="col-md-4">
<form method="post">
<div asp-validation-summary="ModelOnly" class="text-danger"></div>
<div class="form-group">
<label asp-for="MovieVM.Title" class="control-label"></label>
<input asp-for="MovieVM.Title" class="form-control" />
<span asp-validation-for="MovieVM.Title" class="text-danger"></span>
</div>
<div class="form-group">
<label asp-for="MovieVM.ReleaseDate" class="control-label"></label>
<input asp-for="MovieVM.ReleaseDate" class="form-control" />
<span asp-validation-for="MovieVM.ReleaseDate" class="text-danger"></span>
</div>
<div class="form-group">
<input type="submit" value="Create" class="btn btn-primary" />
</div>
</form>
</div>
</div>
@section Scripts {
@{await Html.RenderPartialAsync("_ValidationScriptsPartial");}
}