《JavaEE进阶》----12.<SpringIOC&DI【扫描路径+DI详解+经典面试题+总结】>

CSDN 2024-09-13 13:35:01 阅读 66

本篇博客主要讲解

扫描路径

DI详解:三种注入方式及优缺点

经典面试题

总结

五、环境扫描路径

虽然我们没有告诉Spring扫描路径是什么,但是有一些注解已经告诉Spring扫描路径是什么了

如启动类注解@SpringBootApplication。

里面有一个注解是@componentScan这个注解就是扫描路径注解。

默认扫描路径

默认为:启动类所在的路径

被@componentScan标识的当前类所在的路径。

而启动类@SpringBootApplication包含了@componentScan。因此也是被@SpringBootApplication标识的类。

注:

这五大注解必须在Spring的扫描路径下才会生效。

<code>package com.qiyangyang.iocdome;

因此

我们这个项目Spring的扫描路径是com.qiyangyang.iocdome这个路径。

因此,当我们移动IocDomeApplication这个类到不同的文件中。Spring的默认扫描路径就会发生改变。

我们通常把启动类放在最外层(Controller、Service...的上一层)。这样就可以扫描到使用IoC的类了。如果非要放在别的地方。你希望它扫描哪些路径,你也可以指定。

指定扫描路径

我们也可以通过 @ComponentScan注解 指定扫描路径

指定方法:在启动类上面加上如下注解

扫描路径的起始文件地址是java文件的下一级目录。

<code>@ComponentScan(“扫描路径”)

eg:

@ComponentScan(“com.qiyangyang.iocdome”)

如果什么都没加,默认为被该注解标识的所在的类的路径。

六、依赖注入(DI)详解

DI(Dependency-Injection依赖注入)

容器在运行期间,动态的为应用程序提供运行时所依赖的资源,称之为依赖注入。

程序运⾏时需要某个资源,此时容器就为其提供这个资源. 从这点来看, 依赖注⼊(DI)和控制反转(IoC)是从不同的⻆度的描述的同⼀件事情,就是指通过 引入IoC容器,利用依赖关系注入的方式,实现对象之间的解耦。

依赖注入是⼀个过程,是指IoC容器在创建Bean时,去提供运行时所依赖的资源,而资源指的就是对象. 在下面程序案例中

我们使用了 @Autowired 这个注解,完成了依赖注⼊的操作. 简单来说,就是把对象取出来放到某个类的属性中.。在一些文章中。依赖注⼊也被称之为"对象注入","属性装配",

具体含义需要结合文章的上下文来理解

关于依赖注入,Spring也给我们提供了三种方式:

6.1 @Autowired属性注入(Field-Injection)

注入流程:

这个方法我们在上一篇文章使用过。这是重述一遍

<code>@RestController

@RequestMapping("/book")

public class BookController {

//属性注入

@Autowired

private BookService bookService;

}

由于BookService类已经被@Service注解过。因此已经将BookService类的对象存入了Spring容器

在我们后续使用它的对象的时候。不需要我们自己new这个对象。而是可以通过如上定义一个BookService属性。通过@Autowired注解,将这个对象取出。这样后续就可以直接用这个对象 

6.2 构造方法注入(Constructor-Injection)

注:当我们手动加上构造函数之后。一定要养成习惯把默认无参构造方法写上

当有单个构造函数

Spring知道使用哪个。因此会正常运行。

@RestController

@RequestMapping("/book")

public class BookController {

//构造方法注入

private BookService bookService;

public BookController(BookService bookService){

this.bookService = bookService;

}

@Controller("bean") //创建对象 //括号中是对bean进行重命名 如果没有指定名称spring帮我们指定

public class UserController {

private UserService userService;

private UserRepository userRepository;

public UserController(UserService userService, UserRepository userRepository) {

this.userService = userService;

this.userRepository = userRepository;

}

public void say(){

userService.say();

userRepository.say();

System.out.println("Hi Controller!!!");

}

}

当有多个构造函数

<code>@Controller("bean") //创建对象 //括号中是对bean进行重命名 如果没有指定名称spring帮我们指定

public class UserController {

private UserService userService;

private UserRepository userRepository;

public UserController(){

}

public UserController(UserRepository userRepository) {

this.userRepository = userRepository;

}

public UserController(UserService userService, UserRepository userRepository) {

this.userService = userService;

this.userRepository = userRepository;

}

public void say(){

userService.say();

userRepository.say();

System.out.println("Hi Controller!!!");

}

}

 优先使用默认无参构造函数。如果没有。就会报错。

 @Autowired 指定使用某个构造函数:

<code> @Autowired

public UserController(UserService userService, UserRepository userRepository) {

this.userService = userService;

this.userRepository = userRepository;

}

6.3 Setter 注入(SetterInjection)

 也需要搭配@AutoWired注解才可以正常注入。

<code>private UserService userService;

private UserRepository userRepository;

@Autowired

public void setUserService(UserService userService) {

this.userService = userService;

}

@Autowired

public void setUserRepository(UserRepository userRepository) {

this.userRepository = userRepository;

}

6.3三种依赖注入优缺点

属性注入(Spring最不推荐,但没关系):

优点:

简介,使用方便;

缺点:

1.只能用于 IoC 容器。因为它是Spring提供的。

若不是 IoC 容器不可用,且只有在使用的时候才会出现 NPE(空指针异常)

2.不能注入一个final修饰的属性。

构造方法注入(Spring4.x推荐):

我们可以点进@Autowired注解。再点击目录。看到我们现在使用的Spring版本。

优点:

1.可以注入final修饰的属性

注:final修饰的属性有一个要求,需要满足下列任意条件

①声明时,要完成初始化

②在构造函数中进项赋值。

2.注入的对象不会被修改

3.依赖对象在使用前一定会被完全初始化,因为依赖是在类在构造方法中执行的。而构造方法是在类加载阶段就会执行的方法。

4.通用性好。构造方法是JDK支持的,所以更换任何框架他都适用。

缺点:

1.注入多个对象时,代码会比较繁琐。

Setter注入(Spring3.x推荐)

优点:

方便在类实例之后,重新对该对象进行配置或注入

缺点:

1.不能注入一个final修饰的属性

2.注入对象可能会被改变。因为setter方法可能会被多次调用。就有被修改的风险。

6.4@AutoWired存在的问题

当同一类型存在多个Bean时。使用@Autowired会存在问题

<code>@Configuration

public class BeanConfig {

@Bean

public UserInfo userInfo1(){

UserInfo userInfo = new UserInfo();

userInfo.setId(120);

userInfo.setName("张三");

userInfo.setAge(18);

return userInfo;

}

@Bean

public UserInfo userInfo2(){

UserInfo userInfo2 = new UserInfo();

userInfo2.setId(121);

userInfo2.setName("李四");

userInfo2.setAge(28);

return userInfo2;

}

}

报错的原因是非唯一Bean对象。

Spring提供了一下几种解决方案。

@Primary

在构造多个对象时

使用@Primary注解:当存在多个相同类型的Bean注入时,加上@Primary注解,来确定默认的实现。

<code>@Configuration

public class BeanConfig {

@Primary

@Bean

public UserInfo userInfo1(){

UserInfo userInfo = new UserInfo();

userInfo.setId(120);

userInfo.setName("张三");

userInfo.setAge(18);

return userInfo;

}

@Bean

public UserInfo userInfo2(){

UserInfo userInfo2 = new UserInfo();

userInfo2.setId(121);

userInfo2.setName("李四");

userInfo2.setAge(28);

return userInfo2;

}

}

这样就不会报错了。也就是默认使用张三这个对象进行使用。

@Qualifier

使用@Qualifier注解:在注入时。指定当前要注入的bean对象。在@Qualifier的value属性中,指定注入的bean 的名称。

@Qualifier注解不能单独使用,必须配合@Autowired使用

@Qualifier("userInfo2")//指定bean的名称

@Autowired

private UserInfo userInfo;

public void say(){

System.out.println("Hi Controller!!!");

userService.say();

System.out.println(userInfo);

}

@Resource

使用@Resource注解:是按照bean的名称进行注入。通过name属性指定要注入的bean的名称。

不用@Autowirede注解配合使用。

@Resource(name="userInfo2")code>

private UserInfo userInfo;

public void say(){

System.out.println("Hi Controller!!!");

userService.say();

System.out.println(userInfo);

}


常见面试题:

1.@Autowird与@Resource的区别

• @Autowired是spring框架提供的注解,@Resource是JDK提供的注解

@Autowired 默认是按照类型注入,@Resource是按照名称注入,相比于@Autowired来说, @Resource支持更多的参数设置,例如 name 设置,根据名称获取 Bean。 

2.Spring,SpringBoot,SpringMVC之间的区别和联系,你是如何理解的?

我的理解

1.Spring(很早) 简单来说,Spring是一个轻量级、一站式、模块化的开发应用框架。主要用于简化企业级应用程序开发。

Spring的主要功能:管理对象,以及对象之间的依赖关系。面向切面编程、数据库事务管理、数据访问、web框架支持等。

Spring(这里指Spring-core。Spring家族都具有高度可开放性)具备高度可开放性。并不强制依赖Spring,开发者可以自由选择Spring的部分或者全部,Spring可以无缝集成第三方框架。比如数据访问框架(Hibernate、JPA)、web框架(如Struts、JSF)。

在使用Spring时,不强制使用Spring框架,也可以使用第三方框架。

2.SpringBoot(晚) 是对Spring的一个封装,为简化Spring应用的开发而出现的。中小型企业。没有成本研究自己的框架,使用SpringBoot可以快速的搭建框架,降低开发成本。让开发人员更加专注于Spring应用的开发。而无需过多关注xml的配置和一些底层实现。

SpringBoot是一个手脚架,插拔式搭建项目,可以快速的集成其他框架进来。

比如想使用SpringBoot开发Web项目,只需要引入SpringMVC即可。Web开发的工作是SpringMVC完成的。而不是SpringBoot。想完成数据访问,只需要引入Mybatis框架即可。

SpringBoot只是辅助简化项目开发的,让开发变得更简单,甚至不需要额外的web服务器,直接生产jar包执行即可。

使我们在创建项目的时候可以直接添加一些依赖。并且内置web服务器、提供许多注解方便我们书写代码。对项目进行更多的监控指标,更好的了解项目的运行情况。简化我们的开发。

3.SpringMVC(早) 是一个Spring家族的子框架。是针对Web开发和网络接口的一种MVC的思想的实现。也被称作Spring Web MVC(Spring Web)。

在创建项目时,我们添加的依赖Spring Web实际上引的就是SpringMVC。可以认为Spring给我们提供的Web功能就叫做SpringMVC。

我们现在认为SpringMVC就是SpringWeb。主要进行Web开发(网站开发)。

最后一句总结:Spring MVC和Spring Boot都属于Spring。Spring MVC是基于Spring的一个MVC框架。而Spring Boot是基于Spring的一套快速开发整合包

比如我们的图书系统代码中

整体框架是通过SpringBoot搭建的

IoC、DI功能是Spring的提供的,

web相关功能是Spring MVC提供的。

这三者专注的领域不同,解决的问题也不⼀样,总的来说,Spring就像一个大家族,有众多衍生产 品,但他们的基础都是Spring,用一张图来表他们三个的关系:

假如把Spring看作火车。(而做项目相当于坐火车) 但是它买票不方便。 

因此就可以把SpringBoot看作是12306。而12306不仅可以订票还可以订酒店。打的等等。让我们坐火车(做项目更加的方便)

而SpringMVC 可以认为是火车里面提供的一些功能。比如买票,改签,插座等等。(注解/Cookie&Session)

3.ApplicationContext VS BeanFactory

1.继承关系和功能方面来说:

Spring 容器有两个顶级的接口:BeanFactory和 ApplicationContext。其中BeanFactory提供了基础的访问容器的能力,而ApplicationContext 属于BeanFactory的⼦类,它除了继承了BeanFactory的所有功能之外, 它还拥有独特的特性,还添加了对国际化支持、资源访问支持、以及事件传播等方面的支持.

2.从性能方面来说:

ApplicationContext 是⼀次性加载并初始化所有的 Bean 对象,而BeanFactory 是需要那个才去加载那个,因此更加轻量. (空间换时间) 

4.三种依赖注入优缺点

属性注入(Spring最不推荐,但没关系):

优点:

简介,使用方便;

缺点:

1.只能用于 IoC 容器。因为它是Spring提供的。

若不是 IoC 容器不可用,且只有在使用的时候才会出现 NPE(空指针异常)

2.不能注入一个final修饰的属性。

构造方法注入(Spring4.x推荐):

我们可以点进@Autowired注解。再点击目录。看到我们现在使用的Spring版本。

优点:

1.可以注入final修饰的属性

注:final修饰的属性有一个要求,需要满足下列任意条件

①声明时,要完成初始化

②在构造函数中进项赋值。

2.注入的对象不会被修改

3.依赖对象在使用前一定会被完全初始化,因为依赖是在类在构造方法中执行的。而构造方法是在类加载阶段就会执行的方法。

4.通用性好。构造方法是JDK支持的,所以更换任何框架他都适用。

缺点:

1.注入多个对象时,代码会比较繁琐。

Setter注入(Spring3.x推荐)

优点:

方便在类实例之后,重新对该对象进行配置或注入

缺点:

1.不能注入一个final修饰的属性

2.注入对象可能会被改变。因为setter方法可能会被多次调用。就有被修改的风险。

5.常见注解有哪些?分别是什么作用?

1. web url映射: @RequestMapping

2.参数接收和接口响应:@RequestParam,@RequestBody,@ResponseBody

3.bean的存储:@Controller,@Service,@Repository,@Component,@Configuration,@Bean

4.bean的获取: @Autowired, @Qualifier, @Resource

5.多个bean的默认指定:@Primary

6.Spring两大核心思想IOC和AOP

待续


总结:

告诉spring管理Bean,Bean的存储

1.类注解:五大注解@Controller@Component@Configuration@Service@Repository

2.方法注解:@Bean

Bean的名称:

1.五大注解

类名首字母小写,如果前两位字母均为大写,则为原类名,也可以指定Bean的名称

指定方法:@Controller(“beanName”)

2.@bean

默认名称是方法名,也可以指定名称:@Bean(“beanName”)

使用场景

1.五大注解 自己开发的程序

2.@Bean

①存储第三方的对象(代码不在自己的项目中)

②一个类型需要创建多个对象时

获取bean三种扫描路径的方式

通过Spring运行环境的Spring上下文 

<code>ApplicationContext context = SpringApplication.run(IocDomeApplication.class, args);

获取Bean的功能是BeanFeactory提供的 

三种方式

1.通过类型

UserController bean1 = context.getBean(UserController.class);

bean1.say();

2.通过bean名称

UserService userService =(UserService) context.getBean("userService");

userService.say();

3.通过bean名称+类型

UserComponent userComponent = context.getBean("userComponent", UserComponent.class);

userComponent.say();

扫描路径:

默认扫描路径:启动类所在的路径

指定扫描路径: 在启动类上面加上如下注解

@ComponentScan(“扫描路径”)



声明

本文内容仅代表作者观点,或转载于其他网站,本站不以此文作为商业用途
如有涉及侵权,请联系本站进行删除
转载本站原创文章,请注明来源及作者。