spring5:IOC 操作 Bean 管理(基于注解方式)

Posted _GGBond_

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了spring5:IOC 操作 Bean 管理(基于注解方式)相关的知识,希望对你有一定的参考价值。

IOC操作Bean管理(基于xml方式)



前言

本博主将用CSDN记录软件开发求学之路上亲身所得与所学的心得与知识,有兴趣的小伙伴可以关注博主!也许一个人独行,可以走的很快,但是一群人结伴而行,才能走的更远!让我们在成长的道路上互相学习,欢迎关注!

一、注解

1、概述

注解是代码特殊标记,
格式:@注解名称(属性名称=属性值, 属性名称=属性值..)

使用注解,注解作用在类上面,方法上面,属性上面

使用注解目的:简化 xml 配置

XML配置文件一样,注解本身并不能执行,注解本身仅仅只是做一个标记,具体的功能是框架检测到注解标记的位置,然后针对这个位置按照注解标记的功能来执行具体操作。
本质上:所有一切的操作都是Java代码来完成的,XML和注解只是告诉框架中的Java代码如何执行。

二、入门案例

1、Bean 的创建

  1. @Component:将类标识为普通组件
  2. @Controller:将类标识为控制层组件
  3. @Service:将类标识为业务层组件
  4. @Repository:将类标识为持久层组件

注意:虽然它们本质上一样,但是为了代码的可读性,为了程序结构严谨我们肯定不能随便胡乱标记。

创建控制层组件

@Controller
public class UserController 

创建接口UserService

public interface UserService 

创建业务层组件UserServiceImpl

@Service
public class UserServiceImpl implements UserService 

创建接口UserDao

public interface UserDao 

创建持久层组件UserDaoImpl

@Repository
public class UserDaoImpl implements UserDao 

2、Bean的自动装配

2.1 @Autowired

⭕ 概述

根据属性类型进行自动装配,在成员变量上直接标记@Autowired注解即可完成自动装配,不需要提供setXxx()方法。以后我们在项目中的正式用法就是这样。

⭕ 例子
UserDao接口

public interface UserDao 
    public void add();

UserDaoImpl类

@Repository
public class UserDaoImpl implements com.ir.dao.UserDao 
    @Override
    public void add() 
        System.out.println("dao add.....");
    

UserService类

@Service
public class UserService 
    //定义 dao 类型属性
    //不需要添加 set 方法
    //添加注入属性注解
    @Autowired
    private UserDao userdao;
        public void add() 
            System.out.println("service add.......");
            userdao.add();
        


测试;

@Test
    public void test()
        ApplicationContext context = new ClassPathXmlApplicationContext("bean1.xml");
        UserService userService1 =  context.getBean("userService",UserService.class);
        System.out.println(userService1);//com.ir.service.UserService@8909f18
        userService1.add();//service add.......  dao add.....
    

⭕ 注意点

@Autowired注解可以标记在构造器和set方法上

@Controller
public class UserController 
	private UserService userService;
	@Autowired
	public UserController(UserService userService)
		this.userService = userService;
	
	public void saveUser()
		userService.saveUser();
	

@Controller
public class UserController 
	private UserService userService;
	@Autowired
	public void setUserService(UserService userService)
		this.userService = userService;
	
	userService.saveUser();
	

⭕ @Autowired工作流程

@Controller
public class UserController 
	@Autowired
	@Qualifier("userServiceImpl")
	private UserService userService;
	public void saveUser()
		userService.saveUser();
	

@Autowired中有属性required,默认值为true,因此在自动装配无法找到相应的bean时,会装配失败

可以将属性required的值设置为true,则表示能装就装,装不上就不装,此时自动装配的属性为默认值
但是实际开发时,基本上所有需要装配组件的地方都是必须装配的,用不上这个属性。

2、@Qualifie

根据名称进行注入
@Qualifier 注解的使用,和上面@Autowired 一起使用

UserService类

@Service
public class UserService 
    //定义 dao 类型属性
    //不需要添加 set 方法
    //添加注入属性注解
    @Autowired
    @Qualifier(value = "userDaoImpl1") //根据名称进行注入
    private UserDao userdao;
        public void add() 
            System.out.println("service add.......");
            userdao.add();
        

UserDaoImp类

@Repository(value = "userDaoImpl1")
public class UserDaoImpl implements com.ir.dao.UserDao 
    @Override
    public void add() 
        System.out.println("dao add.....");
    

测试

   @Test
    public void test3()
        ApplicationContext context = new ClassPathXmlApplicationContext("bean1.xml");
        UserService userService1 =  context.getBean("userService",UserService.class);
        System.out.println(userService1);//com.ir.service.UserService@7fc229ab
        userService1.add();//service add.......  dao add.....
    

3、@Resource

可以根据类型注入,可以根据名称注入

 <context:component-scan base-package="com.ir"></context:component-scan>

UserDaoImpl

@Repository(value = "userDaoImpl2")
public class UserDaoImpl implements com.ir.dao.UserDao 
    @Override
    public void add() 
        System.out.println("dao add.....");
    

UserService


@Service
public class UserService 
    //定义 dao 类型属性
    //不需要添加 set 方法
    //添加注入属性注解
    @Resource(name = "userDaoImpl2")
    private UserDao userdao;
        public void add() 
            System.out.println("service add.......");
            userdao.add();
        

测试:

@Test
    public void test4()
        ApplicationContext context = new ClassPathXmlApplicationContext("bean1.xml");
        UserService userService1 =  context.getBean("userService",UserService.class);
        System.out.println(userService1);//com.ir.service.UserService@15761df8
        userService1.add();//service add.......  dao add.....
    

UserService

@Service
public class UserService 
    //定义 dao 类型属性
    //不需要添加 set 方法
    //添加注入属性注解
    @Resource//无添加属性表示此时该注解是通过【属性类型】来注入
    private UserDao userdao;
        public void add() 
            System.out.println("service add.......");
            userdao.add();
        

UserDaoImpl

@Repository(value = "userDaoImpl1")
public class UserDaoImpl implements com.ir.dao.UserDao 
    @Override
    public void add() 
        System.out.println("dao add.....");
    

测试:

   @Test
    public void test()
        ApplicationContext context = new ClassPathXmlApplicationContext("bean1.xml");
        UserService userService1 =  context.getBean("userService",UserService.class);
        System.out.println(userService1);//com.ir.service.UserService@15761df8
        userService1.add();//service add.......  dao add.....
    

4、@Value

注入普通类型属性

UserService类

@Service
public class UserService 
    //定义 dao 类型属性
    //不需要添加 set 方法
    //添加注入属性注解
    @Resource//无添加属性表示此时该注解是通过【属性类型】来注入
    private UserDao userdao;
    
    @Value(value = "abc")
    private String name;
        public void add() 
            System.out.println("service add.......");
            userdao.add();
        

测试

    @Test
    public void test5()
        ApplicationContext context = new ClassPathXmlApplicationContext("bean1.xml");
        UserService userService =  context.getBean("userService",UserService.class);
        System.out.println(userService);//com.ir.service.UserService@add0edd
        userService.add();//service add.......          dao add.....
        System.out.println(userService.name);//abc
    

3、扫描组件

3.1 配置文件版

情况一:最基本的扫描方式

<context:component-scan base-package="com.atguigu">
</context:component-scan>

情况一:最基本的扫描方式

<context:component-scan base-package="com.atguigu">
</context:component-scan>

情况二:指定要排除的组件

<context:component-scan base-package="com.atguigu">
<!-- context:exclude-filter标签:指定排除规则 -->
<!--
type:设置排除或包含的依据
type="annotation",根据注解排除,expression中设置要排除的注解的全类名
type="assignable",根据类型排除,expression中设置要排除的类型的全类名
-->
<context:exclude-filter type="annotation"
expression="org.springframework.stereotype.Controller"/>
<!--<context:exclude-filter type="assignable"
expression="com.atguigu.controller.UserController"/>-->
</context:component-scan>

情况三:仅扫描指定组件

<context:component-scan base-package="com.atguigu" use-default-filters="false">
<!-- context:include-filter标签:指定在原有扫描规则的基础上追加的规则 -->
<!-- use-default-filters属性:取值false表示关闭默认扫描规则 -->
<!-- 此时必须设置use-default-filters="false",因为默认规则即扫描指定包下所有类 -->
<!--
type:设置排除或包含的依据
type="annotation",根据注解排除,expression中设置要排除的注解的全类名
type="assignable",根据类型排除,expression中设置要排除的类型的全类名
-->
<context:include-filter type="annotation"
expression="org.springframework.stereotype.Controller"/>
<!--<context:include-filter type="assignable"
expression="com.atguigu.controller.UserController"/>-->
</context:component-scan>

3.2 注解版

创建配置类,替代 xml 配置文件

@Configuration  //作为配置类,替代xml配置文件
@ComponentScan(basePackages = "com.ir")//扫描包
public class SpringConfig 


4、测试

 @Test
    public void test1()
        ApplicationContext context = new ClassPathXmlApplicationContext("bean1.xml");
        UserService userService1 = (UserService) context.getBean("userService1");
        System.out.println(userService1);//com.ir.service.UserService@7920ba90
        userService1.add();//service add.......
    

Spring5学习笔记 — “IOC操作Bean管理(基于注解)”

IOC 操作 Bean 管理(基于注解方式)


​ 一、什么是注解

​ (1)注解是代码特殊标记

格式:@注解名称(属性名称=属性值, 属性名称=属性值…)

​ (2)使用注解,注解作用在类上面,方法上面,属性上面

​ (3)使用注解目的:简化 xml 配置

​ 二、Spring 针对 Bean 管理操作提供注解

​ 1. 下面四个注解功能相同,都可以用来创建 bean 实例

  • @Component
  • @Service
  • @Controller
  • @Repository

2. 下面四个注解可以用来为对象注入属性

  • @Autowired:根据属性类型注入
  • @Qualifier:根据属性名称注入
  • @Resource:可以根据属性类型注入,也可以根据属性名称注入
  • @Value:注入普通类型属性

前三个都是注入对象类型的属性

三、IOC操作Bean管理(基于注解)

1. 基于注解创建对象

第一步:导入jar包(可直接通过idea将Spring相关jar包自动导入项目)

第二步:开启组件扫描(在Spring配置文件中)
通过<context:component-scan> </context:component-scan>实现

<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 http://www.springframework.org/schema/beans/spring-beans.xsd
                           http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd">
        <!--上方引入context名称空间(第3、5行)-->

    <!--开启组件扫描
        如要扫描多个同包类:
        1.base-package属性中的多个包类路径,用逗号隔开
        2.base-package属性值直接写这几个类的共同“上层包”的名
    -->
    <context:component-scan base-package="demo1"></context:component-scan>
</beans>

第三步:创建Book类并使用注解创建对象

import org.springframework.stereotype.Component;

/*
可以直接写@Component
即value值省略不写
(默认值为类名,首字母小写,例如:book)
*/
@Component("book") //注解等同于配置文件中<bean id="book" class="...">
public class Book 
    public void test()
        System.out.println("book test............");
    

第四步,编写测试类

public class bookTest 
    public static void main(String[] args) 
        ApplicationContext context = 
        new ClassPathXmlApplicationContext("demo1/bean1.xml");
        Book book = context.getBean("book",Book.class);
        book.test();
    

小细节:开启组件扫描细节配置(能看懂就行)

<!--示例 1
 use-default-filters="false" 表示现在不使用默认 filter,自己配置 filter
 context:include-filter ,设置扫描哪些内容
-->
<context:component-scan base-package="com.atguigu" use-defaultfilters="false">
 <context:include-filter type="annotation" 
 expression="org.springframework.stereotype.Controller"/><!--代表只扫描Controller注解的类-->
</context:component-scan>

<!--示例 2
 下面配置扫描包所有内容
 context:exclude-filter: 设置哪些内容不进行扫描
-->
<context:component-scan base-package="com.atguigu">
 <context:exclude-filter type="annotation" 
 expression="org.springframework.stereotype.Controller"/><!--表示Controller注解的类之外一切都进行扫描-->
</context:component-scan>

2. 基于注解注入属性


先把演示demo创建一下:

1.先创建一个UserDao接口,及其实现类UserDaoimpl_1并创建对象

//1.UserDao接口
public interface UserDao 
    public void add();

//2.UserDao接口实现类 UserDaoimpl_1   

@Component
public class UserDaoimpl_1 implements UserDao
    public void add()
        System.out.println("第一个实现类");
    

2.创建UserService类,并创建其对象

//3.创建UserService类
import org.springframework.stereotype.Component;

@Component
public class UserService 
   
    public void test()
        System.out.println("Userservice.......");
        dao.add();
    

3.编写Spring配置文件(代码省略)…


(1)@Autowired:根据属性类型注入属性

import org.springframework.stereotype.Component;

@Component
public class UserService 

    //通过@Autowired添加一个UserDao类型的属性
    //不需要添加set方法
    
    @Autowired
    private UserDao dao;

    public void test()
        System.out.println("Userservice.......");
        dao.add();
    

运行结果,可见引入的属性为第一个实现类的对象

(2)@Qualifier:根据名称进行注入

当我们的UserDao接口有两个实现类的时候,如果还是只用@Autowired注解,系统会不知道该引入哪一个实现类而报错,因此我们此时要使用@Qualifier注解来通过类的名称来注入属性

@Qualifier 注解要和和上面@Autowired 一起使用

第一步,我们再创建一个UserDao的实现类UserDaoimpl_2

@Component
public class UserDaoimpl_2 implements UserDao
    public void add()
        System.out.println("第二个实现类");
    

(此时demo结构为如下)

第二步,重新在UserService中引入属性

@Component
public class UserService 
    //添加一个UserDao类型的属性
    //不需要添加set方法
    //然后在属性上面添加注解
    //注意:value的值必须为被引入类的“对象名”
    
    @Autowired
    @Qualifier(value = "userDaoimpl_2")
    private UserDao dao;

    public void test()
        System.out.println("Userservice.......");
        dao.add();
    

运行结果,可见引入的属性为第二个实现类的对象

(3)@Resource:可以根据类型注入,也可以根据名称注入

//@Resource //根据类型进行注入
@Resource(name = "userDaoImpl1") //根据名称进行注入
private UserDao userDao;

(4)@Value:注入普通类型属性

@Value(value = "abc")
private String name;

3. 完全注解开发(不需要Spring配置文件,全部使用注解)

(1)创建配置类,替代 xml 配置文件

//1.设置该类作为配置类,替代 xml 配置文件
@Configuration 
//2.开启组件扫描(此处demo3为包名)
@ComponentScan(basePackages = "demo3") 
public class SpringConfig 

(2)编写测试类

public class testDemo 
    public static void main(String[] args) 
        //1. 加载配置类,此时选择AnnotationConfigApplicationContext()实现类
        //参数为配置类的类名 注意!!!参数不要加引号!!!
        ApplicationContext context =
                new AnnotationConfigApplicationContext(SpringConfig.class);

        UserService userService = context.getBean("userService",UserService.class);
        userService.test();
    

注意使用配置类时,context对象通过“AnnotationConfigApplicationContext()”来获取,
参数为“配置类名.class”
(参数绝对不要加双引号!!!会报错)

以上是关于spring5:IOC 操作 Bean 管理(基于注解方式)的主要内容,如果未能解决你的问题,请参考以下文章

spring5:IOC 操作 Bean 管理(基于注解方式)

Spring5学习笔记 — “IOC操作Bean管理(基于注解)”

Spring5学习笔记 — “IOC操作Bean管理(基于注解)”

[Spring5]IOC容器_Bean管理XML方式_创建对象_set注入属性and有参构造注入属性

Spring5学习笔记 — “Spring IOC(控制反转)”

Spring5 框架基本概念