打造.NET平台的Lombok:实现构造函数注入、日志注入、构造者模式代码生成等功能

在Java生态系统中,Lombok是一个非常受欢迎的库,它通过注解的方式大大减少了Java开发者需要编写的样板代码量。通过简单的注解,如@Data@Getter@Setter@AllArgsConstructor等,开发者可以自动生成getter/setter方法、构造函数、toString方法等。这不仅提高了开发效率,还减少了代码中的冗余,使代码更加简洁和易于维护。

然而,在.NET生态系统中,虽然没有直接等价于Lombok的官方库,但我们可以通过Roslyn源代码生成器来实现类似甚至更强大的功能。本文将介绍如何在.NET平台上构建一个类似Lombok的代码生成工具,实现构造函数注入、日志注入、构造者模式等代码生成功能。

为什么需要类似Lombok的工具?

在现代软件开发中,我们经常需要编写大量重复的样板代码,例如:

  1. 依赖注入:为服务类编写构造函数注入代码
  2. 数据传输对象(DTO):为实体类创建对应的DTO类
  3. Builder模式:为复杂对象创建Builder构建器
  4. 属性访问器:为私有字段生成公共属性
  5. 映射方法:在不同对象之间进行属性映射

这些样板代码不仅占用大量时间编写,还容易出错且难以维护。通过代码生成工具,我们可以自动化这些重复性工作,让开发者专注于业务逻辑的实现。

Mud代码生成器

Mud代码生成器是一套基于Roslyn的源代码生成器,专门针对.NET平台设计,提供了类似Lombok的功能,甚至更加丰富。它包含两个主要组件:

  1. Mud.EntityCodeGenerator:实体代码生成器,用于根据实体类自动生成各种相关代码
  2. Mud.ServiceCodeGenerator:服务代码生成器,用于自动生成服务层相关代码

这套工具通过在代码中添加特定的特性(Attribute)标记,然后在编译时自动生成相应的代码,大大减少了开发者需要手动编写的代码量。

核心功能

1. 生成构造函数注入代码

在.NET的依赖注入系统中,构造函数注入是最推荐的依赖注入方式。然而,手动编写构造函数注入代码可能会很繁琐,特别是当一个类需要注入多个服务时。

Mud.ServiceCodeGenerator提供了多种注入特性,可以自动生成构造函数注入代码:

ConstructorInjectAttribute 字段注入

使用[ConstructorInject]特性可以将类中已存在的私有只读字段通过构造函数注入初始化:

[ConstructorInject] public partial class UserService {     private readonly IUserRepository _userRepository;     private readonly IRoleRepository _roleRepository;          // 生成的代码将包含:     // public UserService(IUserRepository userRepository, IRoleRepository roleRepository)     // {     //     _userRepository = userRepository;     //     _roleRepository = roleRepository;     // } } 

LoggerInjectAttribute 日志注入

使用[LoggerInject]特性可以为类注入ILogger类型的日志记录器:

[LoggerInject] public partial class UserService {     // 生成的代码将包含:     // private readonly ILogger<UserService> _logger;     //      // public UserService(ILoggerFactory loggerFactory)     // {     //     _logger = loggerFactory.CreateLogger<UserService>();     // } } 

CacheInjectAttribute 缓存管理器注入

使用[CacheInject]特性可以注入缓存管理器实例:

[CacheInject] public partial class UserService {     // 生成的代码将包含:     // private readonly ICacheManager _cacheManager;     //      // public UserService(ICacheManager cacheManager)     // {     //     _cacheManager = cacheManager;     // } } 

UserInjectAttribute 用户管理器注入

使用[UserInject]特性可以注入用户管理器实例:

[UserInject] public partial class UserService {     // 生成的代码将包含:     // private readonly IUserManager _userManager;     //      // public UserService(IUserManager userManager)     // {     //     _userManager = userManager;     // } } 

OptionsInjectAttribute 配置项注入

使用[OptionsInject]特性可以根据指定的配置项类型注入配置实例:

[OptionsInject(OptionType = "TenantOptions")] public partial class UserService {     // 生成的代码将包含:     // private readonly TenantOptions _tenantOptions;     //      // public UserService(IOptions<TenantOptions> tenantOptions)     // {     //     _tenantOptions = tenantOptions.Value;     // } } 

CustomInjectAttribute 自定义注入

使用[CustomInject]特性可以注入任意类型的依赖项:

[CustomInject(VarType = "IRepository<SysUser>", VarName = "_userRepository")] [CustomInject(VarType = "INotificationService", VarName = "_notificationService")] public partial class UserService {     // 生成的代码将包含:     // private readonly IRepository<SysUser> _userRepository;     // private readonly INotificationService _notificationService;     //      // public UserService(IRepository<SysUser> userRepository, INotificationService notificationService)     // {     //     _userRepository = userRepository;     //     _notificationService = notificationService;     // } } 

组合注入示例

多种注入特性可以组合使用,生成器会自动合并所有注入需求:

[ConstructorInject] [LoggerInject] [CacheInject] [UserInject] [OptionsInject(OptionType = "TenantOptions")] [CustomInject(VarType = "IRepository<SysUser>", VarName = "_userRepository")] public partial class UserService {     private readonly IRoleRepository _roleRepository;     private readonly IPermissionRepository _permissionRepository;          // 生成的代码将包含所有注入项:     // private readonly ILogger<UserService> _logger;     // private readonly ICacheManager _cacheManager;     // private readonly IUserManager _userManager;     // private readonly TenantOptions _tenantOptions;     // private readonly IRepository<SysUser> _userRepository;     // private readonly IRoleRepository _roleRepository;     // private readonly IPermissionRepository _permissionRepository;     //     // public UserService(     //     ILoggerFactory loggerFactory,     //     ICacheManager cacheManager,     //     IUserManager userManager,     //     IOptions<TenantOptions> tenantOptions,     //     IRepository<SysUser> userRepository,     //     IRoleRepository roleRepository,     //     IPermissionRepository permissionRepository)     // {     //     _logger = loggerFactory.CreateLogger<UserService>();     //     _cacheManager = cacheManager;     //     _userManager = userManager;     //     _tenantOptions = tenantOptions.Value;     //     _userRepository = userRepository;     //     _roleRepository = roleRepository;     //     _permissionRepository = permissionRepository;     // } } 

2. Builder模式代码生成

Builder模式是一种创建型设计模式,能够分步骤创建复杂对象。使用Builder模式可以创建不同表现的对象,同时避免构造函数参数过多的问题。

Mud.EntityCodeGenerator支持通过[Builder]特性自动生成Builder构建器模式代码:

/// <summary> /// 客户端信息实体类 /// </summary> [DtoGenerator] [Builder] [Table(Name = "sys_client"),SuppressSniffer] public partial class SysClientEntity {     /// <summary>     /// id     /// </summary>     [property: Column(Name = "id", IsPrimary = true, Position = 1)]     [property: Required(ErrorMessage = "id不能为空")]     private long? _id;      /// <summary>     /// 客户端key     /// </summary>     [property: Column(Name = "client_key", Position = 3)]     [property: Required(ErrorMessage = "客户端key不能为空")]     private string _clientKey;      /// <summary>     /// 删除标志(0代表存在 2代表删除)     /// </summary>     [property: Column(Name = "del_flag", Position = 10)]     private string _delFlag; } 

基于以上实体,将自动生成Builder构建器类:

/// <summary> /// <see cref="SysClientEntity"/> 的构建者。 /// </summary> public class SysClientEntityBuilder {     private SysClientEntity _sysClientEntity = new SysClientEntity();      /// <summary>     /// 设置 <see cref="SysClientEntity.Id"/> 属性值。     /// </summary>     /// <param name="id">属性值</param>     /// <returns>返回 <see cref="SysClientEntityBuilder"/> 实例</returns>     public SysClientEntityBuilder SetId(long? id)     {         this._sysClientEntity.Id = id;         return this;     }      /// <summary>     /// 设置 <see cref="SysClientEntity.ClientKey"/> 属性值。     /// </summary>     /// <param name="clientKey">属性值</param>     /// <returns>返回 <see cref="SysClientEntityBuilder"/> 实例</returns>     public SysClientEntityBuilder SetClientKey(string clientKey)     {         this._sysClientEntity.ClientKey = clientKey;         return this;     }      /// <summary>     /// 设置 <see cref="SysClientEntity.DelFlag"/> 属性值。     /// </summary>     /// <param name="delFlag">属性值</param>     /// <returns>返回 <see cref="SysClientEntityBuilder"/> 实例</returns>     public SysClientEntityBuilder SetDelFlag(string delFlag)     {         this._sysClientEntity.DelFlag = delFlag;         return this;     }      /// <summary>     /// 构建 <see cref="SysClientEntity"/> 类的实例。     /// </summary>     public SysClientEntity Build()     {         return this._sysClientEntity;     } } 

使用Builder模式可以链式设置实体属性,创建实体对象更加方便:

var client = new SysClientEntityBuilder()     .SetClientKey("client123")     .SetDelFlag("0")     .Build(); 

3. DTO/VO代码生成

在现代Web应用开发中,数据传输对象(DTO)和视图对象(VO)是常见的设计模式。它们用于在不同层之间传输数据,避免直接暴露实体类。

Mud.EntityCodeGenerator可以自动生成DTO和VO类:

/// <summary> /// 客户端信息实体类 /// </summary> [DtoGenerator] [Table(Name = "sys_client"),SuppressSniffer] public partial class SysClientEntity {     /// <summary>     /// id     /// </summary>     [property: TableField(Fille = FieldFill.Insert, Value = FillValue.Id)]     [property: Column(Name = "id", IsPrimary = true, Position = 1)]     [property: Required(ErrorMessage = "id不能为空")]     private long? _id;      /// <summary>     /// 客户端key     /// </summary>     [property: Column(Name = "client_key", Position = 3)]     [property: Required(ErrorMessage = "客户端key不能为空")]     [property: ExportProperty("客户端key")]     [property: CustomVo1, CustomVo2]     [property: CustomBo1, CustomBo2]     private string _clientKey;      /// <summary>     /// 删除标志(0代表存在 2代表删除)     /// </summary>     [property: Column(Name = "del_flag", Position = 10)]     [property: ExportProperty("删除标志")]     [IgnoreQuery]     private string _delFlag; } 

基于以上实体,将自动生成以下几类代码:

VO类 (视图对象)

/// <summary> /// 客户端信息实体类 /// </summary> [SuppressSniffer, CompilerGenerated] public partial class SysClientListOutput {     /// <summary>     /// id     /// </summary>     public long? id { get; set; }      /// <summary>     /// 客户端key     /// </summary>     [ExportProperty("客户端key")]     [CustomVo1, CustomVo2]     public string? clientKey { get; set; }      /// <summary>     /// 删除标志(0代表存在 2代表删除)     /// </summary>     [ExportProperty("删除标志")]     public string? delFlag { get; set; } } 

QueryInput类 (查询输入对象)

// SysClientQueryInput.g.cs /// <summary> /// 客户端信息实体类 /// </summary> [SuppressSniffer, CompilerGenerated] public partial class SysClientQueryInput : DataQueryInput {     /// <summary>     /// id     /// </summary>     public long? id { get; set; }     /// <summary>     /// 客户端key     /// </summary>     public string? clientKey { get; set; }     /// <summary>     /// 删除标志(0代表存在 2代表删除)     /// </summary>     public string? delFlag { get; set; }      /// <summary>     /// 构建通用的查询条件。     /// </summary>     public Expression<Func<SysClientEntity, bool>> BuildQueryWhere()     {         var where = LinqExtensions.True<SysClientEntity>();         where = where.AndIF(this.id != null, x => x.Id == this.id);         where = where.AndIF(!string.IsNullOrEmpty(this.clientKey), x => x.ClientKey == this.clientKey);         where = where.AndIF(!string.IsNullOrEmpty(this.delFlag), x => x.DelFlag == this.delFlag);         return where;     } } 

CrInput类 (创建输入对象)

// SysClientCrInput.g.cs /// <summary> /// 客户端信息实体类 /// </summary> [SuppressSniffer, CompilerGenerated] public partial class SysClientCrInput {     /// <summary>     /// 客户端key     /// </summary>     [Required(ErrorMessage = "客户端key不能为空"), CustomBo1, CustomBo2]     public string? clientKey { get; set; }     /// <summary>     /// 删除标志(0代表存在 2代表删除)     /// </summary>     public string? delFlag { get; set; }      /// <summary>     /// 通用的BO对象映射至实体方法。     /// </summary>     public virtual SysClientEntity MapTo()     {         var entity = new SysClientEntity();         entity.ClientKey = this.clientKey;         entity.DelFlag = this.delFlag;         return entity;     } } 

UpInput类 (更新输入对象)

/// <summary> /// 客户端信息实体类 /// </summary> [SuppressSniffer, CompilerGenerated] public partial class SysClientUpInput : SysClientCrInput {     /// <summary>     /// id     /// </summary>     [Required(ErrorMessage = "id不能为空")]     public long? id { get; set; }      /// <summary>     /// 通用的BO对象映射至实体方法。     /// </summary>     public override SysClientEntity MapTo()     {         var entity = base.MapTo();         entity.Id = this.id;         return entity;     } } 

4. 实体映射方法生成

在不同对象之间进行属性映射是一项常见但繁琐的工作。Mud.EntityCodeGenerator可以自动生成实体与DTO之间的映射方法:

/// <summary> /// 通用的实体映射至VO对象方法。 /// </summary> public virtual SysClientListOutput MapTo() {     var voObj = new SysClientListOutput();     voObj.id = this.Id;     voObj.clientKey = this.ClientKey;     voObj.delFlag = this.DelFlag;     return voObj; } 

配置和使用

项目配置

在使用Mud代码生成器时,可以通过在项目文件中配置参数来自定义生成行为。

实体代码生成器配置参数

<PropertyGroup>   <EmitCompilerGeneratedFiles>true</EmitCompilerGeneratedFiles>  <!-- 在obj目录下保存生成的代码 -->   <EntitySuffix>Entity</EntitySuffix>  <!-- 实体类后缀配置 -->   <EntityAttachAttributes>SuppressSniffer</EntityAttachAttributes>  <!-- 生成的VO、BO类加上Attribute特性配置,多个特性时使用','分隔 -->      <!-- 属性名配置 -->   <PropertyNameLowerCaseFirstLetter>true</PropertyNameLowerCaseFirstLetter>  <!-- 是否将生成的属性名首字母小写,默认为true -->      <!-- VO/BO 属性配置参数 -->   <VoAttributes>CustomVo1Attribute,CustomVo2Attribute</VoAttributes>  <!-- 需要添加至VO类的自定义特性,多个特性时使用','分隔 -->   <BoAttributes>CustomBo1Attribute,CustomBo2Attribute</BoAttributes>  <!-- 需要添加至BO类的自定义特性,多个特性时使用','分隔 --> </PropertyGroup>  <ItemGroup>   <CompilerVisibleProperty Include="EntitySuffix" />   <CompilerVisibleProperty Include="EntityAttachAttributes" />   <CompilerVisibleProperty Include="PropertyNameLowerCaseFirstLetter" />   <CompilerVisibleProperty Include="VoAttributes" />   <CompilerVisibleProperty Include="BoAttributes" /> </ItemGroup> 

服务代码生成器配置参数

<PropertyGroup>   <EmitCompilerGeneratedFiles>true</EmitCompilerGeneratedFiles>  <!-- 在obj目录下保存生成的代码 -->      <!-- 依赖注入相关配置 -->   <DefaultCacheManagerType>ICacheManager</DefaultCacheManagerType>  <!-- 缓存管理器类型默认值 -->   <DefaultUserManagerType>IUserManager</DefaultUserManagerType>  <!-- 用户管理器类型默认值 -->   <DefaultLoggerVariable>_logger</DefaultLoggerVariable>  <!-- 日志变量名默认值 -->   <DefaultCacheManagerVariable>_cacheManager</DefaultCacheManagerVariable>  <!-- 缓存管理器变量名默认值 -->   <DefaultUserManagerVariable>_userManager</DefaultUserManagerVariable>  <!-- 用户管理器变量名默认值 -->      <!-- 服务生成相关配置 -->   <ServiceGenerator>true</ServiceGenerator>  <!-- 是否生成服务端代码 -->   <EntitySuffix>Entity</EntitySuffix>  <!-- 实体类后缀配置 -->   <ImpAssembly>Mud.System</ImpAssembly>  <!-- 需要生成代码的接口实现程序集 -->      <!-- DTO生成相关配置 -->   <EntityAttachAttributes>SuppressSniffer</EntityAttachAttributes>  <!-- 实体类加上Attribute特性配置,多个特性时使用','分隔 --> </PropertyGroup>  <ItemGroup>   <CompilerVisibleProperty Include="DefaultCacheManagerType" />   <CompilerVisibleProperty Include="DefaultUserManagerType" />   <CompilerVisibleProperty Include="DefaultLoggerVariable" />   <CompilerVisibleProperty Include="DefaultCacheManagerVariable" />   <CompilerVisibleProperty Include="DefaultUserManagerVariable" />   <CompilerVisibleProperty Include="ServiceGenerator" />   <CompilerVisibleProperty Include="EntitySuffix" />   <CompilerVisibleProperty Include="ImpAssembly" />   <CompilerVisibleProperty Include="EntityAttachAttributes" /> </ItemGroup> 

依赖项配置

<ItemGroup>   <!-- 引入的代码生成器程序集 -->   <PackageReference Include="Mud.EntityCodeGenerator" Version="1.1.8" />   <PackageReference Include="Mud.ServiceCodeGenerator" Version="1.1.8" /> </ItemGroup> 

高级特性

忽略字段注入

对于某些不需要通过构造函数注入的字段,可以使用[IgnoreGenerator]特性标记:

[ConstructorInject] public partial class UserService {     private readonly IUserRepository _userRepository;          [IgnoreGenerator]     private readonly string _connectionString = "default_connection_string"; // 不会被注入          // 只有_userRepository会被构造函数注入 } 

自定义属性生成

Mud代码生成器支持通过配置参数为生成的类添加自定义特性:

<PropertyGroup>   <!-- 需要添加至VO类的自定义特性,多个特性时使用','分隔 -->   <VoAttributes>CustomVo1Attribute,CustomVo2Attribute</VoAttributes>   <!-- 需要添加至BO类的自定义特性,多个特性时使用','分隔 -->   <BoAttributes>CustomBo1Attribute,CustomBo2Attribute</BoAttributes> </PropertyGroup> 

与其他工具的比较

与AutoMapper的比较

AutoMapper是一个流行的对象映射工具,但它运行时进行映射,而Mud代码生成器在编译时生成映射代码。这意味着:

  1. 性能:Mud生成的代码在运行时性能更好,因为没有反射开销
  2. 类型安全:编译时生成的代码具有更好的类型安全性
  3. 调试友好:生成的代码可以直接调试,更容易排查问题

与传统手工编码的比较

  1. 开发效率:大大减少了样板代码的编写时间
  2. 维护性:当实体类发生变化时,相关代码会自动更新
  3. 一致性:生成的代码风格统一,减少了人为错误

最佳实践

1. 合理使用特性标记

不要为所有类都添加代码生成特性,只在确实需要的类上使用。过度使用可能导致生成大量不必要的代码。

2. 配置参数优化

根据项目实际情况配置生成参数,例如:

<PropertyGroup>   <EmitCompilerGeneratedFiles>true</EmitCompilerGeneratedFiles>   <EntitySuffix>Entity</EntitySuffix>   <PropertyNameLowerCaseFirstLetter>false</PropertyNameLowerCaseFirstLetter> </PropertyGroup> 

3. 查看生成代码

在开发阶段,建议启用[EmitCompilerGeneratedFiles]参数,以便查看生成的代码:

<PropertyGroup>   <EmitCompilerGeneratedFiles>true</EmitCompilerGeneratedFiles> </PropertyGroup> 

生成的代码将位于obj/[Configuration]/[TargetFramework]/generated/目录下,文件名以.g.cs结尾。

4. 版本管理

生成的代码不需要加入版本管理,因为它们会在编译时自动生成。可以在.gitignore中添加:

**/*.g.cs 

实际应用案例

案例1:用户服务类

[ConstructorInject] [LoggerInject] [CacheInject] [UserInject] public partial class UserService {     private readonly IUserRepository _userRepository;     private readonly IRoleRepository _roleRepository;          public async Task<UserDto> GetUserAsync(long userId)     {         _logger.LogInformation("Getting user with id: {UserId}", userId);                  var user = await _userRepository.GetByIdAsync(userId);         if (user == null)         {             _logger.LogWarning("User with id {UserId} not found", userId);             return null;         }                  var userDto = user.MapTo();         return userDto;     } } 

案例2:订单实体类

[DtoGenerator] [Builder] public partial class OrderEntity {     [property: Column(Name = "id", IsPrimary = true)]     [property: Required]     private long? _id;          [property: Column(Name = "order_no")]     [property: Required]     private string _orderNo;          [property: Column(Name = "amount")]     [property: Required]     private decimal? _amount;          [property: Column(Name = "status")]     private string _status; } 

性能和安全性考虑

性能优势

  1. 编译时生成:所有代码在编译时生成,运行时无额外开销
  2. 无反射调用:生成的代码直接调用,避免反射带来的性能损耗
  3. 类型安全:编译时检查确保类型安全,减少运行时错误

安全性考虑

  1. 代码审查:虽然代码是自动生成的,但仍需要审查生成的代码以确保符合安全要求
  2. 依赖注入:通过构造函数注入确保依赖关系明确,便于测试和维护
  3. 访问控制:生成的属性和方法遵循.NET的访问控制原则

扩展和定制

Mud代码生成器设计为可扩展的,开发者可以根据自己的需求定制代码生成逻辑:

  1. 自定义特性:可以创建自己的特性来控制代码生成行为
  2. 模板定制:可以修改代码生成模板以适应特定需求
  3. 插件机制:可以通过插件机制添加新的代码生成功能

总结

通过Mud代码生成器,我们可以在.NET平台上实现类似Java Lombok的功能,甚至更加丰富和强大。这套工具通过Roslyn源代码生成技术,在编译时自动生成我们需要的样板代码,大大提高了开发效率,减少了手动编写代码的工作量。

主要优势包括:

  1. 提高开发效率:自动生成构造函数注入、Builder模式、DTO等代码
  2. 保证代码质量:生成的代码遵循统一规范,减少人为错误
  3. 提升性能:编译时生成,运行时无额外开销
  4. 易于维护:当源代码变化时,相关代码自动更新

通过合理使用这套工具,我们可以专注于业务逻辑的实现,而将重复性的样板代码交给代码生成器处理,真正实现高效、高质量的软件开发。

无论你是正在开发新的.NET项目,还是想要重构现有项目以减少样板代码,Mud代码生成器都是一个值得考虑的强大工具。它不仅能够显著提高开发效率,还能帮助团队保持代码的一致性和可维护性。

开始使用Mud代码生成器,让你的.NET开发体验更加接近Java Lombok带来的便利,甚至更进一步!

发表评论

评论已关闭。

相关文章