热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

MVC进阶之路:依赖注入(Di)和Ninject

MVC进阶之路:依赖注入(Di)和Ninject0X1什么是依赖注入依赖注入(DependencyInjection),是这样一个过程:某客户类只依赖于服务类的一个接口,而不依赖于

MVC进阶之路:依赖注入(Di)和Ninject 0X1 什么是依赖注入

  依赖注入(Dependency Injection),是这样一个过程:某客户类只依赖于服务类的一个接口,而不依赖于具体服务类,所以客户类只定义一个注入点。在程序运行过程中,客户类不直接实例化具体服务类实例,而是客户类的运行上下文环境或专门组件负责实例化服务类,然后将其注入到客户类中,保证客户类的正常运行。

技术分享

                                                      图1

  如图1所示,数据库操作类DataManager中依赖的IDataBase的接口,而不是以来IDataBase的具体实现类,这样的好处是可以让我们的程序具有扩展性:无论我们要使用SqlServer当我们的数据库操作类还是用Mysql,我们需要改动的地方都很少。

0X2 一个Demo理解Di

  我们创建一个控制台程序,并按照图1中所标识的关系创建四个类。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
public class DataManager
    {
        private IDataBase _database;
        public DataManager(IDataBase database)
        {
            this._database = database;
        }
 
        public void Add()
        {
            _database.Add();
        }
 
        public void Delete()
        {
            _database.Delete();
        }
 
        public void Update()
        {
            _database.Update();
        }
 
        public void Select()
        {
            _database.Select();
        }
    }

IDataBase接口

1
2
3
4
5
6
7
8
public interface IDataBase
    {
        string DbName { get; }
        void Select();
        void Update();
        void Delete();
        void Add();
    }
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
public class SqlServer : IDataBase
    {
        public string DbName
        {
            get
            {
                return "SqlServer";
            }
        }
 
        public void Add()
        {
            Console.WriteLine("Add in " + DbName);
        }
 
        public void Delete()
        {
            Console.WriteLine("Delete in " + DbName);
        }
 
        public void Select()
        {
            Console.WriteLine("Select in " + DbName);
        }
 
        public void Update()
        {
            Console.WriteLine("Update in " + DbName);
        }
    }
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
public string DbName
        {
            get
            {
                return "Mysql";
            }
        }
 
        public void Add()
        {
            Console.WriteLine("Add in " + DbName);
        }
 
        public void Delete()
        {
            Console.WriteLine("Delete in " + DbName);
        }
 
        public void Select()
        {
            Console.WriteLine("Select in " + DbName);
        }
 
        public void Update()
        {
            Console.WriteLine("Update in " + DbName);
        }

如果我们要用Sqlserver作为数据库实现类的话,传统写法我们要New Sqlserver,如下图:

技术分享

项目早期使用的是Sqlserver数据库没问题,后来BOSS忽然要改成Mysql数据库,你心里面一想,没问题啊,无非就是把New后面的Sqlserver换成Mysql嘛,简单。于是你就开始改,越改越不对劲,已经你已经发现当前项目已经有好几十个地方用到New Sqlserver(),而剩余的数量往往还是未知的。这个时候你就在想,有没有方法可以让我只需要改一个地方其他地方不用动就可以呢?答案是肯定的,目前这种方法实现有很多种,最简单的无非就是工厂模式。但是,今天,我们不用工厂,我们将使用Ninject来实现这种功能。

0X3 Ninject的使用

  Ninject是一个IOC容器,用来解决程序中组件的耦合问题,它的目的在于做到最少配置。其他的的IOC工具过于依赖配置文件,需要使用assembly-qualified名称来进行定义,庸长且复杂常常因为打错字而破坏程序。这些是他的优点,也是为什么要选择它。

安装Ninject

  打开Nuget程序包管理控制台输入“Install-Package Ninject”即可安装Ninject。

  技术分享

Ninject使用步骤

  我们一般会在程序启动的入口来创建Ninject的对象负责类型的注册。

1  static void Main(string[] args)
2         {
3             //实例化Ninject对象
4             IKernel Kerner = new StandardKernel();
5         }

  使用Ninject对象分两个步骤,第一步是把接口对象或者说被依赖的对象(IDataBase)绑定到Ninject中,然后在为其绑定对应的实例类型(如果要使用SqlServer则就绑定SqlServer)。

 //实例化Ninject对象
 IKernel Kerner = new StandardKernel();
 //绑定对象
 Kerner.Bind().To();

  第二步则是通过Ninject的Get方法获取IDataBase的实现类

 var Db = Kerner.Get(); //由于上面IDataBase绑定的是SqlServer类型,所以这里获取的类型是SqlServer

技术分享

  上面的代码大家可能体会不到使用Ninject的好处,也没有体会到依赖注入的奥妙所在。依赖注入大致分为三类:接口注入,函数注入,属性注入。我们来通过一个例子来演示属性注入。

  我们增加一个IShowDBInfo的接口类,其有一个Show方法。然后添加一个Show类实现IShowDBInfo接口:

技术分享
 public class Show1 : IShowDBInfo
    {
        public void Show()
        {
            Console.WriteLine(this.GetType().FullName);
        }
    }
技术分享

    我们给DataManager添加一个IShowDBInfo属性并增加一个Show()方法:

技术分享
 public class DataManager
    {
        private IDataBase _database;
        private IShowDBInfo _showDbInfo;
        public DataManager(IDataBase database, IShowDBInfo ishowdbinfo)
        {
            this._database = database;
            _showDbInfo = ishowdbinfo;
        }
       //省略Add,Updata,Delete,Select方法
        public void Show()
        {
            _showDbInfo.Show();
        }
    }     
技术分享

  在应用程序启动的时候注册IShow类和注册DataManager类并启动程序:

技术分享
 static void Main(string[] args)
        {
            //实例化Ninject对象
            IKernel Kerner = new StandardKernel();
            //绑定对象
            Kerner.Bind().To();
            Kerner.Bind().To();
            Kerner.Bind().ToSelf();
            var dataManager=Kerner.Get();
            dataManager.Show();
            Console.Read();
        }
技术分享

技术分享

  从上代码以及运行结果来看,我们只需要在向Ninject实例里面注册对象,然后在其他类中使用的时候我们只需要定义接口就可以了,并不需要实例化对象。这样做的话可以使我们和其他层进行松耦合。

MVC进阶之路:依赖注入(Di)和Ninject


推荐阅读
  • 本文详细介绍了SQL日志收缩的方法,包括截断日志和删除不需要的旧日志记录。通过备份日志和使用DBCC SHRINKFILE命令可以实现日志的收缩。同时,还介绍了截断日志的原理和注意事项,包括不能截断事务日志的活动部分和MinLSN的确定方法。通过本文的方法,可以有效减小逻辑日志的大小,提高数据库的性能。 ... [详细]
  • 本文介绍了lua语言中闭包的特性及其在模式匹配、日期处理、编译和模块化等方面的应用。lua中的闭包是严格遵循词法定界的第一类值,函数可以作为变量自由传递,也可以作为参数传递给其他函数。这些特性使得lua语言具有极大的灵活性,为程序开发带来了便利。 ... [详细]
  • 本文介绍了在开发Android新闻App时,搭建本地服务器的步骤。通过使用XAMPP软件,可以一键式搭建起开发环境,包括Apache、MySQL、PHP、PERL。在本地服务器上新建数据库和表,并设置相应的属性。最后,给出了创建new表的SQL语句。这个教程适合初学者参考。 ... [详细]
  • 本文介绍了数据库的存储结构及其重要性,强调了关系数据库范例中将逻辑存储与物理存储分开的必要性。通过逻辑结构和物理结构的分离,可以实现对物理存储的重新组织和数据库的迁移,而应用程序不会察觉到任何更改。文章还展示了Oracle数据库的逻辑结构和物理结构,并介绍了表空间的概念和作用。 ... [详细]
  • 本文介绍了使用Java实现大数乘法的分治算法,包括输入数据的处理、普通大数乘法的结果和Karatsuba大数乘法的结果。通过改变long类型可以适应不同范围的大数乘法计算。 ... [详细]
  • HDU 2372 El Dorado(DP)的最长上升子序列长度求解方法
    本文介绍了解决HDU 2372 El Dorado问题的一种动态规划方法,通过循环k的方式求解最长上升子序列的长度。具体实现过程包括初始化dp数组、读取数列、计算最长上升子序列长度等步骤。 ... [详细]
  • 本文介绍了如何使用php限制数据库插入的条数并显示每次插入数据库之间的数据数目,以及避免重复提交的方法。同时还介绍了如何限制某一个数据库用户的并发连接数,以及设置数据库的连接数和连接超时时间的方法。最后提供了一些关于浏览器在线用户数和数据库连接数量比例的参考值。 ... [详细]
  • 本文讨论了如何优化解决hdu 1003 java题目的动态规划方法,通过分析加法规则和最大和的性质,提出了一种优化的思路。具体方法是,当从1加到n为负时,即sum(1,n)sum(n,s),可以继续加法计算。同时,还考虑了两种特殊情况:都是负数的情况和有0的情况。最后,通过使用Scanner类来获取输入数据。 ... [详细]
  • 本文介绍了C#中数据集DataSet对象的使用及相关方法详解,包括DataSet对象的概述、与数据关系对象的互联、Rows集合和Columns集合的组成,以及DataSet对象常用的方法之一——Merge方法的使用。通过本文的阅读,读者可以了解到DataSet对象在C#中的重要性和使用方法。 ... [详细]
  • 本文介绍了OC学习笔记中的@property和@synthesize,包括属性的定义和合成的使用方法。通过示例代码详细讲解了@property和@synthesize的作用和用法。 ... [详细]
  • 在说Hibernate映射前,我们先来了解下对象关系映射ORM。ORM的实现思想就是将关系数据库中表的数据映射成对象,以对象的形式展现。这样开发人员就可以把对数据库的操作转化为对 ... [详细]
  • 本文介绍了在SpringBoot中集成thymeleaf前端模版的配置步骤,包括在application.properties配置文件中添加thymeleaf的配置信息,引入thymeleaf的jar包,以及创建PageController并添加index方法。 ... [详细]
  • 本文详细介绍了Linux中进程控制块PCBtask_struct结构体的结构和作用,包括进程状态、进程号、待处理信号、进程地址空间、调度标志、锁深度、基本时间片、调度策略以及内存管理信息等方面的内容。阅读本文可以更加深入地了解Linux进程管理的原理和机制。 ... [详细]
  • 本文介绍了在Mac上搭建php环境后无法使用localhost连接mysql的问题,并通过将localhost替换为127.0.0.1或本机IP解决了该问题。文章解释了localhost和127.0.0.1的区别,指出了使用socket方式连接导致连接失败的原因。此外,还提供了相关链接供读者深入了解。 ... [详细]
  • 解决VS写C#项目导入MySQL数据源报错“You have a usable connection already”问题的正确方法
    本文介绍了在VS写C#项目导入MySQL数据源时出现报错“You have a usable connection already”的问题,并给出了正确的解决方法。详细描述了问题的出现情况和报错信息,并提供了解决该问题的步骤和注意事项。 ... [详细]
author-avatar
林老悲--
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有