南通优普网站建设团队,标智客logo在线设计生成器,做网站要签合同吗,wordpress 获取文章分类id#x1f680; 优质资源分享 #x1f680;
学习路线指引#xff08;点击解锁#xff09;知识定位人群定位#x1f9e1; Python实战微信订餐小程序 #x1f9e1;进阶级本课程是python flask微信小程序的完美结合#xff0c;从项目搭建到腾讯云部署上线#xff0c;打造一… 优质资源分享
学习路线指引点击解锁知识定位人群定位 Python实战微信订餐小程序 进阶级本课程是python flask微信小程序的完美结合从项目搭建到腾讯云部署上线打造一个全栈订餐系统。Python量化交易实战入门级手把手带你打造一个易扩展、更安全、效率更高的量化交易系统
在前面随笔我们介绍过这个基于SqlSugar的开发框架我们区分Interface、Modal、Service三个目录来放置不同的内容其中Modal是SqlSugar的映射实体Interface是定义访问接口Service是提供具体的数据操作实现。在Service层中往往除了本身的一些增删改查等处理操作外也需要涉及到相关业务的服务接口这些服务接口我们通过利用.net 的接口注入方式实现IOC控制反转的处理的。
1、框架Service层的模块
如下面的VS中的项目服务层包含很多业务表的服务接口实现如下所示。 我们以其中简单的Customer业务表为例它的服务类代码如下所示主要关注服务类的定义即可。 /// /// 客户信息应用层服务接口实现/// public class CustomerService : MyCrudServicestring, CustomerPagedDto, ICustomerService{...............}它除了在泛型约束中增加SqlSugar实体类主键类型分页条件对象外还继承接口 ICustomerService 这个接口就是我们实现IOC的第一步服务层继承指定的接口实现对我们实现IOC控制反转提供便利。 /// /// 客户信息服务接口/// public interface ICustomerService : IMyCrudServicestring, CustomerPagedDto, ITransientDependency{}这个客户信息业务处理是比较典型的单表处理案例它没有涉及到相关服务接口的整合如果我们在其中服务接口中需要调用其他服务接口那么我们就需要通过构造函数注入接口对象的方式获得对象的实例如下我们说介绍的就是服务调用其他相关接口的实现。 2、服务层的接口注入
如对于角色服务接口来说它往往和用户、机构有关系因此我们在角色的服务接口层可以整合用户、机构的对应服务接口如下代码所示。 /// /// 角色信息 应用层服务接口实现/// public class RoleService : MyCrudServiceint, RolePagedDto, IRoleService{private IOuService \_ouService;private IUserService \_userService;/// /// 默认构造函数/// /// 机构服务接口/// 用户服务接口public RoleService(**IOuService ouService, IUserService userService**){this.\_ouService ouService;this.\_userService userService;}}通过构造函数的注入我们就可以获得对应接口实现的实例进行调用它的服务层方法使用了。
这样我们在角色的服务接口实现中就可以调用其他如用户、机构相关的服务接口了。 其他模块的处理方式也是类似如字典项目中使用字典类型的服务接口。 /// /// 应用层服务接口实现/// public class DictDataService : MyCrudServicestring, DictDataPagedDto , IDictDataService{/// /// 测试字典类型接口/// protected IDictTypeService \_dictTypeService;/// /// 注入方式获取接口/// /// 字典类型处理public DictDataService(**IDictTypeService dictTypeService**){this.\_dictTypeService dictTypeService;}
}这里值得注意的是由于接口层是同级对象因此要避免接口的相互引用而导致出错依赖关系要清晰才不会发生这个情况。 3、服务接口的实例的容器注册
在服务层中我们是通过参数化构造函数的方式引入对应的接口的这个操作方式是构造函数的注入处理。
不过在此之前我们需要在.net 的内置IOC容器中注册对应的接口实例否则参数化构造函数会因为找不到接口实例而出错。
.net 的内置Ioc容器及注册处理我们需要在nuget引入下面两个引用。
1、Microsoft.Extensions.DependencyInjection
2、Microsoft.Extensions.DependencyInjection.Abstractions.net 中 负责依赖注入和控制反转的核心组件有两个IServiceCollection和IServiceProvider。其中IServiceCollection负责注册IServiceProvider负责提供实例。
在注册接口和类时IServiceCollection提供了三种注册方法如下所示
1、services.AddTransientIDictDataService, DictDataService(); // 瞬时生命周期
2、services.AddScopedIDictDataService, DictDataService(); // 域生命周期
3、services.AddSingletonIDictDataService, DictDataService(); // 全局单例生命周期如果使用AddTransient方法注册IServiceProvider每次都会通过GetService方法创建一个新的实例
如果使用AddScoped方法注册 在同一个域Scope内IServiceProvider每次都会通过GetService方法调用同一个实例可以理解为在局部实现了单例模式
如果使用AddSingleton方法注册 在整个应用程序生命周期内IServiceProvider只会创建一个实例。
我们为了在注册的时候方便通过遍历方式处理接口实例的注册因此我们根据这几种关系定义了几个基类接口便于根据特定的接口方式来构建接口实例。
namespace WHC.Framework.ControlUtil
{//用于定义这三种生命周期的标识接口/// /// 三种标识接口的基类接口/// public interface IDependency{}/// /// 瞬时每次都重新实例/// public interface ITransientDependency : IDependency{}/// /// 单例全局唯一/// public interface ISingletonDependency : IDependency{ }/// /// 一个请求内唯一线程内唯一/// public interface IScopedDependency : IDependency{}
}这样我们在定义注册类型的时候通过它的接口指定属于上面那种类型。如对于字典项目的服务层我们约定采用瞬时的注册方式那么它的接口定义如下所示。 /// /// 字典项目服务接口/// public interface IDictDataService : IMyCrudServicestring, DictDataPagedDto, ITransientDependency{}配置自动注册接口的时候我们添加如下函数处理即可。 /// /// 配置依赖注入对象/// /// public static void ConfigureRepository(IServiceCollection services){#region 自动注入对应的服务接口//services.AddSingleton();//services.AddScoped();var baseType typeof(**IDependency**);var path AppDomain.CurrentDomain.RelativeSearchPath ?? AppDomain.CurrentDomain.BaseDirectory;var getFiles Directory.GetFiles(path, *.dll).Where(Match); //.Where(oo.Match())var referencedAssemblies getFiles.Select(Assembly.LoadFrom).ToList(); //.Select(o Assembly.LoadFrom(o)) var ss referencedAssemblies.SelectMany(o o.GetTypes());var types referencedAssemblies.SelectMany(a a.DefinedTypes).Select(type type.AsType()).Where(x x ! baseType baseType.IsAssignableFrom(x)).ToList();var implementTypes types.Where(x x.IsClass).ToList();var interfaceTypes types.Where(x x.IsInterface).ToList();foreach (var implementType in implementTypes){if (typeof(**IScopedDependency**).IsAssignableFrom(implementType)){var interfaceType interfaceTypes.FirstOrDefault(x x.IsAssignableFrom(implementType));if (interfaceType ! null)services.AddScoped(interfaceType, implementType);}else if (typeof(**ISingletonDependency**).IsAssignableFrom(implementType)){var interfaceType interfaceTypes.FirstOrDefault(x x.IsAssignableFrom(implementType));if (interfaceType ! null)services.AddSingleton(interfaceType, implementType);}else{var interfaceType interfaceTypes.FirstOrDefault(x x.IsAssignableFrom(implementType));if (interfaceType ! null)services.AddTransient(interfaceType, implementType);}}#endregion}上面根据我们自定义接口的不同适当的采用不同的注册方式来加入Ioc容器中从而实现了接口的注册在服务层中就可以通过构造函数注入的方式获得对应的接口实例了。
这样不管是在WInform的启动模块中还是在Web API的启动模块中我们在IOC容器中加入对应的接口即可如下所示。
///
/// 应用程序的主入口点。
///
[STAThread]
static void Main()
{// IServiceCollection负责注册IServiceCollection services new ServiceCollection();//services.AddSingleton();//services.AddSingleton();//添加IApiUserSession实现类services.AddSingleton();//调用自定义的服务注册ServiceInjection.ConfigureRepository(services);// IServiceProvider负责提供实例IServiceProvider provider services.BuildServiceProvider();services.AddSingleton(provider);//注册到服务集合中,需要可以在Service中构造函数中注入使用Web API中的代码如下所示
//添加HTTP上下文访问
builder.Services.AddHttpContextAccessor();//配置依赖注入访问数据库
ServiceInjection.ConfigureRepository(builder.Services);//添加IApiUserSession实现类
builder.Services.AddSingleton();var app builder.Build();都是类似的处理方式。
同样在Web API项目中的控制器处理中也是一样通过构造函数注入的方式使用接口的如下所示。
namespace WebApi.Controllers
{/// /// 客户信息的控制器对象/// public class CustomerController : BusinessControllerstring, CustomerPagedDto{private ICustomerService \_customerService;/// /// 构造函数并注入基础接口对象/// /// public CustomerController(**ICustomerService customerService**) :base(customerService){this.\_customerService customerService;}}
}或者登录处理的控制器定义如下。 /// /// 登录获取令牌授权的处理/// [Route(api/[controller])][ApiController]public class LoginController : ControllerBase{private readonly IHttpContextAccessor \_contextAccessor;private readonly IConfiguration \_configuration;private readonly IUserService \_userService;/// /// 令牌失效天数默认令牌7天有效期/// protected const int expiredDays 7;/// /// 构造函数注入所需接口/// /// 配置对象/// HTTP上下文对象/// 用户信息public LoginController(**IConfiguration configuration, IHttpContextAccessor httpContext, IUserService userService**){this.\_configuration configuration;this.\_contextAccessor httpContext;this.\_userService userService;}系列文章
《基于SqlSugar的开发框架的循序渐进介绍1–框架基础类的设计和使用》
《基于SqlSugar的开发框架循序渐进介绍2-- 基于中间表的查询处理》
《基于SqlSugar的开发框架循序渐进介绍3-- 实现代码生成工具Database2Sharp的整合开发》
《基于SqlSugar的开发框架循序渐进介绍4-- 在数据访问基类中对GUID主键进行自动赋值处理》
《基于SqlSugar的开发框架循序渐进介绍5-- 在服务层使用接口注入方式实现IOC控制反转》