.Net6开发winform程序使用依赖注入学习通http://www.bdgxy.com/

文章来源: 学习通http://www.bdgxy.com/

普学网http://www.boxinghulanban.cn/

智学网http://www.jaxp.net/

表格制作excel教程http://www.tpyjn.cn/

学习通http://www.tsgmyy.cn/


.net? Blazor webassembly 和 webAPI 内建支持依赖注入, Winform 和 Console 应用虽然不带有依赖注入功能, 但增加依赖注入也很简单.?

本文将示例如何为 WinForm 程序增加依赖注入特性, 实现通过DI容器获取Cofiguration 实例, 并读取appsettings.json文件.

安装依赖库, 有点多

  • Microsoft.Extensions.DependencyInjection 库, 依赖注入的类库
  • Microsoft.Extensions.Configuration 库, 包含IConfiguration接口 和 Configuration类
  • Microsoft.Extensions.Configuration.Json 库, 为 IConfiguration 增加了读取 Json 文件功能,
  • Microsoft.Extensions.Hosting 库,? 提供 Host 静态类,? 有能力从 appsettings.{env.EnvironmentName}.json 加载相应 env? 的设定值,? 并将设定值用于IConfiguration/ILoggerFactory中, 同时增加 Console/EventSourceLogger 等 logger. 仅适用于 Asp.Net core 和 Console 类应用
  • Microsoft.Extensions.Logging 库,? 包含 ILogger 和 ILoggerFactory 接口
  • Serilog.Extensions.Logging 库, 为DI 容器提供 AddSerilog() 方法.
  • Serilog.Sinks.File 库, 提供 Serilog rolling logger
  • Serilog.Sinks.Console 库, 增加 serilog console logger
  • Serilog.Settings.Configuration 库, 允许在 appsetting.json? 配置 Serilog, 顶层节点要求是 Serilog.?
  • Serilog.Enrichers.Thread 和 Serilog.Enrichers.Environment 库,? 为输出日志文本增加 Thread和 env 信息

补充库:

  • Microsoft.Extensions.Options.ConfigurationExtensions 库,? 为DI容器增加了从配置文件中实例化对象的能力, 即? serviceCollection.Configure<TOptions>(IConfiguration)
  • Microsoft.Extensions.Options 库,? 提供以强类型的方式读取configuration文件, 这是.Net中首选的读取configuration文件方式.

appsettings.json 配置文件

配置一个 ConnectionString, 另外配 serilog

{

“ConnectionStrings”: {
“oeeDb”: “Server=localhost\SQLEXPRESS01;Database=Oee;Trusted_Connection=True;”
},

“Serilog”: {
“Using”: [ “Serilog.Sinks.Console”, “Serilog.Sinks.File” ],
“MinimumLevel”: “Debug”,
“WriteTo”: [
{ “Name”: “Console” },
{
“Name”: “File”,
“Args”: { “path”: “Logs/serilog.txt” }
}
],
“Enrich”: [ “FromLogContext”, “WithMachineName”, “WithThreadId” ]
}
}

Program.cs , 增加DI容器

using Microsoft.Extensions.Configuration;
using Microsoft.Extensions.DependencyInjection;

using Serilog;

namespace Collector
{
internal static class Program
{
/// <summary>
/// The main entry point for the application.
/// </summary>
[STAThread]
static void Main()
{
ApplicationConfiguration.Initialize();
//未使用依赖注入的写法
//Application.Run(new FormMain());

       //生成 DI 容器
       ServiceCollection services = new ServiceCollection();
       ConfigureServices(services);  //注册各种服务类

        //先用DI容器生成 serviceProvider, 然后通过 serviceProvider 获取Main Form的注册实例
       var serviceProvider =services.BuildServiceProvider();
        
       var formMain = serviceProvider.GetRequiredService&lt;FormMain&gt;();   //主动从容器中获取FormMain实例, 这是简洁写法
       // var formMain = (FormMain)serviceProvider.GetService(typeof(FormMain));  //更繁琐的写法
       Application.Run(formMain); 
    }

   
    /// &lt;summary&gt;
    /// 在DI容器中注册所有的服务类型 
    /// &lt;/summary&gt;
    /// &lt;param name="services"&gt;&lt;/param&gt;
    private static void ConfigureServices(ServiceCollection services)
    {
        //注册 FormMain 类
        services.AddScoped&lt;FormMain&gt;();

        //register configuration
        IConfigurationBuilder cfgBuilder = new ConfigurationBuilder()
            .SetBasePath(Directory.GetCurrentDirectory())
            .AddJsonFile("appsettings.json")
            .AddJsonFile($"appsettings.{Environment.GetEnvironmentVariable("DOTNET_ENVIRONMENT")}.json", optional: true, reloadOnChange: false)
            ;
        IConfiguration configuration=cfgBuilder.Build();
        services.AddSingleton&lt;IConfiguration&gt;(configuration);

        //Create logger instance
        var serilogLogger = new LoggerConfiguration()
            .ReadFrom.Configuration(configuration)
            .Enrich.FromLogContext()
            .CreateLogger();

        //register logger
        services.AddLogging(builder =&gt; {
            object p = builder.AddSerilog(logger: serilogLogger, dispose: true);
        });

    }
} 

}

FormMain.cs , 验证依赖注入的效果

using Microsoft.Extensions.Configuration;
using Microsoft.Extensions.Logging;

namespace Collector
{
public partial class FormMain : Form
{
private readonly IConfiguration _configuration;
private readonly ILogger _logger;

    /// &lt;summary&gt;
    /// 为 FormMain 构造子增加两个形参, 构造子参数将由于DI容器自动注入
    /// &lt;/summary&gt;
    /// &lt;param name="configuration"&gt;&lt;/param&gt;
    /// &lt;param name="logger"&gt;形参必须是 ILogger泛型类型, 不能是 ILogger 类型&lt;/param&gt;
    public FormMain(IConfiguration configuration, ILogger&lt;FormMain&gt; logger)  
    {
        _configuration = configuration;
        _logger = logger;

        InitializeComponent();
        var connectionString = _configuration.GetConnectionString("oeeDb");  //从配置文件中读取oeeDb connectionString 
        _logger.LogInformation(connectionString);   //将connection String 写入到日志文件中
    }

}

}

DI容器管理配置文件Section

上面示例, 我们通过 _configuration.GetConnectionString("oeeDb")? 可以拿到connectionString, 非常方便, 这主要是得益于.Net 已经类库已经考虑到在配置文件中存储 connectionString 是一个普遍的做法, 所以类库内置支持了.

如果在 appsettings.json 中存一些自定义的信息, 如何方便读取呢? 微软推荐的 Options 模式, 下面详细介绍.

首先安装库:

  • Microsoft.Extensions.Options.ConfigurationExtensions 库,? 为DI容器增加了从配置文件中实例化对象的能力, 即? serviceCollection.Configure<TOptions>(IConfiguration)
  • Microsoft.Extensions.Options 库,? 提供以强类型的方式读取configuration文件, 这是.Net中首选的读取configuration文件方式.

假设 appsettings.json 中要存放appKey和appSecret信息, 具体配置如下:

  "AppServiceOptions": {
    "appKey": "appkey1",
    "appSecret": "appSecret1"
  }

定义对应的 Poco Class,? 推荐后缀为 Options,

    public class AppServiceOptions
    {
        public string AppKey { get; set; } = "";
        public string AppSecret { get; set; } = "";
}</pre>

注册函数 ConfigureServices()中,? 注册 AppServiceOptions 类, 告知DI容器, 要基于配置文件AppServiceOptions section来实例化

private static void ConfigureServices(ServiceCollection services)
        {
            //注册 FormMain 类
            services.AddScoped<FormMain>();
        //register configuration
        IConfigurationBuilder cfgBuilder = new ConfigurationBuilder()
            .SetBasePath(Directory.GetCurrentDirectory())
            .AddJsonFile("appsettings.json")
            .AddJsonFile($"appsettings.{Environment.GetEnvironmentVariable("DOTNET_ENVIRONMENT")}.json", optional: true, reloadOnChange: false)
            ;
        IConfiguration configuration=cfgBuilder.Build();
        services.AddSingleton&lt;IConfiguration&gt;(configuration);

        //Create logger instance
        var serilogLogger = new LoggerConfiguration()
            .ReadFrom.Configuration(configuration)
            .Enrich.FromLogContext()
            .CreateLogger();

        //register logger
        services.AddLogging(builder =&gt; {
            object p = builder.AddSerilog(logger: serilogLogger, dispose: true);
        });


        //注册 AppServiceOptions 类, 告知DI容器, 要基于配置文件AppServiceOptions section来实例化
        services.AddOptions();
        services.Configure&lt;AppServiceOptions&gt;(configuration.GetSection("AppServiceOptions"));    
    }</pre>

主动从DI容器中获取 AppServiceOptions 配置信息代码如下, 注意GetRequiredService函数的的泛型参数要使用 IOptions<> 包一下.

   var appServiceOptionsWrapper=serviceProvider.GetRequiredService<IOptions<AppServiceOptions>>();
   AppServiceOptions appServiceOptions= appServiceOptionsWrapper.Value;

将 AppServiceOptions 注入到 FormMain 的代码, 和主动从DI容器中获取 AppServiceOptions 实例一样, 都需要使用 IOptions<> 接口包一下构造子形参.

public partial class FormMain : Form
    { 
        private readonly IConfiguration _configuration; 
        private readonly ILogger _logger;
        private AppServiceOptions _appServiceOptions;
    /// &lt;summary&gt;
    /// 为 FormMain 构造子增加三个形参, 构造子参数将由于DI容器自动注入
    /// &lt;/summary&gt;
    /// &lt;param name="configuration"&gt;形参必须是接口 IConfigurations&lt;/param&gt;
    /// &lt;param name="logger"&gt;形参必须是 ILogger泛型类型, 不能是 ILogger 类型&lt;/param&gt;
    /// &lt;param name="appServiceOptionsWrapper"&gt;形参必须是 IOptions 泛型接口 &lt;/param&gt;
    public FormMain(IConfiguration configuration, ILogger&lt;FormMain&gt; logger, IOptions&lt;AppServiceOptions&gt; appServiceOptionsWrapper)  
    {
        _configuration = configuration;
        _logger = logger;
        _appServiceOptions = appServiceOptionsWrapper.Value;

        InitializeComponent(); 
        var connectionString = _configuration.GetConnectionString("oeeDb");  //从配置文件中读取oeeDb connectionString 
        _logger.LogInformation(connectionString);   //将connection String 写入到日志文件中
    }

    private void button1_Click(object sender, EventArgs e)
    {
        this.Text = _appServiceOptions.AppKey;
    }
}</pre>

.net core 复杂 configuration Section 的读取

appsettings文件定义一个复杂的设置项, 顶层是一个json 数组, 里面又嵌套了另一个数组

"PlcDevices": [
    {
      "PlcDeviceId": "Plc1",
      "IpAddress": "127.0.0.1",
      "Port": 1234,
      "SlaveId": 1,
      "DataPoints": [
        {
          "ModbusAddress": 0,
          "EqpId": "eqp1"
        },
        {
          "ModbusAddress": 0,
          "EqpId": "eqp2"
        }
      ]
    },
{
  "PlcDeviceId": "Plc2",
  "IpAddress": "127.0.0.2",
  "Port": 1234,
  "SlaveId": "2",
  "DataPoints": [
    {
      "ModbusAddress": 0,
      "EqpId": "eqp3"
    },
    {
      "ModbusAddress": 0,
      "EqpId": "eqp4"
    }
  ]
}

]

对应poco对象为:

public class PlcDevice
{
    public string IpAddress { get; set; } = "";
    public int Port { get; set; } = 0;
    public string PlcDeviceId { get; set; } = "";
    public int SlaveId { get; set; } 
    public List<DataPoint> DataPoints { get; set; }

}

public class DataPoint
{ public int ModbusAddress { get; set; }
public string EqpId { get; set; } = “”;
}

读取 json 的C# 代码:

services.AddOptions();
//实例化一个对应 PlcDevices json 数组对象, 使用了 IConfiguration.Get<T>()
var PlcDeviceSettings= configuration.GetSection("PlcDevices").Get<List<PlcDevice>>();
//或直接通过 service.Configure<T>() 将appsettings 指定 section 放入DI 容器, 这里的T 为 List<PlcDevice>
services.Configure<List<PlcDevice>>(configuration.GetSection("PlcDevices"));

到此这篇关于.Net6开发winform程序使用依赖注入的文章就介绍到这了。希望对大家的学习有所帮助,也希望大家多多支持菜鸟教程https://www.piaodoo.com/。

要在 WinForm 程序使用 WebRequest 发送 HTTP GET 请求,你可以按照以下步骤操作: 1. 引用 System.Net 命名空间。 ```csharp using System.Net; ``` 2. 创建一个 WebRequest 对象,并设置请求的 URL。 ```csharp WebRequest request = WebRequest.Create("http://61.155.88.154:7032"); ``` 3. 发送 HTTP GET 请求,并获取响应。 ```csharp WebResponse response = request.GetResponse(); ``` 4. 读取响应内容,并关闭响应。 ```csharp Stream stream = response.GetResponseStream(); StreamReader reader = new StreamReader(stream); string responseBody = reader.ReadToEnd(); response.Close(); ``` 下面是一个完整的示例代码: ```csharp using System; using System.Net; using System.IO; namespace WinFormWebRequestDemo { public partial class Form1 : Form { public Form1() { InitializeComponent(); } private void button1_Click(object sender, EventArgs e) { try { WebRequest request = WebRequest.Create("http://61.155.88.154:7032"); WebResponse response = request.GetResponse(); Stream stream = response.GetResponseStream(); StreamReader reader = new StreamReader(stream); string responseBody = reader.ReadToEnd(); response.Close(); Console.WriteLine(responseBody); } catch (Exception ex) { Console.WriteLine(ex.Message); } } } } ``` 这个示例使用 WebRequest 类发送一个 HTTP GET 请求,并将响应内容输出到控制台。注意,这个示例是在 UI 线程中执行的,如果你要在后台线程中发送请求,请使用异步方法。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值