【Spring】1、Spring 框架的基本使用【读取配置文件、IoC、依赖注入的几种方式、FactoryBean】

本文涉及的产品
云数据库 RDS MySQL,集群系列 2核4GB
推荐场景:
搭建个人博客
RDS MySQL Serverless 基础系列,0.5-2RCU 50GB
RDS MySQL Serverless 高可用系列,价值2615元额度,1个月
简介: 【Spring】1、Spring 框架的基本使用【读取配置文件、IoC、依赖注入的几种方式、FactoryBean】


一、Spring 简介

  • Spring makes Java Simple、modern、productive …

Spring 框架的几个核心概念:

IoC: Inversion of Control:控制反转

DI: Dependency Injection:依赖注入

AOP: Aspect Oriented Programming:面向切面编程

Object Oriented Programming: 面向对象编程

这里使用的 Spring 的版本是:5.2.8.release

🚀耦合:我依赖你,你不见了(不要你了),对我影响很大,我就得改代码

🚀写代码的方向:解耦,降低耦合性

二、读取配置文件、创建对象

①🎄 通过类加载器读取配置文件的输入流

②🎄 通过 Properties 对象的 load() 方法,传入输入流【加载配置文件】

③🎄 通过 Properties 对象的 getProperty(String key) 方法获取到 key 对应的 value

④🎄 使用反射 API,通过全类名创建类的实例

/**
 * 创建对象实例的工厂
 */
public class InstanceFactory {
    // Properties 对象表示【.properties】配置文件
    private static Properties properties;
    static {
        // 获取配置文件的输入流
        try (InputStream is = InstanceFactory.class.getClassLoader().getResourceAsStream("properties.properties")) {
            properties = new Properties();
            // 加载配置文件输入流
            properties.load(is);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
    private static ObjectAble getObjectByName(String key) {
        // 获取配置文件内容
        String fullPath = properties.getProperty(key);
        try {
            Class<?> cls = Class.forName(fullPath);
            return (ObjectAble) cls.newInstance();
        } catch (Exception e) {
            e.printStackTrace();
        }
        return null;
    }
    public static ObjectAble create(String key) {
        return getObjectByName(key);
    }
}

properties.properties 配置文件

personFullPath=com.guoqing.po.PersonV2
dogFullPath=com.guoqing.po.DogV2
boyFullPath=com.guoqing.po.Boy

🎄工厂模式结合配置文件降低类之间的耦合

🎄Spring 的 IoC 就是一个大工厂

三、使用 Spring

(1) 依赖

<dependency>
     <groupId>org.springframework</groupId>
     <artifactId>spring-context</artifactId>
     <version>5.2.8.RELEASE</version>
 </dependency>

(2) Spring 的核心配置文件

  • 按照下图所示创建 applicationContext.xml 文件(Spring 的核心配置文件)

<?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 http://www.springframework.org/schema/beans/spring-beans.xsd">
    <!-- 【bean】所有被 Spring 管理的对象都是 bean -->
    <!-- 配置需要被 Spring 管理的类 -->
    <!-- id: 可通过 id 的值获取到该类的实例 -->
    <!-- class: 该类的全类名 -->
    <bean id="boy" class="com.guoqing.po.Boy" />
    <bean id="personV1" class="com.guoqing.po.PersonV1" />
    <bean id="personV2" class="com.guoqing.po.PersonV2" />
</beans>

(3) 获取 Spring IoC 工厂中的对象实例

类路径的东西:

① java 文件夹中的东西(打包后 classes 中的内容)

② resources 文件夹中的东西

public class QQMain {
    public static void main(String[] args) {
         // 读取 Spring 的核心配置文件, 并得到 Spring 的 IoC 工厂
        ClassPathXmlApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");
        // 通过 IoC 工厂获取实例(boy 是 Spring 的核心配置文件中的 bean 标签的 id 属性配置的内容)
        Boy boy = ctx.getBean("boy", Boy.class);
        System.out.println("boy = " + boy);
    }
}

🍃 读取配置文件,并返回 IoC 容器的类是:ClassPathXmlApplicationContext

🍃 调用 ClassPathXmlApplicationContext 对象的 getBean() 方法,传入 bean 标签的 id 值 获取对象实例

属性的本质含义是:set 方法 后面的值改为小写

setName()name 就是属性】

setBoySchool()boySchool 就是属性】

  • IoC 最重要的作用:解耦合
  • Spring 可以轻松整合日志框架
<dependency>
      <groupId>ch.qos.logback</groupId>
      <artifactId>logback-classic</artifactId>
      <version>1.2.3</version>
  </dependency>

通过 logback-classic 框架可以看到 Spring 相关的日志信息

四、IoC 容器

  • IoC:Inversion of Control【控制反转】
  • 对象创建的控制权转交给了 Spring
  • IoC 容器创建了一系列的 bean
<?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 http://www.springframework.org/schema/beans/spring-beans.xsd">
    <bean id="bookDaoImpl" class="com.guoqing.dao.impl.BookDaoImpl"/>
    <bean id="bookServiceImpl" class="com.guoqing.service.impl.BookServiceImpl">
        <property name="bookDao" ref="bookDaoImpl"/>
    </bean>
    <bean id="bookController" class="com.guoqing.controller.BookController">
        <property name="bookService" ref="bookServiceImpl"/>
    </bean>
</beans>

五、依赖注入(DI)

  • 依赖注入:Dependency Injection

常见的注入类型有3大类:

① bean(自定义类型)【ref 属性

② 基本类型、String、BigDecimal、包装类型【value 属性】

③ 集合类型(数组、Map、List、Set、Properties)☆


常见的注入方式有 2 种:

① 基于 setter(属性)

② 基于 constructor (构造方法)

(1) 基于 setter 注入【bean】

<bean id="cat" class="com.guoqing.domain.Cat">
      <property name="catName" value="哆啦A梦"/>
      <property name="color" value="BLUE"/>
  </bean>
  <bean id="person" class="com.guoqing.domain.Person">
      <property name="cat" ref="cat"/>
  </bean>
<bean id="cat" class="com.guoqing.domain.Cat">
     <property name="catName" value="哆啦A梦"/>
     <property name="color" value="BLUE"/>
 </bean>
 <bean id="person" class="com.guoqing.domain.Person">
     <property name="cat">
         <ref bean="cat"/>
     </property>
 </bean>
<bean id="girl" class="com.guoqing.domain.Girl">
     <property name="cat">
         <bean class="com.guoqing.domain.Cat">
             <property name="catName" value="卡菲"/>
             <property name="color" value="PINK"/>
         </bean>
     </property>
 </bean>

(2) 基于 setter 注入【基本类型、包装类型、String、BigDecimal】

<bean id="person" class="com.guoqing.domain.Person">
     <property name="id" value="666"/>
     <property name="name">
         <value>张国庆</value>
     </property>
     <property name="age" value="8"/>
     <property name="money">
         <value>88888.666</value>
     </property>
 </bean>

(3) 基于 setter 注入【集合类型】

  • 🍃 注入数组
<bean id="whatever" class="com.guoqing.domain.Whatever">
     <property name="books">
         <array>
             <value>Java 编程思想</value>
             <value>红楼梦</value>
             <value>水浒传</value>
             <value>三国演义</value>
             <value>三体</value>
         </array>
     </property>
     <property name="cats">
         <array>
             <bean class="com.guoqing.domain.Cat">
                 <property name="color" value="RED"/>
                 <property name="catName" value="小红"/>
             </bean>
             <bean class="com.guoqing.domain.Cat">
                 <property name="color" value="BLACK"/>
                 <property name="catName" value="小黑"/>
             </bean>
         </array>
     </property>
 </bean>
  • 🍃 注入 List
<bean id="whatever" class="com.guoqing.domain.Whatever">
     <property name="books">
         <!-- ArrayList -->
         <list>
             <value>Thinking In Java</value>
             <value>红楼梦</value>
             <value>水浒传</value>
             <value>三国演义</value>
             <value>三体</value>
         </list>
     </property>
     <property name="cats">
         <list>
             <bean class="com.guoqing.domain.Cat">
                 <property name="color" value="RED"/>
                 <property name="catName" value="小红"/>
             </bean>
             <bean class="com.guoqing.domain.Cat">
                 <property name="color" value="BLACK"/>
                 <property name="catName" value="小黑黑"/>
             </bean>
         </list>
     </property>
 </bean>
  • 🍃 注入 Set

<bean id="whatever" class="com.guoqing.domain.Whatever">
     <property name="goods">
         <!--LinkedHashSet-->
         <set>
             <value>苹果1</value>
             <value>土豆2</value>
             <value>西瓜3</value>
             <value>电脑4</value>
             <value>窗户5</value>
             <value>窗户5</value>
             <value>窗户5</value>
             <value>鼠标6</value>
         </set>
     </property>
 </bean>

HashSet 是无顺序的【可去重】

LinkedHashSet 是有顺序的【可去重】

  • 🍃 注入 Map

<bean id="whatever" class="com.guoqing.domain.Whatever">
     <property name="peoSal">
         <map>
             <entry key="张国庆" value="88888"/>
             <entry key="周杰伦" value="155555"/>
             <entry key="王凡" value="266666"/>
             <entry key="陈铭酒" value="12121"/>
             <entry key="刘德华" value="888888"/>
         </map>
     </property>
     <property name="peoCatMap">
         <map>
             <entry key="ZGQ">
                 <bean class="com.guoqing.domain.Cat">
                     <property name="catName" value="猫猫1"/>
                     <property name="color" value="RED"/>
                 </bean>
             </entry>
             <entry key="CMJ">
                 <bean class="com.guoqing.domain.Cat">
                     <property name="catName" value="猫猫2"/>
                     <property name="color" value="PINK"/>
                 </bean>
             </entry>
         </map>
     </property>
 </bean>
  • 🍃 注入 Properties
<bean id="whatever" class="com.guoqing.domain.Whatever">
     <property name="properties">
         <props>
             <prop key="dao">com.guoqing.dao.impl.BookDaoImpl</prop>
             <prop key="service">com.guoqing.service.impl.BookServiceImpl</prop>
         </props>
     </property>
 </bean>

基于 setter 注入调用的是无参的构造方法

(4) 基于 setter 注入(命名空间)

没有命名空间的写法:

<bean id="cat" class="com.guoqing.domain.Cat">
     <property name="color" value="skyblue"/>
     <property name="catName" value="阿辉"/>
 </bean>
 <bean id="teacher" class="com.guoqing.domain.Teacher">
     <property name="id" value="2"/>
     <property name="name" value="李老师"/>
     <property name="salary" value="12121.666"/>
     <property name="gender" value="男"/>
     <property name="cat" ref="cat"/>
 </bean>

命名空间写法:

增加命名空间:xmlns:p="http://www.springframework.org/schema/p"

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:p="http://www.springframework.org/schema/p"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">
       
    <bean id="cat" class="com.guoqing.domain.Cat" p:catName="阿辉辉" p:color="SKYBLUE"/>
    <bean id="teacher"
          p:name="张国庆老师"
          p:id="22"
          p:salary="6666.666"
          p:gender="男"
          p:cat-ref="cat"
          class="com.guoqing.domain.Teacher"/>
</beans>

(5) 基于构造方法注入

  • 基本类型
<!--调用没有参数的构造方法-->
<bean id="student" class="com.guoqing.domain.Student"/>
<bean id="student" class="com.guoqing.domain.Student">
     <!--调用有一个参数的构造方法-->
     <constructor-arg value="张国庆"/>
 </bean>
<bean id="student" class="com.guoqing.domain.Student">
     <!--调用有2个参数的构造方法-->
     <constructor-arg value="张国庆"/>
     <constructor-arg value="99.5"/>
 </bean>
<bean id="student" class="com.guoqing.domain.Student">
     <!--假如顺序和构造方法参数的顺序不一样的话, 可以指定 type-->
     <!--若不顺序不一样, 且不做任何处理, 必定报错-->
     <constructor-arg value="99.5" type="double"/>
     <constructor-arg value="张国庆" type="java.lang.String"/>
 </bean>
<bean id="student" class="com.guoqing.domain.Student">
     <!--假如顺序和构造方法参数的顺序不一样的话, 可以指定 index-->
     <!--若不顺序不一样, 且不做任何处理, 必定报错-->
     <constructor-arg value="99.5" index="1"/>
     <constructor-arg value="张国庆" index="0"/>
 </bean>

  • bean 注入

<bean id="student" class="com.guoqing.domain.Student">
     <constructor-arg value="张国庆" index="0"/>
     <constructor-arg>
         <bean class="com.guoqing.domain.Cat">
             <property name="catName" value="花儿"/>
             <property name="color" value="ORANGE"/>
         </bean>
     </constructor-arg>
 </bean>
<bean id="cat" class="com.guoqing.domain.Cat">
     <property name="catName" value="流浪猫"/>
     <property name="color" value="GRAY"/>
 </bean>
 <bean id="student" class="com.guoqing.domain.Student">
     <constructor-arg value="张国庆" index="0"/>
     <constructor-arg ref="cat"/>
 </bean>
<bean id="cat" class="com.guoqing.domain.Cat">
     <property name="catName" value="流浪猫"/>
     <property name="color" value="GRAY"/>
 </bean>
 <bean id="student" class="com.guoqing.domain.Student">
     <constructor-arg value="张国庆" index="0"/>
     <constructor-arg>
         <ref bean="cat"/>
     </constructor-arg>
 </bean>

  • 注入集合
<bean id="student" class="com.guoqing.domain.Student">
     <constructor-arg type="java.util.List">
         <list>
             <value>语文</value>
             <value>数学</value>
             <value>英语</value>
             <value>物理</value>
             <value>化学</value>
         </list>
     </constructor-arg>
 </bean>

六、创建过程比较复杂的对象

🥄 Spring 的配置文件中配置的对象只需要执行某些构造方法即可创建

🥄 当遇到创建过程极其复杂的对象(无法通过调用构造方法创建的对象)时就需要下面的方式了

🥄 如:JDBC 的 Connection 对象的创建

<dependency>
      <groupId>mysql</groupId>
      <artifactId>mysql-connector-java</artifactId>
      <version>8.0.19</version>
  </dependency>

(1) 静态工厂(调用静态方法)

public class JdbcConnectionFactory {
    /**
     * 静态工厂
     */
    public static Connection getConnection() throws Exception {
        Class.forName("com.mysql.cj.jdbc.Driver");
        return DriverManager.getConnection("jdbc:mysql://localhost:3306/jump_zgq?serverTimezone=UTC", "root", "root");
    }
}
<bean id="jdbcConnection"
       class="com.guoqing.factory.JdbcConnectionFactory"
       factory-method="getConnection"/>

🎀把创建复杂对象代码写在工厂中

🎀通过该工厂中的 static 方法返回对象的实例

🎀在 Spring 的核心配置文件中,通过 factory-method 属性指定需要调用该工厂类的哪个方法返回对象实例

🎀【缺点】无法传参

(2) 实例工厂

public class JdbcConnectionFactory {
    private String driverClass;
    private String url;
    private String user;
    private String pwd;
    public JdbcConnectionFactory(String driverClass, String url, String user, String pwd) {
        this.driverClass = driverClass;
        this.url = url;
        this.user = user;
        this.pwd = pwd;
    }
    /**
     * 实例工厂
     */
    public Connection getConnection() throws Exception {
        System.out.println("调用了 getConnection() 方法");
        Class.forName(this.driverClass);
        return DriverManager.getConnection(url, user, pwd);
    }
}
<!--创建工厂实例-->
  <bean id="jdbcConnectionFactory" class="com.guoqing.factory.JdbcConnectionFactory">
      <constructor-arg name="driverClass" value="com.mysql.cj.jdbc.Driver"/>
      <constructor-arg name="url" value="jdbc:mysql://localhost:3306/jump_zgq?serverTimezone=UTC"/>
      <constructor-arg name="user" value="root"/>
      <constructor-arg name="pwd" value="root"/>
  </bean>
  <!--通过工厂实例创建 Connection 对象-->
  <bean id="jdbcConnection" factory-bean="jdbcConnectionFactory" factory-method="getConnection"/>

🎉 factory-bean 属性指代工厂类的实例

🎉 factory-method 属性指代要调用工厂对象的哪个实例方法来返回该复杂对象

(3) FactoryBean

🎁 Spring 提供的、用于方便地创建复杂对象的接口

🎁 本质和【实例工厂】类似

public class JdbcConnectionFactoryBean implements FactoryBean<Connection> {
    private String driverClass;
    private String url;
    private String user;
    private String pwd;
    public JdbcConnectionFactoryBean(String driverClass, String url, String user, String pwd) {
        this.driverClass = driverClass;
        this.url = url;
        this.user = user;
        this.pwd = pwd;
    }
    /**
     * 写创建该复杂对象的一系列代码
     */
    @Override
    public Connection getObject() throws Exception {
        System.out.println("调用了 getObject() 方法");
        Class.forName(this.driverClass);
        return DriverManager.getConnection(url, user, pwd);
    }
    @Override
    public Class<?> getObjectType() {
        return Connection.class;
    }
}
<bean id="jdbcConnection" class="com.guoqing.factoryBean.JdbcConnectionFactoryBean">
     <constructor-arg name="driverClass" value="com.mysql.cj.jdbc.Driver"/>
     <constructor-arg name="url" value="jdbc:mysql://localhost:3306/jump_zgq?serverTimezone=UTC"/>
     <constructor-arg name="user" value="root"/>
     <constructor-arg name="pwd" value="root"/>
 </bean>

🍃 监听 class 属性后的是否实现(implements)了FactoryBean 接口

🍃 若实现了 FactoryBean 接口,则调用该getObject() 方法,然后将该方法返回的实例放入 IoC 容器中

若就是想拿到该类的实例,可通过 & 符号拼接上 id 属性的值,然后通过 getBean() 方法获取

public class QQMain {
    public static void main(String[] args) {
        ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");
        
        Object bean = ctx.getBean("&jdbcConnection");
        // bean = com.guoqing.factoryBean.JdbcConnectionFactoryBean@436e852b
        System.out.println("bean = " + bean);
    }
}

若文章有错误请不吝赐教

相关实践学习
如何在云端创建MySQL数据库
开始实验后,系统会自动创建一台自建MySQL的 源数据库 ECS 实例和一台 目标数据库 RDS。
全面了解阿里云能为你做什么
阿里云在全球各地部署高效节能的绿色数据中心,利用清洁计算为万物互联的新世界提供源源不断的能源动力,目前开服的区域包括中国(华北、华东、华南、香港)、新加坡、美国(美东、美西)、欧洲、中东、澳大利亚、日本。目前阿里云的产品涵盖弹性计算、数据库、存储与CDN、分析与搜索、云通信、网络、管理与监控、应用服务、互联网中间件、移动服务、视频服务等。通过本课程,来了解阿里云能够为你的业务带来哪些帮助 &nbsp; &nbsp; 相关的阿里云产品:云服务器ECS 云服务器 ECS(Elastic Compute Service)是一种弹性可伸缩的计算服务,助您降低 IT 成本,提升运维效率,使您更专注于核心业务创新。产品详情: https://www.aliyun.com/product/ecs
相关文章
|
21天前
|
数据采集 监控 前端开发
二级公立医院绩效考核系统源码,B/S架构,前后端分别基于Spring Boot和Avue框架
医院绩效管理系统通过与HIS系统的无缝对接,实现数据网络化采集、评价结果透明化管理及奖金分配自动化生成。系统涵盖科室和个人绩效考核、医疗质量考核、数据采集、绩效工资核算、收支核算、工作量统计、单项奖惩等功能,提升绩效评估的全面性、准确性和公正性。技术栈采用B/S架构,前后端分别基于Spring Boot和Avue框架。
|
17天前
|
JavaScript 安全 Java
如何使用 Spring Boot 和 Ant Design Pro Vue 构建一个前后端分离的应用框架,实现动态路由和菜单功能
本文介绍了如何使用 Spring Boot 和 Ant Design Pro Vue 构建一个前后端分离的应用框架,实现动态路由和菜单功能。首先,确保开发环境已安装必要的工具,然后创建并配置 Spring Boot 项目,包括添加依赖和配置 Spring Security。接着,创建后端 API 和前端项目,配置动态路由和菜单。最后,运行项目并分享实践心得,帮助开发者提高开发效率和应用的可维护性。
35 2
|
16天前
|
消息中间件 NoSQL Java
springboot整合常用中间件框架案例
该项目是Spring Boot集成整合案例,涵盖多种中间件的使用示例,每个案例项目使用最小依赖,便于直接应用到自己的项目中。包括MyBatis、Redis、MongoDB、MQ、ES等的整合示例。
67 1
|
19天前
|
XML 缓存 Java
搞透 IOC、Spring IOC ,看这篇就够了!
本文详细解析了Spring框架的核心内容——IOC(控制反转)及其依赖注入(DI)的实现原理,帮助读者理解如何通过IOC实现组件解耦,提高程序的灵活性和可维护性。关注【mikechen的互联网架构】,10年+BAT架构经验倾囊相授。
|
11天前
|
安全 Java 测试技术
Java开发必读,谈谈对Spring IOC与AOP的理解
Spring的IOC和AOP机制通过依赖注入和横切关注点的分离,大大提高了代码的模块化和可维护性。IOC使得对象的创建和管理变得灵活可控,降低了对象之间的耦合度;AOP则通过动态代理机制实现了横切关注点的集中管理,减少了重复代码。理解和掌握这两个核心概念,是高效使用Spring框架的关键。希望本文对你深入理解Spring的IOC和AOP有所帮助。
24 0
|
Java Spring
spring框架之AOP模块(面向切面),附带通知类型---超详细介绍
spring框架之AOP模块(面向切面),附带通知类型---超详细介绍
122 0
|
缓存 监控 Java
Spring框架之AOP(面向切面编程)
Spring框架之AOP(面向切面编程)
58 0
|
4月前
|
分布式计算 Java MaxCompute
详解 Java 限流接口实现问题之在Spring框架中使用AOP来实现基于注解的限流问题如何解决
详解 Java 限流接口实现问题之在Spring框架中使用AOP来实现基于注解的限流问题如何解决
|
5月前
|
设计模式 SQL Java
Spring框架第四章(AOP概念及相关术语)
Spring框架第四章(AOP概念及相关术语)
|
6月前
|
安全 Java 开发者
在Spring框架中,IoC和AOP是如何实现的?
【4月更文挑战第30天】在Spring框架中,IoC和AOP是如何实现的?
81 0