.NET Core ASP.NET Core Basic 1-2 控制反转与依赖注入

本节内容为控制反转与依赖注入

简介

控制反转IOC

这个内容事实上在我们的C#高级篇就已经有所讲解,控制反转是一种设计模式,你可以这样理解控制反转,假设有一个人他有一部A品牌手机,他用手机进行听歌、打游戏,那么你可以创建一个手机类和一个人类

class APhone : IPhone
{public string Owner{get;set;}public Phone(string own){Owner = own;}void Play(){}void Music(){}
}
class Man
{public string Name{get;set;}void Game(){var p = new APhone(Name);p.Play();}
}

事实上这段代码的耦合度是比较高的?它使用的是正转,也就是我需要什么东西的时候我就自己创建一个这个东西。为什么说他不好呢,如果有一天这个人决定再也不使用A品牌手机了,他决定以后只使用B品牌。那么也就意味着整个的Man类使用过APhone类的地方都需要更改。这是一个非常麻烦的事情,我们这个时候就需要运用我们的IOC控制反转了。我们将实例或者是需要使用的对象的创建交给你的调用者,自己只负责使用,其它人丢给你依赖的这个过程理解为注入。

控制反转的核心就是——原本我保存使用我自己的东西,现在我把控制权交给我的上级,我需要使用的时候再向他要。这个时候,接口的作用不言而喻,A继承了Phone接口,B也继承了,假定我们一开始就使用Phone接口去创建不同的A,B对象,那么是不是可以有效的切换AB对象呢?

依赖注入

依赖注入体现的是一个IOC(控制反转),它非常的简单,我们之前的Man类代码中使用的是正转的方式,也就是我要一个对象,那么我就创建一个。现在我们使用依赖注入就是将我们对这个对象的控制权交给上一级接口,也就成为了这种,我想要一个对象,我就向上级发出请求,上级就给我创建了一个对象。我们通常使用构造函数注入的方式进行依赖的注入。

上文的代码就会变成

class Man
{private readonly IPhone _phone;public Man(IPhone phone){_phone = phone;}
}

假设这个时候你需要将手机换成B品牌,那么只需要再注入的地方传入B品牌的对象即可了。

容器

但是现在又出现了一个新的问题,假设说这个类有100个使用该接口的依赖,如果,我们是不是要在100个地方做这样的事情?控制是反转了,依赖的创建也移交到了外部。现在的问题是依赖太多,我们需要一个地方统一管理系统中所有的依赖,这个时候,我们就使用容器进行集中的管理

容器负责两件事情:

  • 绑定服务与实例之间的关系

  • 获取实例,并对实例进行管理(创建与销毁)

使用

说了那么多,我们如何在.NET Core中使用我们的依赖注入呢?这里我们针对的是所有的.NET Core的应用,在.NET Core中依赖注入的核心分为两个组件:位于Microsoft.Extensions.DependencyInjection命名空间下的IServiceCollection和 IServiceProvider。

其中

  • IServiceCollection 负责注册

  • IServiceProvider 负责提供实例

在默认的容器ServiceCollection中有三个方法

  • .AddTransient<I,C>()

  • .AddSingleton<I,C>()

  • .AddScoped<I,C>()

这里就不得不提一下我们依赖注入的三种生命周期了

  • Singleton指的是单例模式,也就是说,在整个程序运转期间只会生成一次

  • Transient指每一次GetService都会创建一个新的实例

  • Scope指在同一个Scope内只初始化一个实例 ,可以理解为( 每一个request级别只创建一个实例,同一个http request会在一个 scope内)

我们可以尝试使用控制台项目来模拟依赖注入的原理,也就是说我们直接从容器获取我们对象实例,并且我们使用Guid进行唯一性的标记。

interface IPhoneScope{Guid Guid { get; }}interface IPhoneSingleton{Guid Guid { get; }}interface IPhoneTransient{Guid Guid { get; }}class PhoneService:IPhoneScope,IPhoneSingleton,IPhoneTransient{public PhoneService(){this._guid = Guid.NewGuid();}public PhoneService(Guid guid){this._guid = guid;}private Guid _guid;public Guid Guid =&gt; this._guid;}

然后,在我们的主函数中

namespace DI_AND_IOC
{class Program{static void Main(string[] args){var services = new ServiceCollection().AddScoped&lt;IPhoneScope, PhoneService&gt;().AddTransient&lt;IPhoneTransient, PhoneService&gt;().AddSingleton&lt;IPhoneSingleton, PhoneService&gt;();var provider = services.BuildServiceProvider();using (var scope = provider.CreateScope()){var p = scope.ServiceProvider;var scopeobj1 = p.GetService&lt;IPhoneScope&gt;();var transient1 = p.GetService&lt;IPhoneTransient&gt;();var singleton1 = p.GetService&lt;IPhoneSingleton&gt;();var scopeobj2 = p.GetService&lt;IPhoneScope&gt;();var transient2 = p.GetService&lt;IPhoneTransient&gt;();var singleton2 = p.GetService&lt;IPhoneSingleton&gt;();Console.WriteLine($"scope1: {scopeobj1.Guid},\n" +$"transient1: {transient1.Guid}, \n" +$"singleton1: {singleton1.Guid}\n");Console.WriteLine($"scope2: {scopeobj2.Guid}, \n" +$"transient2: {transient2.Guid},\n" +$"singleton2: {singleton2.Guid}\n");}using (var scope = provider.CreateScope()){var p = scope.ServiceProvider;var scopeobj3 = p.GetService&lt;IPhoneScope&gt;();var transient3 = p.GetService&lt;IPhoneTransient&gt;();var singleton3 = p.GetService&lt;IPhoneSingleton&gt;();Console.WriteLine($"scope3: {scopeobj3.Guid}, \n" +$"transient3: {transient3.Guid},\n" +$"singleton3: {singleton3.Guid}");}}}
}

你应该会得到类似以下的数据

scope1: 096d38e5-0c7b-4e50-9c79-241fb18a56ed,
transient1: 289ebd11-8159-4f22-b53e-ed738a317313,
singleton1: b453b7f5-3594-4b66-99c8-a72763abaa83scope2: 096d38e5-0c7b-4e50-9c79-241fb18a56ed,
transient2: 212ad420-e54c-4dd6-9214-abe91aacdd9c,
singleton2: b453b7f5-3594-4b66-99c8-a72763abaa83scope3: 688b6ffd-a8c1-47f4-a20a-872c2285d67c,
transient3: 3d09997d-fffb-43d1-9e53-ccf9771c819d,
singleton3: b453b7f5-3594-4b66-99c8-a72763abaa83

可以发现,singleton对象是不会发生改变的,而scope对象在创建新的scope之后就发生了改变,而transient对象每一次请求都在发生改变。

需要注意的是,在控制台项目使用容器服务需要引入 *** Microsoft.Extensions.DependencyInjection *** 程序集,你可以在引入中导入该dll

通过对注入服务的生命周期管控,在一些ASP.NET Core项目中,有些类(服务)有可能跨越了多个Action或者Controller,那么我们正确的使用生命周期,我们可以尽可能的节省内存,即能减少实例初始化的消耗。

在ASP.NET Core中的使用

在ASP.NET Core中,我们使用依赖注入非常的简单,在StartUp类中的ConfigureServices方法中已经为我们构建好了容器,我们只需要做类似于这样的操作

services.AddScoped&lt;IPhoneScope, PhoneService&gt;();
services.AddDbContext&lt;DbContext&gt;();
services.AddMVC();

如果你需要在控制器中注入服务,官方的推荐方案是使用构造函数注入

public IPhoneScope _ips;
public Controller(IPhoneScope ips)
{_ips = ips;
}

特别的,你如果使用MVC的Razor页面进行注入的话,那么输入以下指令

如果我的文章帮助了您,请您在github.NETCoreGuide项目帮我点一个star,在博客园中点一个关注和推荐。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/314004.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

2019-03-15-算法-进化(有效的数独)

题目描述 判断一个 9x9 的数独是否有效。只需要根据以下规则&#xff0c;验证已经填入的数字是否有效即可。数字 1-9 在每一行只能出现一次。 数字 1-9 在每一列只能出现一次。 数字 1-9 在每一个以粗实线分隔的 3x3 宫内只能出现一次。上图是一个部分填充的有效的数独。 数独…

STL归并排序

STL归并排序 https://blog.csdn.net/weixin_44176696/article/details/104431124 分为原地归并和异地归并&#xff0c;而且可以自定义比较函数&#xff0c;非常方便。

清明梦超能力者黄YY、异或树(线段树合并)

清明梦超能力者黄YY 这题有点像【雨天的尾巴】【永无乡】的结合版本&#xff0c;树上差分&#xff0c;线段树合并&#xff0c;权值线段树查找第kkk大。 对于操作iii&#xff0c;我们可以对u−>vu->vu−>v路径上的点&#xff0c;iii的权值加上111&#xff0c;然后线段…

net core WebApi——尝试企业微信来开发企业内部应用

前言这几天忙活着别的东西&#xff0c;耽误了很长时间&#xff0c;从文件操作完了之后就在考虑着下一步鼓捣点儿啥&#xff0c;因为最开始的业务开发就是企业微信相关的&#xff0c;这刚好来做个内部应用的小例子玩玩。企业微信前身是企业号&#xff0c;当时微信主推的还是公众…

2019-03-18-算法-进化(字符串中的第一个唯一字符)

题目描述 给定一个字符串&#xff0c;找到它的第一个不重复的字符&#xff0c;并返回它的索引。如果不存在&#xff0c;则返回 -1。 案例: s "leetcode" 返回 0.s "loveleetcode", 返回 2.注意事项&#xff1a;您可以假定该字符串只包含小写字母。 思…

P2619 [国家集训队]Tree I(WQS二分/带权二分/最小生成树)

P2619 [国家集训队]Tree I 给定一个n个点&#xff0c;m条边的无向图&#xff0c;每条边有一个颜色黑色或者白色&#xff0c;求解恰好有k条白色边的最小生成树。 那么看到恰好选择k个的最优性问题&#xff0c;我们可以利用WQS二分解决&#xff0c;实际上就是利用了对于每个选择…

E. Party Company(树上问题)

E. Party Company 容易发现这是一颗树形结构&#xff0c;根节点为111&#xff0c;并且有点权从根节点开始递减。 题目大意就是给定一个点u,l,ru, l, ru,l,r&#xff0c;对于于uuu在同一个连通块里&#xff0c;并且点权是在[l,r][l, r][l,r]之间的点答案贡献加一。 如果理解到…

微软发布了开发社区采用.NET Standard的最新信息

最近&#xff0c;微软发布了开发社区当前采用.NET Standard的最新信息。.NET Standard是API的正式规范&#xff0c;现有.NET实现在不同平台的是通用的&#xff08;从而允许跨平台开发&#xff09;。当前规范&#xff08;版本2.0&#xff09;在两年前发布&#xff0c;在.NET Cor…

2019-03-18-算法-进化(实现strStr())

题目描述 实现 strStr() 函数。 给定一个 haystack 字符串和一个 needle 字符串&#xff0c;在 haystack 字符串中找出 needle 字符串出现的第一个位置 (从0开始)。如果不存在&#xff0c;则返回 -1。 示例 1: 输入: haystack "hello", needle "ll" 输…

CF429E Points and Segments(欧拉回路)

CF429E Points and Segments 给定n 条线段[li,ri][l_i,r_i][li​,ri​] ,然后给这些线段红蓝染色&#xff0c;求最后直线上上任意一个点被蓝色及红色线段覆盖次数之差的绝对值不大于1 首先见到绝对值不大于1我们就容易想到欧拉回路&#xff0c;因为欧拉回路可以用来构造恰好相…

卓语言对泛型类的使用

上次发了中文编程语言卓语言《小卓.NET中文编程特点介绍》。这篇文章来看下卓语言对泛型类的使用。泛型是现代编程语言很重要的功能。C#语言可以完全定义和使用泛型类型。卓语言是面向广大非专业人员的&#xff0c;为了减低编程难度&#xff0c;没有实现定义泛型类型&#xff0…

快速傅里叶变换(完整推导过程 + 模板)

快速傅里叶变换 多项式表示 系数表示法&#xff1a; 一个nnn次多项式可以用n1n 1n1个系数表示出来&#xff1a;f(x)a0a1xa2x2⋯an−1xn−1anxnf(x) a_0 a_1 x a_2 x ^ 2 \dots a_{n - 1} x ^{n- 1} a_n x ^nf(x)a0​a1​xa2​x2⋯an−1​xn−1an​xn。 点值表示法&a…

2019-03-19-算法-进化(报数)

题目描述 报数序列是一个整数序列&#xff0c;按照其中的整数的顺序进行报数&#xff0c;得到下一个数。其前五项如下&#xff1a; 1. 1 2. 11 3. 21 4. 1211 5. 1112211 被读作 “one 1” (“一个一”) , 即 11。 11 被读作 “two 1s” (“两个一”&…

AT2675 [AGC018F] Two Trees(欧拉回路)

AT2675 [AGC018F] Two Trees 首先我们看到1或-1&#xff0c;那么就是限制差距在1以内&#xff0c;然后我们可以想到构造一些东西来满足这种东西&#xff0c;然后我们经常利用的就是欧拉回路。 首先这是两个树&#xff0c;然后我们可以根据儿子个数来判断当前点的奇偶性&#x…

.netcore 分布式事务CAP2.6之控制台使用

上一编.netcore 分布式事务CAP2.6 快速入门 讲了cap2.6的快速入门&#xff0c;这次我们来讲讲在控制台中如何使用cap2.6。因为cap2.6的内存模式目前已经可以使用了&#xff0c;相关组件已经更新&#xff0c;所以这次我们以简单的内存模式为例。1&#xff1a;创建项目创建一个名…

2019-03-18-算法-进化(有效的字母异位词)

给定两个字符串 s 和 t &#xff0c;编写一个函数来判断 t 是否是 s 的一个字母异位词。 示例 1: 输入: s "anagram", t "nagaram" 输出: true示例 2: 输入: s "rat", t "car" 输出: false说明: 你可以假设字符串只包含小写字母…

P6378 [PA2010] Riddle(2-sat/前后缀优化建图)

P6378 [PA2010] Riddle n个点m条边的无向图&#xff0c;分为k个部分&#xff0c;从每个部分选择恰好一个关键点&#xff0c;使得每条边至少有一个端点是关键点。 首先有这么多的限制&#xff0c;实际上就是一个选或者不选的问题&#xff0c;每条边的限制相当于一个不选就必须…

2019-03-18-算法-进化(删除链表的倒数第N个节点)

给定一个链表&#xff0c;删除链表的倒数第 n 个节点&#xff0c;并且返回链表的头结点。 示例&#xff1a; 给定一个链表: 1->2->3->4->5, 和 n 2.当删除了倒数第二个节点后&#xff0c;链表变为 1->2->3->5.说明&#xff1a; 给定的 n 保证是有效的。…

TestinPro应用与DevOps之路

文 | 中国农业银行软件研发中心 系统支持部 王晓昕 程伟静 胡莉莉Testin Pro&#xff08;云测平台&#xff09;是一款移动端自动化测试平台工具&#xff0c;帮助用户实现移动端测试自动化&#xff0c;是一套设备统一调配、软硬件一体化的移动端测试方案。Testin Pro具有在线录制…

多项式开根

多项式开根 给定多项式g(x)g(x)g(x)&#xff0c;求f(x)f(x)f(x)&#xff0c;满足f2(x)g(x)f ^ 2(x) g(x)f2(x)g(x)。 假设我们已经得到了g(x)g(x)g(x)&#xff0c;膜x⌈n2⌉x ^{\lceil \frac{n}{2} \rceil}x⌈2n​⌉下的根f0(x)f_0 (x)f0​(x)&#xff0c;要求膜xnx ^ nxn下…