大型网站建设哪家好,网站网页开发公司,WordPress会话有效时间,上门做美容的有什么网站在WPF开发中#xff0c;依赖注入#xff08;Dependency Injection#xff09;和控制反转#xff08;Inversion of Control#xff09;是程序解耦的关键#xff0c;在当今软件工程中占有举足轻重的地位#xff0c;两者之间有着密不可分的联系。今天就以一个简单的小例子依赖注入Dependency Injection和控制反转Inversion of Control是程序解耦的关键在当今软件工程中占有举足轻重的地位两者之间有着密不可分的联系。今天就以一个简单的小例子简述如何在WPF中实现依赖注入和控制反转仅供学习分享使用如有不足之处还请指正。 什么是依赖注入和控制反转 依赖注入又称为依赖项注入那什么是依赖项呢比如在一个类A中实现某中功能而此功能是另外一个类B实现的那就说明A依赖BB就是A的依赖项。或者是另一个对象A所依赖的对象B。示例如下
namespace DemoIoc
{public class MessageWriter{public void Print(string message){Console.WriteLine($MessageWriter.Write(message: \{message}\));}}public class Worker : BackgroundService{private readonly MessageWriter writer new();protected override async Task ExecuteAsync(CancellationToken stoppingToken){while (!stoppingToken.IsCancellationRequested){writer.Print($Worker running at: {DateTimeOffset.Now});await Task.Delay(1_000, stoppingToken);}}}
} 注意在上述示例中Worker类依赖于MessageWriter类所以MessageWriter就是Worker的依赖项。 硬编码的依赖项如前面的示例会产生问题应避免使用。
强依赖关系具有以下几个问题
如果要用不同的实现替换 MessageWriter必须修改 Worker 类。如果 MessageWriter 具有依赖项则必须由 Worker 类对其进行配置且很难进行初始化。这种实现很难进行单元测试。
那如何解决上述依赖关系所造成的弊端呢答案就是依赖项注入。可通过如下几个步骤实现
使用接口或基类将依赖关系实现抽象化。在服务容器中注册依赖关系。将服务注入到使用它的类的构造函数中。 .NET 提供了一个内置的服务容器 IServiceProvider。 服务通常在应用启动时注册并追加到 IServiceCollection。 添加所有服务后可以使用 BuildServiceProvider 创建服务容器。 框架负责创建依赖关系的实例并在不再需要时将其释放。
简单一句话说依赖注入DI将所依赖的对象参数化接口化并且将依赖对象的创建和释放剥离出来这样就做到了解耦并且实现了控制反转IoC。
控制反转IoC具有如下两个特点
高等级的代码不能依赖低等级的代码抽象接口不能依赖具体实现
控制反转解决代码的强耦合增加了代码的可扩展性。依赖注入将依赖具体实现类和控制实现类的创建和释放变成了依赖接口或抽象类不再控制接口的创建和释放。两者之间相辅相成互相成就。 WPF中实现依赖注入的步骤 1. 安装DI库 首先创建一个WPF应用程序然后在Nuget包管理器中安装微软提供的依赖注入库【Microsoft.Extensions.DependencyInjection】如下所示 2. 创建接口和实现类 创建测试用的接口ITextService和实现类TextService如下所示
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;namespace DemoIoc
{public interface ITextService{public string GetText();}public class TextService : ITextService{public string GetText(){return DateTime.Now.ToString(yyyy-MM-dd HH:mm:ss.fff);}}
} 3. 接口注入 在需要调用的地方如MainWindow进行ITextService接口注入如下所示
namespace DemoIoc
{/// summary/// Interaction logic for MainWindow.xaml/// /summarypublic partial class MainWindow : Window{private ITextService textService;public MainWindow(ITextService textService){this.textService textService;InitializeComponent();}private void Window_Loaded(object sender, RoutedEventArgs e){this.txtCurrentTime.Text textService.GetText();}}
} 注意以上可以看出MainWindow依赖ITextService接口而不依赖于接口的实现。这样就实现了依赖注入。 4. 配置容器 在启动程序App.xaml.cs中添加当前对象成员和服务提供对象并在实例化服务对象的时候一次性注册以便在后续需要的时候进行获取。如下所示
namespace DemoIoc
{/// summary/// Interaction logic for App.xaml/// /summarypublic partial class App : Application{/// summary/// 获取当前 App 实例/// /summarypublic new static App Current (App)Application.Current;/// summary/// 获取存放应用服务的容器/// /summarypublic IServiceProvider ServiceProvider { get; }public App(){ServiceProvider ConfigureServices();}/// summary/// 配置应用的服务/// /summaryprivate static IServiceProvider ConfigureServices(){var serviceCollection new ServiceCollection().AddSingletonITextService,TextService().AddSingletonMainWindow();return serviceCollection.BuildServiceProvider();}protected override void OnStartup(StartupEventArgs e){var mainWindow ServiceProvider.GetRequiredServiceMainWindow();mainWindow.Show();}}
} 注意在此示例中MainWindow通过服务注册的方式进行实例化所以需要删除默认的App.xaml中StartUri属性设置否则将提示默认构造函数不存在。 示例测试 经过上述步骤就实现了WPF中依赖注入和控制反转测试结果如下 说明正常输出则表示依赖注入成功。 参考文档 1. .Net依赖项注入https://learn.microsoft.com/zh-cn/dotnet/core/extensions/dependency-injection 以上就是依赖注入和控制反转的全部内容希望可以抛砖引玉一起学习共同进步。