- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
我正在一个典型的三层架构下的新项目中工作:business
、data
和 client
,使用 Angular 作为前端。
在这个项目中,我们将有一个我们想要自动化的重复性任务:CRUD 的创建。我们要做的是生成模型和 Controller (放置、获取、发布、删除)以及来自实体及其属性的其他基本项目信息。
我最好的选择是什么?我曾考虑过模板 T4,但我对它们的无知使我怀疑它是否是最佳选择。
例如,来自这个实体:
public class User
{
public int Id { get; set; }
public string Name {get;set;}
public string Email{ get; set; }
public IEnumerable<Task> Task { get; set; }
}
我想生成以下模型:
public class UserModel
{
public int Id { get; set; }
public string Name {get;set;}
public string Email{ get; set; }
public IEnumerable<Task> Task { get; set; }
}
还有 Controller :
{
/// <summary>
/// User controller
/// </summary>
[Route("api/[controller]")]
public class UserController: Controller
{
private readonly LocalDBContext localDBContext;
private UnitOfWork unitOfWork;
/// <summary>
/// Constructor
/// </summary>
public UserController(LocalDBContext localDBContext)
{
this.localDBContext = localDBContext;
this.unitOfWork = new UnitOfWork(localDBContext);
}
/// <summary>
/// Get user by Id
/// </summary>
[HttpGet("{id}")]
[Produces("application/json", Type = typeof(UserModel))]
public IActionResult GetById(int id)
{
var user = unitOfWork.UserRepository.GetById(id);
if (user == null)
{
return NotFound();
}
var res = AutoMapper.Mapper.Map<UserModel>(user);
return Ok(res);
}
/// <summary>
/// Post an user
/// </summary>
[HttpPost]
public IActionResult Post([FromBody]UserModel user)
{
Usuario u = AutoMapper.Mapper.Map<User>(user);
var res = unitOfWork.UserRepository.Add(u);
if (res?.Id > 0)
{
return Ok(res);
}
return BadRequest();
}
/// <summary>
/// Edit an user
/// </summary>
[HttpPut]
public IActionResult Put([FromBody]UserModel user)
{
if (unitOfWork.UserRepository.GetById(user.Id) == null)
{
return NotFound();
}
var u = AutoMapper.Mapper.Map<User>(user);
var res = unitOfWork.UserRepository.Update(u);
return Ok(res);
}
/// <summary>
/// Delete an user
/// </summary>
[HttpDelete("{id}")]
public IActionResult Delete(int id)
{
if (unitOfWork.UserRepository.GetById(id) == null)
{
return NotFound();
}
unitOfWork.UserRepository.Delete(id);
return Ok();
}
此外,我们需要添加 AutoMapper
映射:
public AutoMapper()
{
CreateMap<UserModel, User>();
CreateMap<User, UserModel>();
}
和工作单元:
private GenericRepository<User> userRepository;
public GenericRepository<User> UserRepository
{
get
{
if (this.userRepository== null)
{
this.userRepository= new GenericRepository<User>(context);
}
return userRepository;
}
}
大多数结构将是相同的,除了一些必须手动完成的 Controller 的特定情况。
最佳答案
这是项目的简化版本,您需要编写该版本才能生成以前的代码。首先创建一个目录,任何 future 的实体都将进入其中。为了简单起见,我将目录命名为 Entities 并创建了一个名为 User.cs 的文件,其中包含 User 类的源代码。
为每个模板创建一个 .tt 文件,该文件以实体名称开头,后跟函数名称。因此,用户模型的 tt 文件将称为 UserModel.tt,您将模型模板放入其中。对于用户 Controller ,USerController.tt,您将在其中放置 Controller 模板。只有 automapper 文件,用户通用存储库将被称为 UserGenericRepository.tt(您已经猜到了)您将通用存储库模板放入其中
模型模板
<#@ template debug="true" hostspecific="true" language="C#" #>
<#@ assembly name="System.Core" #>
<#@ import namespace="System.Linq" #>
<#@ import namespace="System.Text" #>
<#@ import namespace="System.Collections.Generic" #>
<#@ output extension=".cs" #>
<#
var hostFile = this.Host.TemplateFile;
var entityName = System.IO.Path.GetFileNameWithoutExtension(hostFile).Replace("Model","");
var directoryName = System.IO.Path.GetDirectoryName(hostFile);
var fileName = directoryName + "\\Entities\\" + entityName + ".cs";
#>
<#= System.IO.File.ReadAllText(fileName).Replace("public class " + entityName,"public class " + entityName + "Model") #>
我注意到源文件没有命名空间或使用,因此如果不将使用添加到 User.cs 文件,UserModel 文件将无法编译,但是文件确实按照规范生成
Controller 模板
<#@ template debug="true" hostspecific="true" language="C#" #>
<#@ assembly name="System.Core" #>
<#@ import namespace="System.Linq" #>
<#@ import namespace="System.Text" #>
<#@ import namespace="System.Collections.Generic" #>
<#@ output extension=".cs" #>
<#
var hostFile = this.Host.TemplateFile;
var entityName = System.IO.Path.GetFileNameWithoutExtension(hostFile).Replace("Controller","");
var directoryName = System.IO.Path.GetDirectoryName(hostFile);
var fileName = directoryName + "\\" + entityName + ".cs";
#>
/// <summary>
/// <#= entityName #> controller
/// </summary>
[Route("api/[controller]")]
public class <#= entityName #>Controller : Controller
{
private readonly LocalDBContext localDBContext;
private UnitOfWork unitOfWork;
/// <summary>
/// Constructor
/// </summary>
public <#= entityName #>Controller(LocalDBContext localDBContext)
{
this.localDBContext = localDBContext;
this.unitOfWork = new UnitOfWork(localDBContext);
}
/// <summary>
/// Get <#= Pascal(entityName) #> by Id
/// </summary>
[HttpGet("{id}")]
[Produces("application/json", Type = typeof(<#= entityName #>Model))]
public IActionResult GetById(int id)
{
var <#= Pascal(entityName) #> = unitOfWork.<#= entityName #>Repository.GetById(id);
if (<#= Pascal(entityName) #> == null)
{
return NotFound();
}
var res = AutoMapper.Mapper.Map<<#= entityName #>Model>(<#= Pascal(entityName) #>);
return Ok(res);
}
/// <summary>
/// Post an <#= Pascal(entityName) #>
/// </summary>
[HttpPost]
public IActionResult Post([FromBody]<#= entityName #>Model <#= Pascal(entityName) #>)
{
Usuario u = AutoMapper.Mapper.Map<<#= entityName #>>(<#= Pascal(entityName) #>);
var res = unitOfWork.<#= entityName #>Repository.Add(u);
if (res?.Id > 0)
{
return Ok(res);
}
return BadRequest();
}
/// <summary>
/// Edit an <#= Pascal(entityName) #>
/// </summary>
[HttpPut]
public IActionResult Put([FromBody]<#= entityName #>Model <#= Pascal(entityName) #>)
{
if (unitOfWork.<#= entityName #>Repository.GetById(<#= Pascal(entityName) #>.Id) == null)
{
return NotFound();
}
var u = AutoMapper.Mapper.Map<<#= entityName #>>(<#= Pascal(entityName) #>);
var res = unitOfWork.<#= entityName #>Repository.Update(u);
return Ok(res);
}
/// <summary>
/// Delete an <#= Pascal(entityName) #>
/// </summary>
[HttpDelete("{id}")]
public IActionResult Delete(int id)
{
if (unitOfWork.<#= entityName #>Repository.GetById(id) == null)
{
return NotFound();
}
unitOfWork.<#= entityName #>Repository.Delete(id);
return Ok();
}
}
<#+
public string Pascal(string input)
{
return input.ToCharArray()[0].ToString() + input.Substring(1);
}
#>
AutoMapper 的模板
<#@ template debug="true" hostspecific="true" language="C#" #>
<#@ assembly name="System.Core" #>
<#@ import namespace="System.Linq" #>
<#@ import namespace="System.Text" #>
<#@ import namespace="System.Collections.Generic" #>
<#@ output extension=".cs" #>
<#
var directoryName = System.IO.Path.GetDirectoryName(this.Host.TemplateFile) + "\\Entities";
var files = System.IO.Directory.GetFiles(directoryName, "*.cs");
#>
public class AutoMapper
{
<#
foreach(var f in files)
{
var entityName = System.IO.Path.GetFileNameWithoutExtension(f);
#>
CreateMap<<#= entityName #>Model, <#= entityName #>>();
CreateMap<<#= entityName #>, <#= entityName #>Model>();
<#
}
#>}
这基本上遍历 Entities 文件夹中的每个文件并在实体和实体模型之间创建映射器
通用存储库的模板
<#@ template debug="true" hostspecific="true" language="C#" #>
<#@ assembly name="System.Core" #>
<#@ import namespace="System.Linq" #>
<#@ import namespace="System.Text" #>
<#@ import namespace="System.Collections.Generic" #>
<#@ output extension=".cs" #>
<#
var hostFile = this.Host.TemplateFile;
var entityName = System.IO.Path.GetFileNameWithoutExtension(hostFile).Replace("GenericRepository","");
var directoryName = System.IO.Path.GetDirectoryName(hostFile);
var fileName = directoryName + "\\" + entityName + ".cs";
#>
public class GenericRepository
{
private GenericRepository<<#= entityName #>> <#= Pascal(entityName) #>Repository;
public GenericRepository<<#= entityName #>> UserRepository
{
get
{
if (this.<#= Pascal(entityName) #>Repository == null)
{
this.<#= Pascal(entityName) #>Repository = new GenericRepository<<#= entityName #>>(context);
}
return <#= Pascal(entityName) #>Repository;
}
}
}<#+
public string Pascal(string input)
{
return input.ToCharArray()[0].ToString() + input.Substring(1);
}
#>
关于c# - 在 .NET Core 下的分层架构中自动创建 CRUD,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/50345097/
在 Django 中如何处理分层 URL?有什么最佳做法吗?例如。如果我有一个像 /blog/category1/category2/myblogentry 这样的 URL(使用例如 django-m
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 要求我们推荐或查找工具、库或最喜欢的场外资源的问题对于 Stack Overflow 来说是偏离主题的,
有没有办法在 R 中创建这样的图表? 以下是图表中显示的数据的摘录: df % group_by(Animal) %>% unite(col=Type, Animal:Name, sep =
我一直在努力处理一些时间戳数据(直到现在才需要处理日期,并且它表明)。希望您能帮忙。 我正在处理来自网站的数据,该数据显示每个客户 (ID) 各自的访问以及这些访问的时间戳。它的分组是指一个客户可能有
我一直在努力处理一些时间戳数据(直到现在才需要处理日期,并且它表明)。希望您能帮忙。 我正在处理来自网站的数据,该数据显示每个客户 (ID) 各自的访问以及这些访问的时间戳。它的分组是指一个客户可能有
我正在尝试完成这段代码: ORDER BY IF(j.groups IS NULL OR j.groups = '', IF(j.title IS NULL, i.title), j.groups)
我有一个非常抽象的问题,因为我不确定如何提出它。我的其中一个 View 上有一个 UIImageView。我想让 ImageView 看起来“压入 super View ”。我不确定技术术语是什么,但
我希望 100% 宽的包含图像的 div 位于我的页面下方。在这些 div 之上,我想要一个 1210 像素宽的 div,我可以在其中放置我的内容。 例子: http://mudchallenger.
我目前正在做一个类似于 http://www.beoplay.com/Products/BeoplayA9#under-the-hood 的元素使用 Javascript、HTML5 和 CSS3。我
我想像上面那样创建图像缩略图..为此,我在下面创建了 XML activity_main.xml
我想知道是否可以定义一个分层 MapReduce 作业?。换句话说,我想要一个 map-reduce 作业,在 mapper 阶段将调用不同的 MapReduce 作业。可能吗?您对如何操作有什么建议
程序设计: A 类,实现较低级别的数据处理 类 B-E,为 A 提供更高级别的接口(interface)以执行各种功能 F 类,它是根据用户输入与 B-E 交互的 UI 对象 在任何给定时间只能有一个
CTE 对我来说有点新,所以我希望有人可以帮助我编写的以下内容将采用类别表并从中构建层次结构以进行显示。我知道这种事情一直被问到,但我认为我的排序情况使它有点独特。 我希望有一些使用 Hierarch
我有关于 的问题群 在聚类分析(层次聚类)中。例如,这是 的完全链式的树状图。虹膜数据集 . 我使用后 > table(cutree(hc, 3), iris$Species) 这是输出 : se
数据 我有以下(简化的)数据集,我们称之为 df从现在开始: species rank value 1
Delphi 2009 中的分层窗口和系统菜单存在问题。也就是说,我们的分层窗口(没有边框)没有系统菜单。当我说系统菜单时,我指的是单击应用程序的图标、右键单击其标题栏或(在 Windows 7 中,
我正在制作一个 pototype HMTL5 Canvas 动画,该动画将导出到 Quicktime。 我有一个动态生成的背景,上面有动态屏蔽的元素。 我可以获取要制作的背景,并将其作为逐帧动画(pn
好吧,我有一个打印棋盘的类和另一个打印国际象棋的类 如何使用 LayeredPane 将它们合并在一起,如上面的示例图片所示?我一整天都在尝试,但似乎没有任何效果。我正在使用 JFrame 打印图片。
这是我的场景。我有两个类(class) ClassA 和 ClassB。 B类继承A类。 我在它们两个上使用@Component注释来使它们成为Spring bean。 @Component publ
这不是一道问题题,而是一道使用工具——leiningen——的题。 在一个主项目下创建分层的 lein 项目是否有优势,如果有,优势是什么? 如果我使用 lein new bene-cmp 创建一个项
我是一名优秀的程序员,十分优秀!