using Microsoft.Extensions.DependencyInjection;
using System;
using System.Linq;
using System.Reflection; // 自定义属性来标记服务类型
[AttributeUsage(AttributeTargets.Class, AllowMultiple = false, Inherited = false)]
public class ServiceTypeAttribute : Attribute
{
public ServiceLifetime Lifetime { get; } public ServiceTypeAttribute(ServiceLifetime lifetime)
{
Lifetime = lifetime;
}
} public static class DependencyInjectionExtensions
{
/// <summary>
/// 批量注册服务
/// </summary>
/// <param name="services">IServiceCollection</param>
/// <param name="servicePrefix">服务类的前缀,例如 "Service"</param>
/// <param name="interfacePrefix">接口的前缀,例如 "I"</param>
public static void RegisterServices(this IServiceCollection services, string servicePrefix, string interfacePrefix)
{
if (services == null)
{
throw new ArgumentNullException(nameof(services));
} // 获取当前程序集中所有类
var types = Assembly.GetExecutingAssembly().GetTypes(); // 筛选出符合服务命名约定的类
var serviceTypes = types
.Where(t => t.Name.StartsWith(servicePrefix) && !t.IsInterface && !t.IsAbstract)
.ToList(); // 遍历所有服务类
foreach (var serviceType in serviceTypes)
{
// 尝试查找对应的接口
var interfaceType = types.FirstOrDefault(t => t.Name == interfacePrefix + serviceType.Name.Substring(servicePrefix.Length) && t.IsInterface); if (interfaceType != null)
{
// 获取服务类型上的 ServiceTypeAttribute
var serviceTypeAttribute = serviceType.GetCustomAttribute<ServiceTypeAttribute>();
var lifetime = serviceTypeAttribute?.Lifetime ?? ServiceLifetime.Scoped; // 默认使用 Scoped // 根据不同的生命周期注册服务
switch (lifetime)
{
case ServiceLifetime.Singleton:
services.AddSingleton(interfaceType, serviceType);
break;
case ServiceLifetime.Scoped:
services.AddScoped(interfaceType, serviceType);
break;
case ServiceLifetime.Transient:
services.AddTransient(interfaceType, serviceType);
break;
default:
services.AddScoped(interfaceType, serviceType); // 默认使用 Scoped
break;
}
}
else
{
// 如果没有找到对应的接口,则可以选择只注册服务本身,或者抛出异常
// 这里选择抛出异常,提示开发者需要有对应的接口
throw new InvalidOperationException($"Service {serviceType.Name} does not have a corresponding interface (expected interface name: {interfacePrefix}{serviceType.Name.Substring(servicePrefix.Length)})");
}
}
}
} // 示例接口和服务
public interface IUserService
{
void DoSomething();
} [ServiceType(ServiceLifetime.Scoped)] // 使用属性标记生命周期
public class UserService : IUserService
{
public void DoSomething()
{
Console.WriteLine("UserService.DoSomething() called.");
}
} public interface IOrderService
{
void ProcessOrder();
} [ServiceType(ServiceLifetime.Transient)]
public class OrderService: IOrderService
{
public void ProcessOrder()
{
Console.WriteLine("OrderService.ProcessOrder() called.");
}
} public class Program
{
public static void Main(string[] args)
{
// 创建 ServiceCollection
IServiceCollection services = new ServiceCollection(); // 注册服务
services.RegisterServices("Service", "I"); // 添加其他必要的服务,例如 MVC
services.AddMvc(); // 如果是 ASP.NET Core MVC 项目 // 构建 ServiceProvider
IServiceProvider serviceProvider = services.BuildServiceProvider(); // 从 ServiceProvider 中获取服务并使用
var userService = serviceProvider.GetService<IUserService>();
userService?.DoSomething(); var orderService = serviceProvider.GetService<IOrderService>();
orderService?.ProcessOrder();
Console.ReadKey();
}
}

代码说明

  1. ServiceTypeAttribute: 这是一个自定义属性,用于显式指定服务类的生命周期(Singleton、Scoped 或 Transient)。
  2. RegisterServices 扩展方法:
    • 它扩展了 IServiceCollection,提供了一个便捷的方法来批量注册服务。
    • servicePrefix 参数指定服务类的前缀(例如,"Service")。
    • interfacePrefix 参数指定接口的前缀(例如,"I")。
    • 它使用反射来查找程序集中所有符合命名约定的类(例如,以 "Service" 开头的类)。
    • 它假定接口的命名约定是接口前缀 + 服务类名(去掉服务前缀)。 例如,如果服务类是 UserService,则对应的接口应该是 IUserService
    • 它使用 GetCustomAttribute 方法获取服务类上的 ServiceTypeAttribute 属性,并根据属性中指定的生命周期注册服务。 如果未找到此属性,则默认使用 Scoped 生命周期。
    • 如果找不到与服务类对应的接口,它会抛出一个异常,指示缺少必需的接口。
  3. 示例:
    • IUserServiceUserService 演示了如何定义一个服务及其接口。
    • IOrderServiceOrderService 演示了另一个服务及其接口,并使用[ServiceType] 特性标记了生命周期
    • Main 方法中,我们创建了一个 ServiceCollection,调用 RegisterServices 方法注册服务,然后从 ServiceProvider 中解析并使用这些服务。

使用说明

  1. 定义您的服务接口和服务类,并确保它们遵循命名约定(例如,IUserServiceUserService)。
  2. 使用 ServiceTypeAttribute 属性标记您的服务类,以指定其生命周期。 如果省略此属性,则默认使用 Scoped 生命周期。
  3. 在您的应用程序的启动配置中(例如,Main 方法或 ConfigureServices 方法中),获取 IServiceCollection 的实例,并调用 RegisterServices 扩展方法,传递服务类前缀和接口前缀。
  4. 像往常一样使用依赖注入来获取服务实例。

这个方法提供了一种灵活且类型安全的方式来批量注册服务,并允许您通过属性显式控制每个服务的生命周期。

  

c# 批量注入示例代码的更多相关文章

  1. 在.NET Core中批量注入Grpc服务

    GRPC 是谷歌发布的一个开源.高性能.通用RPC服务,尽管大部分 RPC 框架都使用 TCP 协议,但其实 UDP 也可以,而 gRPC 干脆就用了 HTTP2.还有就是它具有跨平台.跨语言 等特性 ...

  2. .NET 使用自带 DI 批量注入服务(Service)和 后台服务(BackgroundService)

    今天教大家如何在asp .net core 和 .net 控制台程序中 批量注入服务和 BackgroundService 后台服务 在默认的 .net 项目中如果我们注入一个服务或者后台服务,常规的 ...

  3. 去掉tppabs冗余代码,怎样批量去掉tppabs代码

    去掉tppabs冗余代码,怎样批量去掉tppabs代码 刚用teleport pro拉了一个整站到本地 所有的超链都被强行加了一句tppabs="   就玩了一把dw的替换功能 查找范围:整 ...

  4. C/C++ 开源库及示例代码

    C/C++ 开源库及示例代码 Table of Contents 说明 1 综合性的库 2 数据结构 & 算法 2.1 容器 2.1.1 标准容器 2.1.2 Lockfree 的容器 2.1 ...

  5. 看完48秒动画,让你不敢再登录HTTP网站(附完整示例代码)

    在我的 单点登录SSO示例代码 一文中,强烈不建议部署HTTP的SSO服务站点. 在此写个基于网络包嗅探的HTTP会话劫持程序,给大家一个直观的危害性展示. 示例中,我在一台Mac上登录58同城,被另 ...

  6. SmartStore.Net、NopCommerce 全局异常处理、依赖注入、代码研究

    以下是本人最近对NopCommerce和SmartStore.net部分代码的研究和总结,主要集中于:依赖注入.异常处理.对象映射.系统缓存.日志这些方面,供大家参考. NOP 3.8 /// < ...

  7. SQL注入之代码层防御

    [目录] 0x0 前言 0x1 领域驱动的安全 1.1 领域驱动的设计 1.2 领域驱动的安全示例 0x2 使用参数化查询 2.1 参数化查询 2.2 Java中的参数化语句 2.3 .NET(C#) ...

  8. 50.AngularJs directive详解及示例代码

    转自:https://www.cnblogs.com/best/tag/Angular/ 本教程使用AngularJs版本:1.5.3 AngularJs GitHub: https://github ...

  9. mybatis的sql中使用$会出现sql注入示例

    mybatis的sql中使用$会出现sql注入示例: 模拟简单登录场景: 页面代码: function login(){ //sql注入 var user = { username : "' ...

  10. .netcore之DI批量注入(支持泛型) - xms

    一旦系统内模块比较多,按DI标准方法去逐个硬敲AddScoped/AddSingleton/AddTransient缺乏灵活性且效率低下,所以批量注入提供了很大的便捷性,特别是对于泛型的服务类,下面介 ...

随机推荐

  1. C# Graphic 操作记录

    C# 在图片上绘制文字 /// <summary> /// 生成文字图片 /// </summary> /// <param name="text"& ...

  2. 通过 fork 为项目做出贡献

    本文旨在帮助新手小伙伴了解学习如何参与 GitHub 项目,为其献上自己的一份力,留下属于自己的足迹. 普遍流程 通过 fork 为项目做出贡献一个普遍的流程如下图: sequenceDiagram ...

  3. QT5笔记:11. Qt类库的模块

    基础模块:提供了Qt的最基本的功能 附加模块:实现了一些特定功能的提供了附加价值的模块 增值模块:商业版才有的模块,单独发布的提供额外附加价值的模块或工具 技术预览模块:一些处于开发阶段,但是可以作为 ...

  4. flutter - [02] 基本语法

    题记部分 一.注释 ///这是一个注释 //这也是个注释 /* 这还是个注释 */ void main(List<String> args) { print ('你好 dart'); } ...

  5. MySQL - [13] binlog、redolog、undolog、delaylog

    题记部分 bin log redo log undo log 3.1.什么是undo log 事务是需要保证原子性的,也就是说,事务中的操作要么全部完成,要么什么也不做.但有如下情况,会造成事务执行不 ...

  6. Selenium KPI接口 键盘操作

    实现功能:百度搜索框输入selenium->复制内容->sogo搜索框粘贴内容. 首先导入Keys方法 使用格式: driver.findelementbyid("kw" ...

  7. python 如何生成exe程序

    cmd 安装包 pip install pyinstaller cd 进入程序文件夹内 3.输入命令pyinstaller -F xx.py 在dist文件夹下找到安装包即可

  8. Anaconda使用记录

    1 安装 windows下,安装完添加环境变量(哦安装时勾选添加环境变量选项就是加这些变量的) ## (记anaconda软件目录为%ANACONDA3%) %ANACONDA3%\ %ANACOND ...

  9. jquery简单的上传图片预览

    html <div id="bcd"></div> <input type="file" id="abc"&g ...

  10. HTTP内容编码和HTTP压缩的区别

    HTTP压缩,在HTTP协议中,其实是内容编码的一种. 在http协议中,可以对内容(也就是body部分)进行编码, 可以采用gzip这样的编码. 从而达到压缩的目的. 也可以使用其他的编码把内容搅乱 ...