第3章 .NETCore核心基础组件:3.1 .NET Core依赖注入
3.1.1 什么是控制反转、依赖注入
杨老师在书中进行了一系列的文字阐述,总结一下就是:软件设计模式中有一种叫做【控制反转】的设计模式,而依赖注入是实现这种设计模式的一个很重要的方式。也就是说学习依赖注入,是学习怎样实现控制反转这一设计模式。
1. 控制反转
控制反转是一种编程思想,它将对象的创建、依赖关系的管理等控制权从代码内部转移到外部容器。在传统编程中,对象的创建和依赖关系的处理是在代码中硬编码实现的,这会导致代码之间的耦合度较高。而控制反转通过将这些控制权交给外部容器,使得代码更加灵活和可扩展。
实现方式:
控制反转主要有两种实现方式:依赖注入(Dependency Injection,DI)和依赖查找(Dependency Lookup)。
2. 依赖注入
依赖注入是控制反转最常用的实现方式,它是指将对象的依赖关系通过构造函数、属性或者方法参数的方式注入到对象中。常见的依赖注入有三种:
2.1 构造函数注入
// 定义一个接口
public interface IService
{void DoSomething();
}// 实现接口
public class Service : IService
{public void DoSomething(){Console.WriteLine("Doing something...");}
}// 依赖于 IService 的类
public class Client
{private readonly IService _service;// 通过构造函数注入依赖public Client(IService service){_service = service;}public void Execute(){_service.DoSomething();}
}// 使用示例
class Program
{static void Main(){IService service = new Service();Client client = new Client(service);client.Execute();}
}
Client
类通过构造函数接收一个 IService
类型的对象,这样 Client
类就不负责创建 IService
对象,而是由外部负责创建并注入,实现了依赖的反转。
2.2 属性注入
// 定义一个接口
public interface IService
{void DoSomething();
}// 实现接口
public class Service : IService
{public void DoSomething(){Console.WriteLine("Doing something...");}
}// 依赖于 IService 的类
public class Client
{public IService Service { get; set; }public void Execute(){if (Service != null){Service.DoSomething();}}
}// 使用示例
class Program
{static void Main(){IService service = new Service();Client client = new Client();client.Service = service;client.Execute();}
}
在属性注入中,Client
类通过公共属性 Service
接收依赖对象。
2.3 方法注入
// 定义一个接口
public interface IService
{void DoSomething();
}// 实现接口
public class Service : IService
{public void DoSomething(){Console.WriteLine("Doing something...");}
}// 依赖于 IService 的类
public class Client
{public void Execute(IService service){service.DoSomething();}
}// 使用示例
class Program
{static void Main(){IService service = new Service();Client client = new Client();client.Execute(service);}
}
方法注入是指在调用方法时将依赖对象作为参数传递给方法。
3.1.2 .NET Core 依赖注入的基本使用
原著讲:依赖注入框架中注册的服务有一个重要的概念叫做“生命周期”,通俗的说就是“获取服务的时候是创建一个新对象还是用之前的对象”。依赖注入框架中服务的生命周期有三种:
1. 瞬态
每次请求创建新对象。避免多段代码共用一个对象造成混乱,缺点耗资源。
2. 范威
在给定范围内,多次请求共享同一个服务对象,服务每次请求的时候都会返回同一个对象。
3. 单例
全局共享一个服务对象。为了避免并发修改问题,单例的服务对象最好是无状态对象。
杨老师建议选择:
如果一个类没有状态,建议把服务的生命周期设置为单例。
如果有状态且框架环境中有范围控制,建议设置为范围,通常在范围控制下代码都在同一线程,所以没有并发修改问题。
在使用瞬态时,尽量在子范围中使用,以免控制不当造成内存泄漏。
3.1.3 依赖注入的魅力所在
书上举得例子代码不太完善,但杨老师在视频教程中演示了一个例子,比较简介明了,我实操了一下:
当然,在使用依赖注入之前还需要安装Microsoft.Extensions.DependencyInjection
using Microsoft.Extensions.DependencyInjection;
using System.Runtime.CompilerServices;class Program
{static void Main(){// 创建一个服务容器实例,用于注册服务ServiceCollection services1 = new ServiceCollection();// 注册Controller类为作用域服务services1.AddScoped<Controller>();// 注册ILog接口的实现类LogImpl为作用域服务services1.AddScoped<ILog, LogImpl>();// 注册IStorage接口的实现类StorgeImpl为作用域服务services1.AddScoped<IStorage, StorgeImpl>();// 注册IConfig接口的实现类ConfigImpl为作用域服务services1.AddScoped<IConfig, ConfigImpl>();// 构建服务提供者,用于解析服务using (var sp = services1.BuildServiceProvider()){// 从服务提供者中获取Controller实例var c = sp.GetRequiredService<Controller>();// 调用Controller的Test方法c.Test();}}
}// Controller类用于处理业务逻辑
class Controller
{private readonly ILog log;private readonly IStorage storage;// 构造函数,通过依赖注入接收ILog和IStorage接口的实现类public Controller(ILog ig, IStorage IS){this.log = ig;this.storage = IS;}// Test方法用于测试日志记录和文件存储功能public void Test(){this.log.Log("开始上传");this.storage.Save("bufweqrijfhyurjfnkjbhs", "1998.txt");this.log.Log("上传完毕!!!");}
}// ILog接口定义日志记录方法
interface ILog
{public void Log(string msg);
}// LogImpl类实现ILog接口,用于向控制台输出日志信息
class LogImpl : ILog
{public void Log(string msg){Console.WriteLine($"日志:{msg}");}
}// IConfig接口定义获取配置值方法
interface IConfig
{public string GetValue(string name);
}// ConfigImpl类实现IConfig接口,用于返回固定的配置值
class ConfigImpl : IConfig
{public string GetValue(string name){return "OKOKOKK";}
}// IStorage接口定义文件保存方法
interface IStorage
{public void Save(string content, string name);
}// StorgeImpl类实现IStorage接口,用于根据配置信息保存文件内容到指定服务器
class StorgeImpl : IStorage
{private readonly IConfig config;// 构造函数,通过依赖注入接收IConfig接口的实现类public StorgeImpl(IConfig CG){this.config = CG;}// Save方法用于根据配置信息保存文件内容到指定服务器public void Save(string content, string name){string service = config.GetValue("server");Console.WriteLine($"服务器:{service}, {content}:{name}");}
}
杨老师在这个例子里讲到依赖注入的核心奥义就是:降低各个模块之间的耦合度。再补充描述一下:这种依赖注入的模式极大的提高了项目的可维护性和各个模块的可插拔性。
比如例子当中获取数据存储的配置参数,当配置参数的实现或者类型变化时,只需要改变【IConfig】 的实现即可,业务层的【StorgeImpl】完全不用管【IConfig config】是怎么实现或者怎么变化的,只要是【IConfig】对象即可。
服务的实现和服务的调用高度解耦,真正意义上实现了且超级简易的实现了“我创建对象”到“我要对象”的改变。
相关文章:
第3章 .NETCore核心基础组件:3.1 .NET Core依赖注入
3.1.1 什么是控制反转、依赖注入 杨老师在书中进行了一系列的文字阐述,总结一下就是:软件设计模式中有一种叫做【控制反转】的设计模式,而依赖注入是实现这种设计模式的一个很重要的方式。也就是说学习依赖注入,是学习怎样实现控…...
排序与算法:插入排序
执行效果 插入排序的执行效果是这样的: 呃……看不懂吗?没关系,接着往下看介绍 算法介绍 插入排序(Insertion Sort)是一种简单直观的排序算法。它的工作原理是通过构建有序序列,对于未排序数据,…...
uniapp 打包安卓 集成高德地图
接入高德地图 let vm this;uni.chooseLocation({success: function (res) {// console.log(位置名称: res.name);// console.log(详细地址: res.address);// console.log(纬度: res.latitude);// console.log(经度: res.long…...
python爬虫系列课程2:如何下载Xpath Helper
python爬虫系列课程2:如何下载Xpath Helper 一、访问极简插件官网二、点击搜索按钮三、输入xpath并点击搜索四、点击推荐下载五、将下载下来的文件解压缩六、打开扩展程序界面七、将xpath.crx文件拖入扩展程序界面一、访问极简插件官网 极简插件官网地址:https://chrome.zzz…...
win10系统上的虚拟机安装麒麟V10系统提示找不到操作系统
目录预览 一、问题描述二、原因分析三、解决方案四、参考链接 一、问题描述 win10系统上的虚拟机安装麒麟V10系统提示找不到操作系统,报错:Operating System not found 二、原因分析 国产系统,需要注意的点: 需要看你的系统类…...
基于微信小程序的宿舍报修管理系统设计与实现,SpringBoot(15500字)+Vue+毕业论文+指导搭建视频
运行环境 jdkmysqlIntelliJ IDEAmaven3微信开发者工具 项目技术SpringBoothtmlcssjsjqueryvue2uni-app 宿舍报修小程序是一个集中管理宿舍维修请求的在线平台,为学生、维修人员和管理员提供了一个便捷、高效的交互界面。以下是关于这些功能的简单介绍: …...
分布式同步锁:原理、实现与应用
分布式同步锁:原理、实现与应用 引言1. 分布式同步锁的基本概念1.1 什么是分布式同步锁?1.2 分布式锁的特性 2. 分布式锁的实现方式2.1 基于数据库的分布式锁实现原理优缺点示例 2.2 基于 Redis 的分布式锁实现原理优缺点示例Redlock 算法 2.3 基于 ZooK…...
Chrome多开终极形态解锁!「窗口管理工具+IP隔离插件
Web3项目多开,继ads指纹浏览器钱包被盗后,更多人采用原生chrome浏览器,当然对于新手,指纹浏览器每月成本也是一笔不小开支,今天逛Github发现了这样一个解决方案,作者开发了窗口管理工具IP隔离插件ÿ…...
FreeSwitch的应用类模块
FreeSWITCH 应用类模块(Applications)完整表格 模块名称功能描述mod_callcenter提供呼叫中心功能,支持队列、座席管理、监控等。mod_conference提供多方会议功能,支持音频、视频会议。mod_blacklist提供黑名单功能,阻…...
【蓝桥杯集训·每日一题2025】 AcWing 6123. 哞叫时间 python
6123. 哞叫时间 Week 1 2月18日 农夫约翰正在试图向埃尔茜描述他最喜欢的 USACO 竞赛,但她很难理解为什么他这么喜欢它。 他说「竞赛中我最喜欢的部分是贝茜说 『现在是哞哞时间』并在整个竞赛中一直哞哞叫」。 埃尔茜仍然不理解,所以农夫约翰将竞赛以…...
Unity 淡入淡出
淡入(Fade in):类似打开幕布 淡出(Fade out):类似关上幕布 方案一 使用Dotween(推荐) using DG.Tweening; using UnityEngine; using UnityEngine.UI;public class Test : MonoB…...
PBR光照模型相关知识
PBR是基于物理的光照模型,与lambert光照模型以及Blinn-Phong光照模型有所不同 一、三种光照模型的区别 原理基础 Lambert 光照模型:基于朗伯余弦定律,该定律表明,漫反射光的强度与入射光的方向和物体表面法线的夹角的余弦值成正比…...
【Go | 从0实现简单分布式缓存】-2:HTTP服务端与一致性哈希
本文目录 一、回顾1.1 复习接口 二、http标准库三、实现HTTP服务端四、一致性哈希 本文为极客兔兔“动手写分布式缓存GeeCache”学习笔记。 一、回顾 昨天已经开发了一部分项目,我们先来看看项目结构。 分布式缓存需要实现节点间通信,建立基于 HTTP 的…...
STM32 低功耗模式
目录 背景 低功耗模式 睡眠模式 进入睡眠模式 退出睡眠模式 停止模式 进入停止模式 退出停止模式 待机模式 进入待机模式 退出待机模式 程序 睡眠模式 休眠模式配置 进入休眠模式 退出睡眠模式 停止模式 停止模式配置 进入停止模式 退出停止模式 待机模式…...
AI 百炼成神:逻辑回归, 垃圾邮件分类
第二个项目:逻辑回归垃圾邮件分类 项目代码下载地址:https://download.csdn.net/download/m0_56366541/90398247 项目目标 学习逻辑回归的基本概念。使用逻辑回归算法来实现垃圾邮件的分类。理解如何处理文本数据以及如何评估分类模型的性能。项目步骤 准备数据集 我们将使…...
#渗透测试#批量漏洞挖掘#Apache Log4j反序列化命令执行漏洞
免责声明 本教程仅为合法的教学目的而准备,严禁用于任何形式的违法犯罪活动及其他商业行为,在使用本教程前,您应确保该行为符合当地的法律法规,继续阅读即表示您需自行承担所有操作的后果,如有异议,请立即停止本文章读。 目录 Apache Log4j反序列化命令执行漏洞 一、…...
Docker__持续更新......
Docker 1. 基本知识1.1 为什么有Docker?1.2 Docker架构与容器化 画图解释 画图解释2. 项目实战 1. 基本知识 1.1 为什么有Docker? 用一行命令跨平台安装项目,在不同平台上运行项目。把项目打包分享运行应用。 1.2 Docker架构与容器化 准备机器,在机…...
什么是语料清洗、预训练、指令微调、强化学习、内容安全; 什么是megatron,deepspeed,vllm推理加速框架
什么是语料清洗、预训练、指令微调、强化学习、内容安全 目录 什么是语料清洗、预训练、指令微调、强化学习、内容安全语料清洗预训练指令微调强化学习内容安全什么是megatron,deepspeed,vllm推理加速框架语料清洗 语料清洗是对原始文本数据进行处理的过程,旨在去除数据中的…...
Ubuntu虚拟机NDK编译ffmpeg
目录 一、ffmpeg源码下载1、安装git(用于下载ffmpeg源码)2、创建源码目录,下载ffmpeg源码 二、下载ubuntu对应的NDK,并解压到opt下1、下载并解压2、配置 ~/.bashrc 三、源码编译、1、创建编译脚本2、脚本文件内容3、设置可执行权限并运行4、编译的结果在…...
SQLAlchemyError: A transaction is already begun on this Session.
资料 sqlalchemy 事务 - 简书 在 SQLAlchemy 中,事务是通过会话来管理的。当你开始一个事务(例如使用 async with db.begin()),它会开启一个新的事务,并在事务块结束时自动提交或回滚。如果在同一个会话中,…...
STM32 HAL库USART串口DMA IDLE中断编程:避坑指南
HAL_UART_Receive接收最容易丢数据了,STM32 HAL库UART查询方式实例 可以考虑用中断来实现,但是HAL_UART_Receive_IT还不能直接用,容易数据丢失,实际工作中不会这样用,STM32 HAL库USART串口中断编程:演示数据丢失, 需要在此基础优化一下. STM32F103 HAL库USART串口…...
打造一个有点好看的 uniapp 网络测速软件
大家好,我是一名前端小白。今天想和分享一个有点好看的网络测速 uniapp 组件的实现过程。这个组件不仅外观精美,而且具有完整的功能性,是一个非常适合学习和实践的案例。 设计理念 在开始coding之前,先聊聊设计理念。一个好的测…...
DeepSeek AI 视频创作完整指南:从注册到制作
DeepSeek AI 视频创作完整指南:从注册到制作 前言 DeepSeek作为国产AI的新星,不仅在代码能力上表现出色,在创意内容生成方面同样令人惊艳。本教程将带您从注册到实操,完整体验DeepSeek的强大功能。 第一步:获取Deep…...
DeepSeek学术指南:DeepSeek在学术翻译改写能力应用操作案例!
DeepSeek:助力学术研究的智能工具 DeepSeek作为一种先进的智能工具,为学术研究提供了强大的支持。它不仅能够处理复杂的学术文本,还能在翻译和润色方面表现出色,极大地提高了学术写作的效率和质量。通过其强大的语言处理能力&…...
栈回溯基础
指令集区分 thumb指令集 长度:thumb指令通常是 16 位。特点:thumb 指令集是为了压缩指令集长度减少程序占用空间。对齐方式:2字节对齐,存放 thumb 指令的地址一般会被1,设置为奇数,用于表示地址上存放的是…...
JavaScript系列(76)--浏览器API深入
JavaScript浏览器API深入 🌐 浏览器提供了丰富的API,使JavaScript能够与浏览器环境进行交互。本文将深入探讨常用的浏览器API、最佳实践和性能优化技巧。 核心浏览器API 🌟 💡 小知识:浏览器API是连接JavaScript与浏…...
计算机网络(3)TCP格式/连接
1、TCP三大特点:面向连接、可靠、基于字节流 2、如何唯一确定一个TCP连接?TCP四元组:源地址、源端口、目的地址、目的端口 源地址和目标地址的字段(32 位)是在 IP 头部中,作用是通过 IP 协议发送报文给对方主机源端口和目标端口…...
下载安装运行测试开源vision-language-action(VLA)模型OpenVLA
1. 安装 项目官网OpenVLA 首先按照官网提示的以下代码,执行创建环境->安装最小依赖->git克隆项目等 # Create and activate conda environment conda create -n openvla python3.10 -y conda activate openvla# Install PyTorch. Below is a sample comma…...
3D与2D机器视觉机械臂引导的区别
3D与2D机器视觉在机械臂引导中的主要区别如下: 数据维度 2D视觉:仅处理平面图像,提供X、Y坐标信息,无法获取深度(Z轴)数据。 3D视觉:处理三维空间数据,提供X、Y、Z坐标及物体的姿态…...
网站搭建基本流程
需求分析: 实现网站搭建的过程:首先进行网站的需求性分析 网站可分为前台系统和后台系统,由不同的功能拆分为不同的模块 如下是一个电商网站可以拆分出的模块: 在编写代码前,我们要先对网站进行架构,通过…...
SpringBoot启动时报错:cannot use an unresolved DNS server address: I:53
报错如下: 2025-02-17 13:59:41.374 [main] ERROR org.springframework.boot.SpringApplication:835 - Application run failed org.springframework.beans.factory.UnsatisfiedDependencyException: Error creating bean with name mySwaggerResourceProvider def…...
Ollama命令使用指南
Ollama 命令使用指南 Ollama 命令使用指南1. Ollama 命令概览2. Ollama 命令详解2.1 启动 Ollama2.2 创建模型2.3 查看模型信息2.4 运行模型2.5 停止运行的模型2.6 从注册表拉取模型2.7 推送模型到注册表2.8 列出本地模型2.9 查看正在运行的模型2.10 复制模型2.11 删除模型 3. …...
每日一题——将数字字符串转化为IP地址
将数字字符串转化为IP地址 题目描述解题思路回溯法步骤分解 代码实现全局变量有效性验证函数回溯函数主函数完整代码 复杂度分析关键点说明总结 这题难度还挺大的,整体上实现并不容易。建议参考视频 和https://programmercarl.com/0093.%E5%A4%8D%E5%8E%9FIP%E5%9C%…...
2013年下半年软件设计师上午题考察知识点及其详细解释(附真题及答案解析)
以下是2013年下半年软件设计师上午题的所有题目(从第1题到第75题)的总结,按顺序列出每道题目的考察知识点及其详细解释,供考生背诵记忆: 1. Cache与主存的地址映像 知识点:存储管理解释:Cache与…...
实现可拖拽的 Ant Design Modal 并保持下层 HTML 可操作性
前言 在开发复杂的前端界面时,我们常常需要一个可拖拽的弹窗(Modal),同时又希望用户能够在弹窗打开的情况下操作下层的内容。Ant Design 的 Modal 组件提供了强大的功能,但默认情况下,弹窗会覆盖整个页面&…...
unity学习43:子状态机 sub-state machine
目录 1sub-state machine子状态机 1.1 创建 sub-state machine 1.2 sub-state machine 内容 1.3 子状态机的应用 2 子状态机不同于blend tree的嵌套 3 应用例子:若角色拿不同武器的动画设计,可以使用2种方法 3.1 在1个图层layer里,使用…...
HTML之JavaScript DOM(document)编程处理事件
HTML之JavaScript DOM(document)编程处理事件 <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"…...
线性模型 - Logistic 回归
Logistic 回归(Logistic Regression,LR)是一种常用的处理二分类问题的 线性模型。 Logistic 回归是一种用于二分类问题的统计模型,它通过将输入特征的线性组合映射到一个概率值来进行分类决策。 Logistic回归是机器学习中最经典的分类算法之一…...
分割 学习笔记cvpr2024
目录 LiteMedSam 模型37m LightM-Unet 500 str 依赖项: MLWnet 73 star memsam 340M 126 star LiteMedSam 模型37m https://github.com/bowang-lab/MedSAM/blob/LiteMedSAM/README.md LightM-Unet 500 str https://github.com/MrBlankness/LightM-UNet/blob model = Li…...
ShenNiusModularity项目源码学习(9:项目结构)
ShenNiusModularity源码主要有11个project(其实还有officialweb、test两个文件夹,大致有4、5个project,但看着跟主要项目代码没太大关系,暂时不管),这11个project的依赖关系如下图所示,其中最下…...
【复现DeepSeek-R1之Open R1实战】系列6:GRPO源码逐行深度解析(上)
目录 4 GRPO源码分析4.1 数据类 GRPOScriptArguments4.2 系统提示字符串 SYSTEM_PROMPT4.3 奖励函数4.3.1 accuracy_reward函数4.3.2 verify函数4.3.3 format_reward函数 4.4 将数据集格式化为对话形式4.5 初始化GRPO Trainer 【复现DeepSeek-R1之Open R1实战】系列3࿱…...
【LangChain实践开发】如何对大模型I/O封装?
LangChain 的核心组件 模型 I/O 封装 LLMs:大语言模型Chat Models:一般基于 LLMs,但按对话结构重新封装PromptTemple:提示词模板OutputParser:解析输出 数据连接封装 Document Loaders:各种格式文件的加载…...
LangChain4j
LangChain4j 是一个基于 Java 的框架,旨在简化与大型语言模型(LLMs)的集成和应用开发。它类似于 Python 的 LangChain 框架,但专为 Java 开发者设计,提供了构建基于 LLM 的应用程序所需的工具和组件。 主要功能 LLM …...
如何系统成为高级Qt工程师?
要系统性地成为高级Qt工程师,需要从基础到进阶逐步构建知识体系,并结合实战经验、源码分析和架构设计能力的提升。以下是分阶段的系统性学习路径和建议: 一、夯实基础阶段 C++深度掌握 精通C++11/14/17特性(智能指针、lambda、移动语义等)理解面向对象设计、设计模式(如观…...
LLM 架构
LLM 分类 : 自编码模型 (encoder) : 代表模型 : BERT自回归模型 (decoder) : 代表模型 : GPT序列到序列模型 (encoder-decoder) : 代表模型 : T5 自编码模型 (AutoEncoder model , AE) 代表模型 : BERT (Bidirectional Encoder Representation from Transformers)特点 : Enc…...
MVTEC数据集笔记
前言 网上的博客只有从论文里摘出的介绍,没有数据集文件详细的样子,下载数据集之后,对数据集具体的构成做一个补充的笔记。 下载链接:https://ai-studio-online.bj.bcebos.com/v1/7d4a3cf558254bbaaf4778ea336cb14ed8bbb96a7f2a…...
[算法学习笔记]1. 枚举与暴力
一、枚举算法 定义 枚举是基于已有知识来猜测答案的问题求解策略。即在已知可能答案的范围内,通过逐一尝试寻找符合条件的解。 2. 核心思想 穷举验证:对可能答案集合中的每一个元素进行尝试终止条件:找到满足条件的解,或遍历完…...
javacv将mp4视频切分为m3u8视频并播放
学习链接 ffmpeg-demo 当前对应的 gitee代码 Spring boot视频播放(解决MP4大文件无法播放),整合ffmpeg,用m3u8切片播放。 springboot 通过javaCV 实现mp4转m3u8 上传oss 如何保护会员或付费视频?优酷是怎么做的? - HLS 流媒体加密 ffmpe…...
docker 进阶命令(基于Ubuntu)
数据卷 Volume: 目录映射, 目录挂载 匿名绑定: 匿名绑定的 volume 在容器删除的时候, 数据卷也会被删除, 匿名绑定是不能做到持久化的, 地址一般是 /var/lib/docker/volumes/xxxxx/_data 绑定卷时修改宿主机的目录或文件, 容器内的数据也会同步修改, 反之亦然 # 查看所有 vo…...
鸿蒙(HarmonyOS)开发学习路线指南:从零到实战
随着鸿蒙生态的快速发展,HarmonyOS 已成为物联网时代的重要开发平台。其分布式架构和“一次开发、多端部署”的理念吸引了大量开发者。本文将从零开始梳理鸿蒙开发的学习路径,帮助开发者高效掌握核心技能。 一、学习路线概览 总目标:掌握鸿蒙…...