知识储备
- spring有两个很大的特点:面向切面编程(AOP)和依赖注入(DI),现在只了解DI简单介绍DI
- 应用场景:类A的某个属性是B类对象,这种情况实例化类的时候必须要创建这个B对象,也就意味着需要向A类中引入另一个类B
import B;
class A {
private B b;
public A(){
b= new B()
}
}
---->使用上述方式缺点明显在于:
- AB耦合度很高
- 并且对A做单元测试的时候,没有办法mock B
- 因此DI引入。
DI简单理解
- 依赖注入是什么:创建一个需要依赖其他类B的对象A,DI会将A依赖的B给,而不是让A自己去获取
- DI实际功能:DI帮助我们创建了对象并且管理依赖关系。当我们将类的创建和装配交给别的类管理时候,我们只需要简单的使用例如@Inject@Autowired这些注解直接注入对象即可。
- 如何实现DI:XML或者创建ApplicationContext类(这个后面会详细解析)
- DI
- 但是上一条的前提是要求我们的类比如A使用构造器注入方式
class A {
private B b;
public A(B b){
this.b = b;
}
}
- 装配:创建应用组件之间协作的行为(比如我要创建一个A的bean但是我需要B那么创建A的Bean的过程就是装配)
注意点:使用IDEA+gradle当你引入新的depend的时候要点击下方的import否则包不会引入--->也就等同于如果不npm install 那么就不会引入库
spring的配置
- 作用:1.告诉spring那些类需要Bean2.告诉spring针对那些有依赖的类,如何创建他的Bean
- 方式:1.使用xml文件配置 2. 使用javaConfig类配置 3.隐式的bean发现机制和自动赚
一: spring的自动化配置
自动化配置分成两个方向:
- 1.组件扫描:扫描就是找到需要创建Bean的类并且将Bean创建好放入到spring的上下文
- 2.自动装配:类似于上文中的A类bean的创建需要依赖B的bean这时候创建A的Bean将B的bean注入A就是自动装配
自动化扫描
- what:就是通过为类添加注解,告诉spring那些类需要创建Bean
- how:1.为需要bean的类添加@Component/@Service/@等注解,作用是告诉spring这些类是需要创建Bean的(类似于在某个方法上添加@Bean注解)。 2.添加configuration类:作用是告诉spring去找类似于@Component这类创建Bean
3.简单的在你的运行类中加载spring的时候让他去执行configuration类即可 - example:https://github.com/iepgnahz/auto-configuration.git commitId 2f6775f2660787fcfacc6e8c62032bbead5f7e25
自动化扫描中遇到的bug
- Q1:我想通过测试自动注入的cd是否存在,来确定我们使用的@Component和@ComponentScan是否可以spring创建CD的bean,但是测试中我只添加了@Runwith注解立刻就爆出没办法创建Test bean,为什么?
- A1:虽然我们创建了config类,但是在运行测试的时候并没有让测试执行config类,因此,我们需要在运行类的定义处添加注解@ContextConfiguration提醒运行类执行config类
@RunWith(SpringRunner.class)
@ContextConfiguration(classes = CDPlayerConfig.class)
//@SpringBootTest //你会发现使用这个注解不需要上面那个注解也可以成功运行,我认为是这个注解自动去扫描了@Component类并且为你创建了Bean
public class AutoConfigApplicationTests {
@Autowired
private CompactDisc cd;
@Test
public void play_should_print_CD_info() throws Exception {
assertThat(cd).isNotNull();
} //这测试仅仅是用来确定cd 可以自动扫描并创建bean(仅仅使用两个注解)
}
- Q2:为什么当我们只添加上面代码中的@SpringBootTest就可以运行成功?
- A2:我猜测是springBoot自动帮我们创建了Config类并让spring执行,找到所有带有@Component@Service等的类并未他们创建Bean
自动化装配
- why/what:某个类A其中有个属性是B对象,当创建A的bean的时候需要到spring上下文中找B的bean才能创建成功,希望B的bean可以自动注入A的Bean中就是自动化装配
- how:使用@Autowired注解自动注入B入A
class A{
private B b;
@Autowired
public A(B b){
this.b =b
} //这是构造方法注入,推荐使用这种方式,这样可以方便的对A做单元测试,因为可以在使用A的构造方法同时注入一个mock的B bean
}
class A{
@Autowired
private B b;
//这是直接注入
}
@Autowired可以用在属性的声明之前还可以用在各种带有Bean参数的方法前,诸如构造方法,setB方法,但是要求参数必须是可以注入的Bean
- example:https://github.com/iepgnahz/auto-configuration/ commitId cf0352aa5bfef4ccc4f62eb8967cc055dcca1704
自动化装配demo中遇到的bug
- Q1:想要对CDplayer类做一个单元测试,因此需要Mock掉使用的CompactDisc的对象测试CDPlayer的playCD方法,然而playCD是一个没有返回值的方法,那我该如何测试呢?并且其中调用了CompactDisc的play方法也是没有返回值的,那么该如何mock?
- A1:针对没有返回值函数的测试,我们使用verify测试一下函数是否被调用即可。
- Q2:那么verify该如何使用呢?
- A2:verify可以用来测试某一个被mock的对象中的方法是否被调用或者调用几次或者使用什么参数调用,但是注意:verify的第一个参数就是那个被mock的对象,并且verify只能检测被mock掉的对象中方法的调用情况,没有被mock的对象没办法测试
- Q3.无法确定veriy能够处理的对象有哪些呢?因为我记得在nodejs中只有把对象spy了才能够监测他,在java中不需要吗?
- A3:verify既可以处理Mock的对象也可以处理spy对象