一、IOC控制反转和DI依赖注入
1.控制反转,字面可以理解为:主动权的转移,原来一个应用程序内的对象是类通过new去主动创建并实例化的,对对像创建的主动权在程序代码中。程序不仅要管理业务逻辑也要管理对的象创建和依赖关系。这是很累的,也跟软件工程 "低耦合高内聚" 的概念不十分符合。
有了spring的ioc容器之后,对象的实例化和依赖关系管理都由IOC容器进行统一管理,主体类只要依赖ioc容器就够了,需要啥,容器会给他注入进去,也就是只要声明对象不用再主动去new,ioc容器帮忙把相应的对象注入到声明对象中,使其变成实例化对象。(类似主体类提供一个躯体,ioc容器把灵魂注入进去,使其变成一个生命体,激活他),这样创建对象的主动权就转移交接了,
二、使用xml配置方式实现IOC
1.在ioc容器中配置了dao实现类和service类的bean,在容器加载的时候就会实例化这些bean到内存中。(bean.xml配置如下)
<beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd "> <!-- BookDao Bean --> <bean id="bookDao" class="com.study.DaoImpl.BookDaoImpl"></bean> <!-- BookService Bean--> <bean id="bookService" class="com.study.Service.BookService"> <!-- BookService中的声明了BookDao对象,通过ref属性将BookDao的bean注入到对象中 --> <property name="bookDao" ref="bookDao"/> </bean> </beans>
2. service类中需要用到dao类的实例(正常情况下需要new一个dao类对象),但是用ioc容器接管后只需要声明dao接口对象即可,然后写一个dao对象的set方法。(要注入的对象必须要有set方法,否则将报错 Bean property 'bookDao' is not writable or has an invalid setter method)因为spring注入是根据反射机制实现的,他在反射注入的时候会调用该方法名的set方法,如果set方法写错,或者根本没写,那么注入就会失败。(BookService类如下)
public class BookService { private BookDao bookDao; public BookService() { System.out.println("BookService实例化"); } public void setBookDao(BookDao bookDao) { System.out.println("BookService属性初始化装配成功"); this.bookDao = bookDao; } public void storeBook(String bookname){ System.out.println("图书上架"); System.out.println(bookDao.addBook(bookname)); } }
如上代码:BookSerivce类需要用到BookDao对象,但是却没有new对象,只有一个set方法,这个set方法就是ioc容器注入的入口(必不可少),
3.此处我们用ApplicationContext作为容器,初始化配置文件,然后从容器中根据id名取出容器中已经帮我们实例化好的对象。
public class TestDmeo { BookService bookService; @Test public void testStoreBook(){ System.out.println("容器初始化"); ApplicationContext app = new ClassPathXmlApplicationContext("bean.xml"); bookService = (BookService) app.getBean("bookService");//将对象注入到声明好的BookService对象中。(bookService就是配置文件中的id) bookService.storeBook("Spring MVC"); } }
4.dao类和实现类如下:
接口类:
public interface BookDao { public String addBook(String BookName); }
实现类:
public class BookDaoImpl implements BookDao { public BookDaoImpl() { System.out.println("BookDao实例化"); } public String addBook(String BookName) { return BookName+"添加成功"; } }
5.运行测试结果:
6.大体思路如下图:
程序中除了初始化容器用了new对象,其余的基本没有new的存在。
二、注解方式配置IOC
注解配置方式目的和xml配置的目的一样,都是为了实现bean的创建。常用的注解如下:
@Component 在类定义之前添加@Component注解,他会被spring容器识别,并转为bean。
@Repository 对Dao实现类进行注解 (特殊的@Component)
@Service 用于对业务逻辑层进行注解, (特殊的@Component)
@Controller 用于控制层注解 , (特殊的@Component)
装配注解如下:
@Autowired 默认按照类型装配注入,想按照名称来装配的话要结合@Quapfier(“name”)一起使用,使用@Autowired注解可以不用set方法。@Autowired 注释进行自动注入时,Spring 容器中匹配的候选 Bean 数目必须有且仅有一个
@Quapfier("name") 中的name是bean的名字,也就是id,和@Autowired可以作为限定专配对象的名称
@Resource 默认按照名称装配注入,当找不到对应名成的bean的时候就按照类型匹配,如果还是找不到的话就会报错,@Autowired是spring提供的,@Resource是javaee提供,使用@Resource可以减少对spring的依赖
范例:
1.例子同上,只是配置bean的方式从xml文件中转移到了代码中,用注解体现。
2.除了把配置文件中<bean id="" class=""/>变成对应的注解外,另外一个区别在于,bean.xml文件中的修改,需要做如下,配置才能够使注解生效
context的配置有如下方法:
1.仅扫描特定包下的特定类:
<context:component-scan base-package="com.study" resource-pattern="Service/B*.class"/>
这是扫描Service包下B开头的所有类。
2.使用<context:include-filter .../>和<context:exclude-filter .../>配置那些需要和不需要的扫描的包
Filter Type | Examples Expression | Description |
annotation | org.example.SomeAnnotation | 符合SomeAnnoation的target class(注解类) |
assignable | org.example.SomeClass | 指定class或interface的全名(指定确切的类或接口) |
aspectj | org.example..*Service+ | AspectJ语法 |
regex | org\.example\.Default.* | Regelar Expression (正则表达式) |
custom | org.example.MyTypeFilter | Spring3新增自定义Type,org.springframework.core.type.TypeFilter |
<!-- 容器扫描包下的注解配置组件 --> <context:component-scan base-package="com.study" use-default-filters="false"> <context:include-filter type="aspectj" expression="com.study.Service.*"/> <!-- 模糊过滤 --> <context:include-filter type="annotation" expression="org.springframework.stereotype.Component"/><!-- 过滤指定的注解 --> <context:include-filter type="assignable" expression="com.study.Service.BookService"/><!-- 过滤指定的类或接口,路径要完整,如果是接口的话,所有派生类都会被过滤 --> <context:include-filter type="regex" expression="com.*"/> </context:component-scan>
<context:exclude-filter ../>要配在<context:include-filter .../>的后面。
以上这篇详谈Spring对IOC的理解(推荐篇)就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持小牛知识库。
本文向大家介绍spring data jpa使用详解(推荐),包括了spring data jpa使用详解(推荐)的使用技巧和注意事项,需要的朋友参考一下 使用Spring data JPA开发已经有一段时间了,这期间学习了一些东西,也遇到了一些问题,在这里和大家分享一下。 前言: Spring data简介: Spring Data是一个用于简化数据库访问,并支持云服务的开源框架。其主要目标是使
本文向大家介绍Nginx配置详解(推荐),包括了Nginx配置详解(推荐)的使用技巧和注意事项,需要的朋友参考一下 序言 Nginx是lgor Sysoev为俄罗斯访问量第二的rambler.ru站点设计开发的。从2004年发布至今,凭借开源的力量,已经接近成熟与完善。 Nginx功能丰富,可作为HTTP服务器,也可作为反向代理服务器,邮件服务器。支持FastCGI、SSL、Virtual Hos
本文向大家介绍可伸缩的textview详解(推荐),包括了可伸缩的textview详解(推荐)的使用技巧和注意事项,需要的朋友参考一下 在Android原生的TextView的基础上,可收缩/扩展的TextView:PhilExpandableTextView。 实现原理:核心是控制TextView的max lines。在TextView的初始化阶段但尚未绘制出View的时候,使用ViewTree
本文向大家介绍详解Golang 推荐的命名规范,包括了详解Golang 推荐的命名规范的使用技巧和注意事项,需要的朋友参考一下 Golang 推荐的命名规范 很少见人总结一些命名规范,也可能是笔者孤陋寡闻, 作为一个两年的golang 开发者, 我根据很多知名的项目,如 moby, kubernetess 等总结了一些常见的命名规范。 命名规范可以使得代码更容易与阅读, 更少的出现错误。 如有不同
本文向大家介绍谈谈你对闭包的理解?相关面试题,主要包含被问及谈谈你对闭包的理解?时的应答技巧和注意事项,需要的朋友参考一下 说明: bar在foo函数的代码块中定义。我们称bar是foo的内部函数。 在bar的局部作用域中可以直接访问foo局部作用域中定义的m、n变量。 简单的说,这种内部函数可以使用外部函数变量的行为,就叫闭包。 闭包的意义与应用
本文向大家介绍谈谈对jquery ui tabs 的理解,包括了谈谈对jquery ui tabs 的理解的使用技巧和注意事项,需要的朋友参考一下 jquery ui 点击此处下载,也可以去官网查看文档帮助。 1 属性 1.11 ajaxOptions,当选项卡加载内容时,添加一个ajax选项。只有ajax时,添加的ajax选项才起作用。默认值为null。上面的例子中,添加了beforeSend
本文向大家介绍请谈谈你对JVM的理解?相关面试题,主要包含被问及请谈谈你对JVM的理解?时的应答技巧和注意事项,需要的朋友参考一下 Java虚拟机(JVM)是运行 Java 字节码的虚拟机。JVM有针对不同系统的特定实现(Windows,Linux,macOS),目的是使用相同的字节码,它们都会给出相同的结果。 什么是字节码?采用字节码的好处是什么? 在 Java 中,JVM可以理解的代码就叫做(
本文向大家介绍谈谈你对多态的理解?相关面试题,主要包含被问及谈谈你对多态的理解?时的应答技巧和注意事项,需要的朋友参考一下 多态就是指程序中定义的引用变量所指向的具体类型和通过该引用变量发出的方法调用在编程时并不确定,而是在程序运行期间才确定,即一个引用变量到底会指向哪个类的实例对象,该引用变量发出的方法调用到底是哪个类中实现的方法,必须在程序运行期间才能决定。因为在程序运行时才确定具体的类,这样