五、Spring(2)

一、Bean的自动装配

官网 Autowired

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

在Spring中有三种装配方式:

  1. 在xml中显示的配置
  2. 在java中显示的配置
  3. 隐式的自动装配bean【🌻重点哦🌻】

1.1 测试案例准备

一个人有两个宠物

(1)创建子工程(spring-04-autowired)

在这里插入图片描述

(2)创建对象(Cat.java,Dog.java,People.java)
Cat.java

package com.zql.pojo;

import java.io.Serializable;

/**
 * @Author:Daniel
 * @Version 1.0
 */
public class Cat implements Serializable {

    public void shout(){

          System.out.println("喵喵喵🐈🐈🐈");
    }
}

Dog.java

package com.zql.pojo;

import java.io.Serializable;

/**
 * @Author:Daniel
 * @Version 1.0
 */
public class Dog implements Serializable {

    public void shout(){
       System.out.println("汪汪汪🐕🐕🐕");
    }
}

People.java

package com.zql.pojo;

import java.io.Serializable;

/**
 * @Author:Daniel
 * @Version 1.0
 */
public class People implements Serializable {

    private Cat cat;
    private Dog dog;
    private String name;

    public Cat getCat() {
        return cat;
    }

    public void setCat(Cat cat) {
        this.cat = cat;
    }

    public Dog getDog() {
        return dog;
    }

    public void setDog(Dog dog) {
        this.dog = dog;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    @Override
    public String toString() {
        return "People{" +
                "cat=" + cat +
                ", dog=" + dog +
                ", name='" + name + '\'' +
                '}';
    }
}

(3) 创建beans.xml
applicationContext.xml

<?xml version="1.0" encoding="UTF-8"?>
<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
        https://www.springframework.org/schema/beans/spring-beans.xsd">

        <bean id="cat" class="com.zql.pojo.Cat"></bean>
        <bean id="dog" class="com.zql.pojo.Dog"></bean>

        <bean id="people" class="com.zql.pojo.People">
        	<property name="name" value="Daniel🤣🤣"></property>
            <property name="cat" ref="cat"></property>
            <property name="dog" ref="dog"></property>
        </bean>

</beans>

(4)测试类
MyTest.java

import com.zql.pojo.People;
import org.junit.Test;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

/**
 * @Author:Daniel
 * @Version 1.0
 */
public class MyTest {

    @Test
    public void test(){

        ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");

        People people = context.getBean("people", People.class);
		System.out.println(people.getName());
        people.getCat().shout();
        people.getDog().shout();
    }
}

在这里插入图片描述

1.2 装配分类

在这里插入图片描述

1.2.1 ByName自动装配

<!--ByName:会自动在容器上下文中查找,和自己对象set方法后面的值对应bean id!-->
<bean id="people" class="com.zql.pojo.People" autowire="byName">
    <property name="name" value="Daniel🤣🤣"/>
</bean>

在这里插入图片描述

1.2.2 ByType自动装配

<bean  class="com.zql.pojo.Cat"></bean>
<bean  class="com.zql.pojo.Dog"></bean>
<!--ByName:会自动在容器上下文中查找,和自己对象set方法后面的值对应bean id!-->
<!--ByType:会自动在容器上下文中查找,和自己对象属性类型相同的 bean!-->
<bean id="people" class="com.zql.pojo.People" autowire="byType">
    <property name="name" value="Daniel🤣🤣"/>
</bean>

在这里插入图片描述

总结:

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

1.2.3 使用注解实现自动装配

官网:使用注解实现自动装配
在这里插入图片描述

  • jdk 1.5 支持的注解,Spring 2.5就支持支持注解了!
  • The introduction of annotation-based configuration raised the question of whether this approach is “better” than XML.

既然使用注解,就要知道:👇🏾👇🏾

  1. 导入约束: context约束
  2. 配置注解的支持:context:annotation-config/
<?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!

科普:

@Nullable 字段标记了这个注解,说明这个字段为null;

注解 @Autowired 底层源码👇🏾👇🏾

public @interface Autowired {
    boolean required() default true;
}

测试代码:

People.java

package com.zql.pojo;

import org.springframework.beans.factory.annotation.Autowired;

import java.io.Serializable;

/**
 * @Author:Daniel
 * @Version 1.0
 */
public class People implements Serializable {

    //如果显示定义了Autowired的required属性为false,说明这个对象可以为null,否则不允许为空
    @Autowired(required = false)
    private Cat cat;
    @Autowired
    private Dog dog;
    private String name;

    public Cat getCat() {
        return cat;
    }

    public Dog getDog() {
        return dog;
    }

    public String getName() {
        return name;
    }

    @Override
    public String toString() {
        return "People{" +
                "cat=" + cat +
                ", dog=" + dog +
                ", name='" + name + '\'' +
                '}';
    }
}

如果 @Autowired自动装配的环境比较复杂,自动装配无法通过一个注解【 @Autowired】完成的时候,我们可以使用 @Qualifier(value=“xxx”)去配置@Autowired的使用,指定一个唯一的bean对象注入!

public class People implements Serializable {

    //如果显示定义了Autowired的required属性为false,说明这个对象可以为null,否则不允许为空
    @Autowired(required = false)
    private Cat cat;
    @Autowired
    @Qualifier(value="dog222") //dog222为beans.xml中id的参数
    private Dog dog;
    private String name;
}

@Resource注解

小结:

public class People implements Serializable {


    @Resource(name="cat2")
    private Cat cat;
    @Resource(name="dog")
    private Dog dog;
    private String name;
}

@Resource 和 @Autowired的区别

  1. 都是用来自动装配的,都可以放在属性字段上。
  2. @Autowired通过byname的方式实现,而且必须要求这个对象存在!【常用】
  3. @Resource默认通过byname的方式实现,如果找不到名字,则通过byType实现!如果两个都找不到的情况下,就报错!
  4. 执行顺序不同: @Autowired通过byType的方式实现。 @Resource默认通过byname的方式实现。

二、Spring使用注解开发

2.1 bean

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

在这里插入图片描述

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

<?xml version="1.0" encoding="UTF-8"?>
<!--suppress ALL -->
<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"
       xmlns:aop="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
        http://www.springframework.org/schema/aop
        https://www.springframework.org/schema/aop/spring-aop.xsd">
        
  		 <!--开启注解的支持-->
        <context:annotation-config/>
</beans>

(1) 创建子工程(spring-05-anno)

在这里插入图片描述

(2) User.java

package com.zql.pojo;

import org.springframework.stereotype.Component;

import java.io.Serializable;

/**
 * @Author:Daniel
 * @Version 1.0
 */
//组件 相当于: <bean id="user" class="com.zql.pojo.User"></bean>
@Component
public class User implements Serializable {

    public String name = "Daniel";
}

(3) applicationContext.xml

<?xml version="1.0" encoding="UTF-8"?>
<!--suppress ALL -->
<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"
       xmlns:aop="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
        http://www.springframework.org/schema/aop
        https://www.springframework.org/schema/aop/spring-aop.xsd">

    <!--指定要扫描的包,这个包下的注解就会生效-->
    <context:component-scan base-package="com.zql.pojo"></context:component-scan>
    <!--开启注解的支持-->
    <context:annotation-config/>
</beans>

(4) MyTest.java

import com.zql.pojo.User;
import org.junit.Test;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

/**
 * @Author:Daniel
 * @Version 1.0
 */
public class MyTest {

    @Test
    public void test(){
        ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");

        User user = context.getBean("user", User.class);

        System.out.println(user.name);
    }
}

在这里插入图片描述

  • @Component:组件,放在类上,说明这个类被Spring管理了,就是bean!

2.2 属性如何注入

User.java

package com.zql.pojo;

import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Component;

import java.io.Serializable;

/**
 * @Author:Daniel
 * @Version 1.0
 */
//组件 相当于: <bean id="user" class="com.zql.pojo.User"></bean>
@Component
public class User implements Serializable {

    @Value("Jenny")  //相当于<property name="name" value="Jenny"></property>
    public String name;
}

在这里插入图片描述

2.3 衍生的注解

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

  • dao【@Repository】
  • service【@Service】
  • controller【@Controller】

在这里插入图片描述

UserDao.java

package com.zql.dao;

import org.springframework.stereotype.Repository;

/**
 * @Author:Daniel
 * @Version 1.0
 */
@Repository
public class UserDao {
}

UserService.java

package com.zql.service;

import org.springframework.stereotype.Service;

/**
 * @Author:Daniel
 * @Version 1.0
 */
@Service
public class UserService {

}

UserController.java

package com.zql.controller;

import org.springframework.stereotype.Controller;

/**
 * @Author:Daniel
 * @Version 1.0
 */
@Controller
public class UserController {
}

applicationContext.xml

<?xml version="1.0" encoding="UTF-8"?>
<!--suppress ALL -->
<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"
       xmlns:aop="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
        http://www.springframework.org/schema/aop
        https://www.springframework.org/schema/aop/spring-aop.xsd">

    <!--指定要扫描的包,这个包下的注解就会生效-->
    <context:component-scan base-package="com.zql"></context:component-scan>
    <!--开启注解的支持-->
    <context:annotation-config/>


    <bean id="user" class="com.zql.pojo.User">
        <property name="name" value="Jenny"></property>
    </bean>
</beans>

在这里插入图片描述

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

2.4 自动装配配置

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

2.5 作用域

package com.zql.pojo;

import org.springframework.beans.factory.annotation.Value;
import org.springframework.context.annotation.Scope;
import org.springframework.stereotype.Component;

import java.io.Serializable;

/**
 * @Author:Daniel
 * @Version 1.0
 */
//组件 相当于: <bean id="user" class="com.zql.pojo.User"></bean>
@Component
@Scope("singleton")  //单例 :singleton  注解开发: prototype
public class User implements Serializable {
    
    public String name;
    @Value("Jenny")  //相当于<property name="name" value="Jenny"></property>
    public void setName(String name) {
        this.name = name;
    }
}

2.6 小结 (xml与注解)

xml与注解:

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

xml与注解最佳实践:

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

注解说明总结:

  • @Autowired :自动装配通过类型。名字
  • 如果Autowired 不能唯一自动装配上属性,则需要通过@Qualifier(value=“xxx”)
  • @Nullable 字段标记了这个注解,说明这个字段可以为null
  • @Resource:自动装配通过名字。类型
  • @Component:组件,放在类上,说明这个类被Spring管理了,就是bean!

三、 使用Java的方式配置Spring(拓展)

  • 我们现在要完全不使用Spring的xml配置了,全权交给Java来做!
  • JavaConfig是Spring的一个子项目,在Spring 4 之后,他成为了一个核心功能!

官网:👉🏾👉🏾 @Configuration

这种配置方式,在后期总结的Springboot中会随处可见!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Daniel521-Spark

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值