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

.NET 依赖注入和配置系统

文章目录

    • 依赖注入
      • DI几个概念
      • .NET 中使用DI
      • 生命周期
      • IServiceProvider的服务定位器方法
    • 配置系统
      • Json文件配置
      • 绑定类读取配置

依赖注入

依赖注入(Dependency Injection,DI)是控制反转(Inversion of Control,IOC)思想的实现方式。

控制反转的两种实现方式:

  1. 服务定位器(ServiceLocator);
  2. 依赖注入(Dependency Injection,DI);

DI几个概念

服务(service):对象;
注册服务;
服务容器:负责管理注册的服务;
查询服务:创建对象及关联对象;
对象生命周期:Transient(瞬态); Scoped(范围); Singleton(单例);

.NET 中使用DI

1、Install-Package Microsoft.Extensions.DependencyInjection
2、using Microsoft.Extensions.DependencyInjection
3、ServiceCollection用来构造容器对象IServiceProvider 。调用ServiceCollectionBuildServiceProvider()创建的ServiceProvider,可以用来获取BuildServiceProvider()之前ServiceCollection中的对象。

  internal interface ITestService{public string Name { get; set; }public void SayHi();}internal class TestServiceImpl{public string Name { get; set; }public void SayHi(){Console.WriteLine($"Hi, I'm {Name}");}}internal class TestServiceImpl2
{public string Name { get; set; }public void SayHi(){Console.WriteLine($"你好,我是{Name}");}
}static void Main(string[] args){ServiceCollection services = new ServiceCollection();services.AddTransient<TestServiceImpl>();using (ServiceProvider sp = services.BuildServiceProvider()){TestServiceImpl testService = sp.GetRequiredService<TestServiceImpl>();testService.Name = "tom";testService.SayHi();}}

生命周期

Transient:

static void Main(string[] args)
{ServiceCollection services = new ServiceCollection();services.AddTransient<TestServiceImpl>();using (ServiceProvider sp = services.BuildServiceProvider()){var ts1 = sp.GetService<TestServiceImpl>();var ts2 = sp.GetService<TestServiceImpl>();Console.WriteLine(object.ReferenceEquals(ts1, ts2));}
}

输出:False

Singleton:

static void Main(string[] args)
{ServiceCollection services = new ServiceCollection();services.AddSingleton<TestServiceImpl>();using (ServiceProvider sp = services.BuildServiceProvider()){var ts1 = sp.GetService<TestServiceImpl>();var ts2 = sp.GetService<TestServiceImpl>();Console.WriteLine(object.ReferenceEquals(ts1, ts2));}
}

输出:True

Scoped:

 static void Main(string[] args){ServiceCollection services = new ServiceCollection();services.AddScoped<TestServiceImpl>();using (ServiceProvider sp = services.BuildServiceProvider()){using (IServiceScope scope = sp.CreateScope()){var ts1 = scope.ServiceProvider.GetService<TestServiceImpl>();var ts2 = scope.ServiceProvider.GetService<TestServiceImpl>();Console.WriteLine(object.ReferenceEquals(ts1, ts2));}}}

输出:True

GetServiceGetRequiredService有生命区别?
GetService如果找不到注入的类会返回null,
GetRequiredService如果找不到注入的类会抛出异常。

Tips: 生命周期的选择:如果类无状态,建议为Singleton;如果类有状态,且有Scope控制,建议为Scoped,因为通常这种Scope控制下的代码都是运行在同一个线程中的,没有并发修改的问题;在使用Transient的时候要谨慎。

IServiceProvider的服务定位器方法

  • T GetService<T>() 如果获取不到对象,则返回null。
  • object GetService(Type serviceType)
  • T GetRequiredService<T>()如果获取不到对象,则抛异常
  • object GetRequiredService(Type serviceType)
  • IEnumerable<T> GetServices<T>()适用于可能有很多满足条件的服务
  • IEnumerable<object> GetServices(Type serviceType)

总结:
1、依赖注入是有“传染性”的,如果一个类的对象是通过DI创建的,那么这个类的构造函数中声明的 所有服务类型的参数都会被DI赋值;但是如果一个对象是程序员手动创建的,那么 这个对象就和DI没有关系,它的构造函数中声明的服务类型参数就不会被自动赋值。
2、.NET的DI默认是 构造函数注入
3、第三方DI容器:Autofac等。Autofac优点:支持属性注入、基于名字注入、基于约定的注入等。

配置系统

Json文件配置

1、创建一个json文件,文件名随意,比如config.json,设置“ 如果较新则复制”。
2、NuGet安装Microsoft.Extensions.ConfigurationMicrosoft.Extensions.Configuration.Json
3、
config.json

{"exclude": ["**/bin","**/bower_components","**/jspm_packages","**/node_modules","**/obj","**/platforms"],"name": "yh","age": 18,"proxy": {"address": "bbaa","port": 80}
}
    static void Main(string[] args){ConfigurationBuilder configBuilder = new ConfigurationBuilder();configBuilder.AddJsonFile(
"config.json", optional: false, reloadOnChange: false);IConfigurationRoot config = configBuilder.Build();string name = config["name"];string proxyAddress = config.GetSection("proxy:address").Value;Console.WriteLine(name);Console.WriteLine(proxyAddress);}

optional参数:表示这个文件是否可选。初学时,建议optional设置为false,这样写错了的话能够及时发现。
reloadOnChange参数:表示如果文件修改了,是否重新加载配置。

绑定类读取配置

绑定一个类,自动完成配置的读取。

1、NuGet安装:Microsoft.Extensions.Configuration.Binder
2、

static void Main(string[] args)
{ConfigurationBuilder configBuilder = new ConfigurationBuilder();configBuilder.AddJsonFile(
"config.json", optional: false, reloadOnChange: false);IConfigurationRoot configRoot = configBuilder.Build();Proxy proxy = configRoot.GetSection("proxy").Get<Proxy>();Console.WriteLine(proxy.Address);Console.WriteLine(proxy.Port);Config config = configRoot.Get<Config>();Console.WriteLine(config.Name);Console.WriteLine(config.Proxy.Address);
}class Config{public string Name { get; set; }public int Age { get; set; }public Proxy Proxy { get; set; }}class Proxy{public string Address { get; set; }public int Port { get; set; }}
http://www.lryc.cn/news/325037.html

相关文章:

  • 什么是”法兰“?
  • Vulnhub靶机:HackLAB_Vulnix
  • 软件推荐 篇三十七:开源免费无广告的在线音乐免费播放 | MusicFree纯净无广告体验-小众冷门推荐
  • Hive SQL必刷练习题:留存率问题(*****)
  • 在Linux/Ubuntu/Debian中创建自己的命令快捷方式
  • vue学习笔记——Vue3循环生成表单时,对每一行新生成的数据添加表单验证的方法
  • 用C++做一个植物大战僵尸
  • 政安晨:【深度学习实践】【使用 TensorFlow 和 Keras 为结构化数据构建和训练神经网络】(三)—— 随机梯度下降
  • 普通用户无法连接到docker服务
  • Rancher(v2.6.3)——Rancher部署Nginx(单机版)
  • java问题解释
  • TSN协议原理!看完这一篇就够了(1)——时钟同步IEEE802.1AS-2020
  • Intel被喷惨的大小核CPU终于有救,12、13代也沾了光
  • JAVA入门第一步
  • 【scala】使用gradle和scala构建springboot程序
  • Linux sfdisk命令教程:硬盘分区表编辑器(附实例详解和注意事项)
  • Godot 学习笔记(5):彻底的项目工程化,解决GodotProjectDir is null+工程化范例
  • 算法打卡day23|回溯法篇03|Leetcode 39. 组合总和、40.组合总和II、131.分割回文串
  • Google研究者们提出了VLOGGER模型
  • Python从入门到精通秘籍十九
  • 解决:您还有0天的时间继续使用internet download manager
  • 操作系统目录
  • 常用的Node.js命令集锦
  • 2021年XX省赛职业院校技能大赛”高职组 计算机网络应用赛项 网络构建模块竞赛真题
  • 80386 ATT汇编语法
  • 如何在Linux系统使用宝塔面板搭建Inis博客并发布至公网【内网穿透】
  • 【漏洞复现】netgear路由器 boarddataww 存在RCE漏洞
  • 什么是原型链
  • 什么是虚拟线程?
  • node.js是什么怎么用常用方法