当前位置: 首页 > news >正文

十、Spring IoC注解式开发

1 声明Bean的注解


负责声明Bean的注解,常见的包括四个:
  • @Component
  • @Controller
  • @Service
  • @Repository

@Controller、@Service、@Repository这三个注解都是@Component注解的别名。
也就是说:这四个注解的功能都一样。用哪个都可以。

只是为了增强程序的可读性,建议:

  • 控制器类上使用:Controller
  • service类上使用:Service
  • dao类上使用:Repository

他们都是只有一个value属性。value属性用来指定bean的id,也就是bean的名字

2 Spring注解的使用


如何使用以上的注解呢?
  • 第一步:加入aop的依赖

    当加入spring-context依赖之后,会关联加入aop的依赖。

    在这里插入图片描述

  • 第二步:在配置文件中添加context命名空间

    <?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 http://www.springframework.org/schema/beans/spring-beans.xsdhttp://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd">
    </beans>
    
  • 第三步:在配置文件中指定扫描的包

    <!--给spring框架指定要扫描的包-->
    <context:component-scan base-package="com.powernode.spring6.bean"/>
    
  • 第四步:在Bean类上使用注解

    package com.powernode.spring6.bean;import org.springframework.stereotype.Component;@Component("userBean")
    public class User {
    }
    

    以上注解就相当于以下这个配置信息:

    <bean id="userBean" class="com.powernode.spring6.bean.User"/>
    

    测试

    @Testpublic void testBeanComponent(){ApplicationContext applicationContext = new ClassPathXmlApplicationContext("spring.xml");User user = applicationContext.getBean("userBean", User.class);System.out.println(user);}
    

注解的属性名是value,那么value是可以省略的。
如果把value属性全部省略,Bean有默认的名称:Bean类名首字母小写。

如果是多个包有两种解决方案:
第一种:在配置文件中指定多个包,用逗号隔开。
第二种:指定多个包的共同父包。

<!--多个包,使用,隔开-->
<context:component-scan base-package="com.powernode.spring6.bean, com.powernode.spring6.dao"/>
<!--多个包,也可以指定这多个包共同的父包,牺牲一部分效率-->
<context:component-scan base-package="com.powernode.spring6"/>

3 选择性实例化Bean

use-default-filters=“true” 表示:使用spring默认的规则,只要有Component、Controller、Service、Repository中的任意一个注解标注,则进行实例化。

use-default-filters=“false” 表示:不再spring默认实例化规则,即使有Component、Controller、Service、Repository这些注解标注,也不再实例化。

<!--第一种解决方案:use-default-filters="false"如果这个属性是false,表示com.powernode.spring6.bean包下所有的带有声明Bean的注解全部失效。@Component @Controller @Service @Repository全部失效。
-->
<context:component-scan base-package="com.powernode.spring6.bean" use-default-filters="false"><!--包含--><!--只有@Repository @Service被包含进来,生效--><context:include-filter type="annotation" expression="org.springframework.stereotype.Repository"/><context:include-filter type="annotation" expression="org.springframework.stereotype.Service"/>
</context:component-scan><!--第二种解决方案:use-default-filters="true"如果这个属性的值是true,表示com.powernode.spring6.bean下的所有的带有声明Bean的注解全部生效。use-default-filters="true" 默认值就是true,不用写。
-->
<context:component-scan base-package="com.powernode.spring6.bean"><!--排除掉--><!--@Controller失效--><context:exclude-filter type="annotation" expression="org.springframework.stereotype.Controller"/>
</context:component-scan>

4 负责注入的注解


给Bean属性赋值需要用到这些注解:
  • @Value

  • @Autowired

  • @Qualifier

  • @Resource

    1 @Value

    当属性的类型是简单类型时,可以使用@Value注解进行注入。

    使用@Value注解注入的话,可以用在属性上,并且可以不提供setter方法

    // 使用@Value注解注入的话,可以用在属性上,并且可以不提供setter方法
    @Value("com.mysql.cj.jdbc.Driver")
    private String  driver;
    @Value("jdbc:mysql://localhost:3306/spring6")
    private String  url;
    @Value("root")
    private String  username;
    @Value("root")
    private String  password;
    

    @Value注解也可以使用在setter方法上

    private String  driver;
    private String  url;
    private String  username;
    private String  password;
    // @Value注解也可以使用在setter方法上
    @Value("com.mysql.cj.jdbc.Driver")
    public void setDriver(String driver) {this.driver = driver;
    }
    @Value("jdbc:mysql://localhost:3306/spring6")
    public void setUrl(String url) {this.url = url;
    }
    @Value("root")
    public void setUsername(String username) {this.username = username;
    }
    @Value("root")
    public void setPassword(String password) {this.password = password;
    }
    

    @Value注解也可以用在构造方法的形参上

    private String  driver;
    private String  url;
    private String  username;
    private String  password;
    // @Value注解也可以用在构造方法的形参上
    public MyDataSource(@Value("com.mysql.cj.jdbc.Driver") String driver, @Value("jdbc:mysql://localhost:3306/spring6") String url, @Value("root") String username,@Value("root") String password) {this.driver = driver;this.url = url;this.username = username;this.password = password;
    }
    

    2 @Autowired与@Qualifier

    @Autowired注解可以用来注入非简单类型。被翻译为:自动连线的,或者自动装配。
    单独使用@Autowired注解,默认根据类型装配。【默认是byType】

    OrderDao接口

    package org.powernode.dao;public interface OrderDao {void insert();
    }
    

    OrderDao接口的实现类

    package org.powernode.dao.impl;import org.powernode.dao.OrderDao;
    import org.springframework.stereotype.Repository;@Repository("orderDaoImplForMySQL")
    public class OrderDaoImplForMySQL implements OrderDao {@Overridepublic void insert() {System.out.println("MySQL数据库正在保存订单信息....");}
    }
    

    OrderService

    package org.powernode.service;import org.powernode.dao.OrderDao;
    import org.springframework.beans.factory.annotation.Autowired;
    import org.springframework.stereotype.Service;@Service("orderService")
    public class OrderService {// @Autowired注解使用的时候,不需要指定任何属性,直接使用// 根据类型byType进行自动装配@Autowiredprivate OrderDao orderDao;public void generate(){orderDao.insert();}
    }
    

    配置包扫描

    <?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 http://www.springframework.org/schema/beans/spring-beans.xsdhttp://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd"><context:component-scan base-package="org.powernode"/>
    </beans>
    

    测试程序

    @Testpublic void testAutowired(){ApplicationContext applicationContext = new ClassPathXmlApplicationContext("spring-autowired.xml");OrderService orderService = applicationContext.getBean("orderService", OrderService.class);orderService.generate();}
    

    @Autowired注解默认是byType进行注入的,也就是说根据类型注入的,如果以上程序中,UserDao接口如果有多个实现类,会出现错误,错误信息中说:不能装配,UserDao这个Bean的数量大于1.

    怎么解决这个问题?byName,根据名称进行装配了。
    @Autowired注解和@Qualifier注解联合起来才可以根据名称进行装配,在@Qualifier注解中指定Bean名称。

    package org.powernode.service;import org.powernode.dao.OrderDao;
    import org.springframework.beans.factory.annotation.Autowired;
    import org.springframework.beans.factory.annotation.Qualifier;
    import org.springframework.stereotype.Service;@Service("orderService")
    public class OrderService {//  @Autowired和@Qualifier联合使用,可以根据名字自动装配@Autowired@Qualifier("orderDaoImplForMySQL")private OrderDao orderDao;public void generate(){orderDao.insert();}
    }
    


    @Autowired注解可以出现的位置:属性上、构造方法上、构造方法的参数上、setter方法上。

    当带参数的构造方法只有一个,并且构造方法上的参数和属性能够对应上,@Autowired注解可以省略。

    3 @Resource

    @Resource注解也可以完成非简单类型注入。那它和@Autowired注解有什么区别?

    • @Resource注解是JDK扩展包中的,也就是说属于JDK的一部分。所以该注解是标准注解,更加具有通用性。(JSR-250标准中制定的注解类型。JSR是Java规范提案。)
    • @Autowired注解是Spring框架自己的。
    • @Resource注解默认根据名称装配byName,未指定name时,使用属性名作为name。通过name找不到的话会自动启动通过类型byType装配。
    • @Autowired注解默认根据类型装配byType,如果想根据名称装配,需要配合@Qualifier注解一起用。
    • @Resource注解用在属性上、setter方法上。
    • @Autowired注解用在属性上、setter方法上、构造方法上、构造方法参数上。

    引入依赖

    Spring6+版本使用这个依赖

    <dependency><groupId>jakarta.annotation</groupId><artifactId>jakarta.annotation-api</artifactId><version>2.1.1</version>
    </dependency>
    

    StudentDao接口

    package cn.powernode.dao;public interface StudentDao {void deleteById();
    }
    

    StudentDao接口的实现类

    package cn.powernode.dao.impl;import cn.powernode.dao.StudentDao;
    import org.springframework.stereotype.Repository;@Repository("studentDaoImplForMySQL")
    public class StudentDaoImplForMySQL implements StudentDao {@Overridepublic void deleteById() {System.out.println("MySQL正在删除学生信息...");}
    }
    

    StudentService

    package cn.powernode.service;import cn.powernode.dao.StudentDao;
    import jakarta.annotation.Resource;
    import org.springframework.stereotype.Service;@Service("studentService")
    public class StudentService {@Resource(name = "studentDaoImplForMySQL")private StudentDao studentDao;/*@Resource(name = "studentDaoImplForMySQL")public void setStudentDao(StudentDao studentDao) {this.studentDao = studentDao;}*/public void deleteStudent(){studentDao.deleteById();}
    }
    

    配置包扫描

    <?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 http://www.springframework.org/schema/beans/spring-beans.xsdhttp://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd"><context:component-scan base-package="cn.powernode"/>
    </beans>
    

    测试程序:

    @Test
    public void testResource(){ApplicationContext applicationContext = new ClassPathXmlApplicationContext("spring-resource.xml");StudentService studentService = applicationContext.getBean("studentService", StudentService.class);studentService.deleteStudent();}
    

    @Resource注解:默认byName注入,没有指定name时把属性名当做name,根据name找不到时,才会byType注入。byType注入时,某种类型的Bean只能有一个。

5 全注解式开发

写一个配置类来代替配置文件。

package cn.powernode;import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;/*** 编写一个类代替Spring框架的配置文件*/
@Configuration
// 组件扫描
@ComponentScan({"cn.powernode.dao", "cn.powernode.service"})
public class Spring6Config {
}

编写测试程序:不再new ClassPathXmlApplicationContext()对象了。
new AnnotationConfigApplicationContext()对象

@Test
public void testNoXML(){ApplicationContext applicationContext= new AnnotationConfigApplicationContext(Spring6Config.class);StudentService studentService = applicationContext.getBean("studentService", StudentService.class);studentService.deleteStudent();
}
http://www.lryc.cn/news/28806.html

相关文章:

  • Linux系统GPIO应用编程
  • 手敲Mybatis-反射工具天花板
  • Java -数据结构,【优先级队列 / 堆】
  • Python+Qt指纹录入识别考勤系统
  • K_A14_004 基于STM32等单片机驱动旋转角度传感器模块 串口与OLED0.96双显示
  • 2023年全国最新机动车签字授权人精选真题及答案12
  • Linux小黑板(10):信号
  • GO 语言基础语法一 (快速入门 Go 语言)
  • Java高效率复习-SpringMVC[SpringMVC-2]
  • 【前端】一个更底层库-React基础知识点
  • C++ 之枚举类型
  • 软件测试用例(3)
  • Spring——Bean管理-注解方式进行属性注入
  • 【设计模式之美 设计原则与思想:设计原则】20 | 理论六:我为何说KISS、YAGNI原则看似简单,却经常被用错?
  • Java代码弱点与修复之——Copy-paste error(复制粘贴错误)
  • Editor.md 的使用方法及图片处理
  • 剑指 Offer II 018. 有效的回文
  • Elasticsearch分析器(Analyzer)
  • P6入门:了解P6 Professional 工具栏及地图分享
  • 习题30 if elif else 语句
  • 32 openEuler使用LVM管理硬盘-管理卷组
  • Jackson CVE-2017-17485 反序列化漏洞
  • 十大排序(C++版)
  • SpringMVC中的常用注解
  • English Learning - L2-3 英音地道语音语调 小元音 [ʌ] [ɒ] [ʊ] [ɪ] [ə] [e] 2023.02.27 周一
  • fastadmin后台登录页修改
  • Java 面向对象(OOP)的三大特性
  • Java:openjdk: error: Student is abstract; cannot be instantiated;java编译环境
  • 28个案例问题分析---019---临时解决方案和最终解决方案--思想
  • 计算机网络的166个概念你知道几个 第四部分