当前位置: 首页 > news >正文

.NET基于类名约定的自动依赖注入完整指南

🚀 .NET基于类名约定的自动依赖注入完整指南

基于类名约定的自动依赖注入可大幅减少手动注册服务的工作量,本文将通过清晰的结构、美观的排版和丰富的示例,帮助你快速掌握这一实用技术。

🌈 核心特性概览

特性说明
类名约定自动识别以 Service 结尾的类(不区分大小写)
接口优先匹配优先注册到 I{ClassName} 形式的接口(如 UserServiceIUserService
多生命周期支持支持 Transient/Scoped/Singleton 三种生命周期
灵活扫描控制可指定任意程序集或默认扫描调用程序集
无接口自注册自动注册未实现接口的类为自身类型

📦 完整代码实现(含所有重载)

using Microsoft.Extensions.DependencyInjection;
using System;
using System.Linq;
using System.Reflection;/// <summary>
/// 依赖注入扩展方法集合(基于类名约定)
/// </summary>
public static class ServiceCollectionExtensions
{/// <summary>/// 自动注册以"Service"结尾的类(默认Transient生命周期)/// </summary>public static IServiceCollection AutoRegisterServices(this IServiceCollection services,Assembly assembly = null){assembly ??= Assembly.GetCallingAssembly();// 扫描规则:公共类、非抽象、非泛型、类名以Service结尾var serviceTypes = assembly.GetTypes().Where(t => t.IsClass && !t.IsAbstract && t.IsPublic &&!t.ContainsGenericParameters &&t.Name.EndsWith("Service", StringComparison.OrdinalIgnoreCase)).ToArray();// 注册服务到容器foreach (var implType in serviceTypes){var interfaces = implType.GetInterfaces();if (interfaces.Any()){// 匹配I{ClassName}接口(如UserService→IUserService)var matchingInterface = interfaces.FirstOrDefault(i => i.Name == "I" + implType.Name);if (matchingInterface != null){services.AddTransient(matchingInterface, implType);}else{// 注册到所有实现的接口foreach (var @interface in interfaces){services.AddTransient(@interface, implType);}}}else{// 无接口时注册自身services.AddTransient(implType);}}return services;}/// <summary>/// 自动注册以"Service"结尾的类(支持自定义生命周期)/// </summary>public static IServiceCollection AutoRegisterServices(this IServiceCollection services,ServiceLifetime lifetime,Assembly assembly = null){assembly ??= Assembly.GetCallingAssembly();var serviceTypes = GetServiceTypes(assembly);foreach (var implType in serviceTypes){var interfaces = implType.GetInterfaces();var descriptor = CreateServiceDescriptor(implType, interfaces, lifetime);services.Add(descriptor);}return services;}// 辅助方法:获取服务类型(提取公共逻辑)private static Type[] GetServiceTypes(Assembly assembly) =>assembly.GetTypes().Where(t => t.IsClass && !t.IsAbstract && t.IsPublic &&!t.ContainsGenericParameters &&t.Name.EndsWith("Service", StringComparison.OrdinalIgnoreCase)).ToArray();// 辅助方法:创建服务描述符(提取公共逻辑)private static ServiceDescriptor CreateServiceDescriptor(Type implType, Type[] interfaces, ServiceLifetime lifetime){if (interfaces.Any()){var matchingInterface = interfaces.FirstOrDefault(i => i.Name == "I" + implType.Name);if (matchingInterface != null){return new ServiceDescriptor(matchingInterface, implType, lifetime);}// 返回第一个接口(避免注册多个描述符)return new ServiceDescriptor(interfaces[0], implType, lifetime);}return new ServiceDescriptor(implType, implType, lifetime);}
}

🚀 使用示例(清晰排版)

1. 在ASP.NET Core中注册(Program.cs

var builder = WebApplication.CreateBuilder(args);// 方式1:默认Transient(扫描调用程序集)
builder.Services.AutoRegisterServices();// 方式2:指定Scoped生命周期
builder.Services.AutoRegisterServices(ServiceLifetime.Scoped);// 方式3:扫描指定程序集(如业务层)
var businessAssembly = Assembly.Load("MyBusinessLayer");
builder.Services.AutoRegisterServices(ServiceLifetime.Singleton, businessAssembly);var app = builder.Build();

2. 服务类示例(符合约定的实现)

// ✅ 示例1:接口匹配型服务
public interface IUserService { string GetInfo(); }
public class UserService : IUserService 
{public string GetInfo() => "User Service Running";
}// ✅ 示例2:多接口实现服务
public interface IAuthService { void Login(); }
public interface ILogService { void WriteLog(string msg); }
public class AuthService : IAuthService, ILogService 
{public void Login() { /* 登录逻辑 */ }public void WriteLog(string msg) { /* 日志逻辑 */ }
}// ✅ 示例3:无接口自注册服务
public class DataService 
{public void ProcessData() { /* 数据处理 */ }
}// ❌ 示例4:不符合约定的类(不会被注册)
public class ServiceHelper { }        // 类名不以Service结尾
public abstract class BaseService { } // 抽象类

🔧 扩展优化方案(带emoji标记)

1. 🌐 基于类名的生命周期自动识别

private static ServiceLifetime GetLifetimeFromName(string className)
{if (className.Contains("Singleton", StringComparison.OrdinalIgnoreCase))return ServiceLifetime.Singleton;if (className.Contains("Scoped", StringComparison.OrdinalIgnoreCase))return ServiceLifetime.Scoped;return ServiceLifetime.Transient;
}

2. 📌 特性标记增强控制

[AttributeUsage(AttributeTargets.Class)]
public class AutoRegisterAttribute : Attribute 
{public ServiceLifetime Lifetime { get; set; } = ServiceLifetime.Transient;public bool IsEnabled { get; set; } = true;
}

3. ⚡ 缓存扫描结果提升性能

private static readonly object LockObj = new();
private static Type[] _cachedServiceTypes;private static Type[] GetCachedServiceTypes(Assembly assembly)
{if (_cachedServiceTypes == null){lock (LockObj){_cachedServiceTypes = assembly.GetTypes().Where(t => /* 扫描规则 */).ToArray();}}return _cachedServiceTypes;
}

📝 最佳实践指南

  1. 📦 混合注册策略

    • 核心服务(如DbContext)手动注册:
      services.AddDbContext<AppDbContext>(options => {...});
    • 业务服务自动注册:
      builder.Services.AutoRegisterServices();
  2. 🔍 精准扫描范围

    // 仅扫描当前程序集中的服务
    builder.Services.AutoRegisterServices(typeof(UserService).Assembly);
    
  3. ✅ 单元测试验证

    [Fact]
    public void Should_Resolve_Service_By_Convention()
    {var services = new ServiceCollection();services.AutoRegisterServices(typeof(IAuthService).Assembly);var provider = services.BuildServiceProvider();var service = provider.GetService<IAuthService>();Assert.NotNull(service);
    }
    
http://www.lqws.cn/news/448597.html

相关文章:

  • 【AI时代速通QT】第二节:Qt SDK 的目录介绍和第一个Qt Creator项目
  • node.js在vscode的配置
  • 大气商务工作汇报总结PPT模版分享
  • 华为云Flexus+DeepSeek征文 | 利用Dify平台构建多智能体协作系统:从单体到集群的完整方案
  • 以太坊节点搭建私链(POA)
  • davinci本地启动
  • 全面掌握 C++ 基础:关键特性与进化
  • uni-app-配合iOS App项目开发apple watch app
  • 巧用云平台API实现开源模型免费调用的实战教程
  • 电子电气架构 --- 软件供应商如何进入OEM体系
  • Git 命令全景图:从 clone 到 merge 的完整流程解析
  • 基于深度学习的智能视频行为识别系统:技术与实践
  • 【音视频 | RTP】RTP协议详解(H.264的RTP封包格式、AAC的RTP封包格式)
  • CSS3 3D 转换
  • GitHub Copilot 是什么,怎么使用
  • 上海人工智能实验室明珠湖会议首开,解答AI前沿疑问,推进科学智能
  • 【新手向】GitHub Desktop 的使用说明(含 GitHub Desktop 和 Git 的功能对比)
  • java面试题02访问修饰符有哪些?区别是什么?
  • 如何自建服务器并开启公网IP:本地内网网址让外网访问详细教学
  • 华为CE交换机抓包
  • 如何导出和迁移离线 Conda 环境
  • Java八股文——数据结构「排序算法篇」
  • 【目标检测】什么是目标检测?应用场景与基本流程
  • Spring 中的依赖注入(DI)详解
  • Transformer实战——Hugging Face环境配置与应用详解
  • 【编译原理】语句的翻译
  • Docker环境部署
  • Centos 离线部署(MQTT)EMOX脚本并设置开机自启
  • 4、做中学 | 二年级下期 Golang整型和浮点型
  • 高并发网络通信Netty之空轮询问题