1.Bean自动装配

  • 自动装配是Spring满足Bean依赖的一种方式!
  • Spring会在上下文中自动寻找,并自动给Bean装配属性

在Spring中有三种装配方式

1.在xml显示的配置

2.在Java中显示的配置

3.隐式的自动装配Bean(重要)

1.1 ByName自动装配

在之前的set注入的时候会有重复操作出现例如:

<bean id="cat" class="com.wangxu.POJO.Cat"/>
    <bean id="dog" class="com.wangxu.POJO.Dog"/>
    <bean id="people" class="com.wangxu.POJO.People">
        <property name="name" value="王旭旭呀"/>
        <property name="dog" ref="dog"/>
        <property name="cat" ref="cat"/>
</bean>

所以我们采用ByName自动装配的方法,可以简化操作并且可以实现自动装配注入

<bean id="people" class="com.wangxu.POJO.People" autowire="byName">
        <property name="name" value="王旭旭呀"/>
</bean>

ByName:会在容器上下文中自动查找,找到和自己对象set方法后面的值对应的beanId!

1.2 byType自动装配

byType:会在容器上下文中自动查找,找到和自己对象属性对应的bean!

<bean id="cat" class="com.wangxu.POJO.Cat"/>
    <bean id="dog123123" class="com.wangxu.POJO.Dog"/>
    <bean id="people" class="com.wangxu.POJO.People" autowire="byType">
        <property name="name" value="王旭旭呀"/>
</bean>

总结:

  • byName的时候,需要保证所有bean的id唯一,并且这个bean需要和自动注入的属性的set方法的值一致!
  • byType的时候,需要保证所有bean的class唯一,并且这个bean需要和自动注入的属性的类型一致!

1.3 使用注解实现自动装配

jdk1.5支持的注解,spring2.5就支持注解了

要使用注解须知:

1.导入约束

<context:annotation-config/>

2.配置注解支持

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xmlns:context="http://www.springframework.org/schema/context"
    xsi:schemaLocation="http://www.springframework.org/schema/beans
        https://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/context
        https://www.springframework.org/schema/context/spring-context.xsd">
    <context:annotation-config/>
</beans>

@Autowired

可以直接属性或者set方法上使用即可,如果使用@Autowired注解可以忽略set方法,前提是你这个自动装配的属性在 IOC(Spring)容器中存在且符合名字byName

科普:两种可以使字段为null的方法

@Nullable 字段记住了这个注解,说明这个字段可以为null
public @interface Autowired {
    boolean required() default true;
}
//如果显示定义了Autowired的required的值为false,说明这个对象可以为null,否则不允许为空
    @Autowired(required = false)

@Qualifier

如果@Autowired的环境比较复杂,自动装配无法通过一个注解去完成的时候,我们可以使用@Qualifier(value = “x’x’x”)去配合@Autowired的使用,指定一个唯一的bean对象注入,qualifier的意思是合格者,通过这个标示,表明了哪个实现类才是我们所需要的,我们修改调用代码,添加@Qualifier注解,需要注意的是@Qualifier的参数名称必须为我们之前定义@Service注解的名称之一!

@Qualifier(value = "dog")
private Dog dog;

@Resource

①如果同时指定了name和type,则从Spring上下文中找到唯一匹配的bean进行装配,找不到则抛出异常。

②如果指定了name,则从上下文中查找名称(id)匹配的bean进行装配,找不到则抛出异常。

③如果指定了type,则从上下文中找到类似匹配的唯一bean进行装配,找不到或是找到多个,都会抛出异常。

④如果既没有指定name,又没有指定type,则自动按照byName方式进行装配;如果没有匹配,则回退为一个原始类型进行匹配,如果匹配则自动装配。

总结@Resource和@Autowired的区别

  • 他们两个都是用来自动装配的,都可以放在属性字段上。
  • @Autowired通过byType的方式实现,而且必须要求这个对象的存在
  • @Resource默认通过byName的方式实现,如果找不到,则通过byType实现!如果两个都找不到的情况下,就报错!

2.使用注解开发

在spring4之后,要使用注解开发,必须要保证aop的包导入了

使用注解需要导入context约束和增加注解的支持

2.1 bean

import org.springframework.stereotype.Component;
//等价于<bean id="user" class="com.wangxu.POJO.User"/>
//@Component组件
@Component
public class User {
    public String name = "王旭";
}

2.2 属性如何注入

第一种直接注入在变量上
    //相当于 <bean id="user" class="com.wangxu.POJO.User">
    //         <property name="name" value="王旭"/>
    //     </bean>
    @Value("王旭")
    public String name;
第二种注入在set方法上
    @Value("王旭")
    public void setName(String name) {
        this.name = name;
    }

2.3 衍生的注解

@Component有几个衍生注解,我们在web开发中,会按照mvc三层架构分层

  • dao [@Repository]
  • service[@Service]
  • controller[@Controller]

这四个注解功能都是一样的,都是代表将某个类注入到Spring容器中,装配Bean

2.4 自动装配

@Autowired:自动装配 类型-->名字
    如果Autowired不能唯一自动装配上属性,则需要通过@Qualifier(value = "xxx")
@Nullable:字段记住了这个注解,说明这个字段可以为null
@Resource:自动装配 名字-->类型

2.5 作用域

@Scope(“singleton”)----单例模式

@Scope(“prototype”)—原型模式

2.6 小结

xml与注解:

  • xml更加万能,适用于任何场合,维护更加简单方便
  • 注解:不是自己的类使用不了,维护相对复杂

最佳实践:

  • xml用来管理bean
  • 注解负责属性注入
  • 我们在使用的过程中,只需要使用注意一个问题:必须让注解生效,就需要开启注解支持!
<!--指定要扫描的包,这个包下的注解就会生效-->
 <context:component-scan base-package="com.wangxu"/>
 <!--注解依赖-->
 <context:annotation-config/>

3.使用Java的方式去配置spring

完全不使用xml配置,全权交给Java来做!

Java Config是spring的一个子项目,在spring4后他成了一个新功能

首先建造一个实体类

package com.wangxu.POJO;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Component;
//@Component就说明这个类被spring接管了,注册到了容器中
@Component
public class User {
    private String name;

    public String getName() {
        return name;
    }
    @Value("王旭")//向set注入值
    public void setName(String name) {
        this.name = name;
    }
    @Override
    public String toString() {
        return "User{" +
                "name='" + name + '\'' +
                '}';
    }
}

其次建造一个Config配置类

//@Configuration也会被spring容器托管,注册到容器中,因为他本身就是一个@Component
//@Configuration代表这是一个配置类,就和之前的beans.xml是一样的
@Configuration
@ComponentScan("com.wangxu.POJO")
@Import(MyConfig2.class)
public class MyConfig {
    //@Bean注册一个bean,就相当于我们之前写的一个bean标签
    //这个方法的名字就相当于bean标签中的id属性
    //这个方法的返回值,就相当于bean标签中的class属性
    @Bean
    public User getUser(){
        return new User();//就是返回要注入到bean的对象!
    }
}

最后测试

public class MyTest {
    public static void main(String[] args) {
        //如果完全使用了配置类方式去做,我们就只能通过AnnotationConfigApplicationContext来获取容器
        //通过配置类的class对象加载
        AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(MyConfig.class);
        User getUser = context.getBean("getUser", User.class);
        System.out.println(getUser.getName());
    }
}

我们可以发现通过这种新加一个Config配置类的方法,可以使我们完全摒弃xml配置文件!