【JAVA届的一代神】——Spring框架体系及Spring IOC思想【面试常问!】

news2024/11/26 18:51:41

文章目录

    • Spring简介
    • Spring体系结构
    • SpringIOC
      • 控制反转思想
      • 自定义对象容器
      • Spring实现IOC
      • Spring容器类型
        • 容器接口
        • 容器实现类
      • 对象的创建方式
        • 使用构造方法
        • 使用工厂类的方法
        • 使用工厂类的静态方法
      • 对象的创建策略
      • 对象的销毁时机
      • 生命周期方法
      • 获取Bean对象的方式
        • 通过id/name获取
        • 通过类型获取
        • 通过类型+id/name获取

Spring简介

在这里插入图片描述

Spring是一个开源框架,为简化企业级开发而生。它以IOC(控制反转)和AOP(面向切面)为思想内核,提供了控制层SpringMVC、数据层SpringData、服务层事务管理等众多技术,并可以整合众多第三方框架。

Spring将很多复杂的代码变得优雅简洁,有效的降低代码的耦合度,极大的方便项目的后期维护、升级和扩展。

Spring官网地址:https://spring.io/

Spring体系结构

在这里插入图片描述

Spring框架根据不同的功能被划分成了多个模块,这些模块可以满足一切企业级应用开发的需求,在开发过程中可以根据需求有选择性地使用所需要的模块。

  • Core Container:Spring核心模块,任何功能的使用都离不开该模块,是其他模块建立的基础。
  • Data Access/Integration:该模块提供了数据持久化的相应功能。
  • Web:该模块提供了web开发的相应功能。
  • AOP:提供了面向切面编程实现
  • Aspects:提供与AspectJ框架的集成,该框架是一个面向切面编程框架。
  • Instrumentation:提供了类工具的支持和类加载器的实现,可以在特定的应用服务器中使用。
  • Messaging:为Spring框架集成一些基础的报文传送应用
  • Test:提供与测试框架的集成

SpringIOC

在这里插入图片描述

控制反转思想

IOC(Inversion of Control) :程序将创建对象的权利交给框架。

之前在开发过程中,对象实例的创建是由调用者管理的,代码如下:

public interface StudentDao {
  // 根据id查询学生
  Student findById(int id);
}


public class StudentDaoImpl implements StudentDao{
  @Override
  public Student findById(int id) {
    // 模拟从数据库查找出学生
    return new Student(1,"张三","北京");
   }
}


public class StudentService {
  public Student findStudentById(int id){
    // 此处就是调用者在创建对象
    StudentDao studentDao = new StudentDaoImpl();
    return studentDao.findById(1);
   }
}

这种写法有两个缺点:

  1. 浪费资源:StudentService调用方法时即会创建一个对象,如果不断调用方法则会创建大量StudentDao对象。
  2. 代码耦合度高:假设随着开发,我们创建了StudentDao另一个更加完善的实现类StudentDaoImpl2,如果在StudentService中想使用StudentDaoImpl2,则必须修改源码。

而IOC思想是将创建对象的权利交给框架,框架会帮助我们创建对象,分配对象的使用,控制权由程序代码转移到了框架中,控制权发生了反转,这就是Spring的IOC思想。而IOC思想可以完美的解决以上两个问题。

自定义对象容器

接下来我们通过一段代码模拟IOC思想。创建一个集合容器,先将对象创建出来放到容器中,需要使用对象时,只需要从容器中获取对象即可,而不需要重新创建,此时容器就是对象的管理者。

  1. 创建实体类

    public class Student {
      private int id;
      private String name;
      private String address;
        // 省略getter/setter/构造方法/tostring
    }
    
  2. 创建Dao接口和实现类

    public interface StudentDao {
      // 根据id查询学生
      Student findById(int id);
    }
    
    
    public class StudentDaoImpl implements StudentDao{
      @Override
      public Student findById(int id) {
        // 模拟从数据库查找出学生
        return new Student(1,"张三","北京");
       }
    }
    
    
    public class StudentDaoImpl2 implements StudentDao{
      @Override
      public Student findById(int id) {
        // 模拟根据id查询学生
        System.out.println("新方法!!!");
        return new Student(1,"张三","北京");
       }
    }
    
  3. 创建配置文件bean.properties,该文件中定义管理的对象

    studentDao=com.Spring.dao.StudentDaoImpl
    
  4. 创建容器管理类,该类在类加载时读取配置文件,将配置文件中配置的对象全部创建并放入容器中。

    public class Container {
      static Map<String,Object> map = new HashMap();
    
    
      static {
        // 读取配置文件
        InputStream is = Container.class.getClassLoader().getResourceAsStream("bean.properties");
        Properties properties = new Properties();
        try {
          properties.load(is);
         } catch (IOException e) {
          e.printStackTrace();
         }
    
    
        // 遍历配置文件的所有配置
        Enumeration<Object> keys = properties.keys();
        while (keys.hasMoreElements()){
          String key = keys.nextElement().toString();
          String value = properties.getProperty(key);
          try {
            // 创建对象
            Object o = Class.forName(value).newInstance();
            // 将对象放入集合中
            map.put(key,o);
           } catch (Exception e) {
            e.printStackTrace();
           }
         }
       }
    
    
      // 从容器中获取对象
      public static Object getBean(String key){
        return map.get(key);
       }
    }
    
  5. 创建Dao对象的调用者StudentService

    public class StudentService {
      public Student findStudentById(int id){
        // 从容器中获取对象
        StudentDao studentDao = (StudentDao) Container.getBean("studentDao");
        System.out.println(studentDao.hashCode());
        return studentDao.findById(id);
       }
    }
    
  6. 测试StudentService

    public class Test {
      public static void main(String[] args) {
       StudentService studentService = new StudentService();
       System.out.println(studentService.findStudentById(1));
       System.out.println(studentService.findStudentById(1));
       }
    }
    

测试结果:

  1. StudentService从容器中每次拿到的都是同一个StudentDao对象,节约了资源。

  2. 如果想使用StudentDaoImpl2对象,只需要修改bean.properties的内容为

    studentDao=com.Spring.dao.StudentDaoImpl2
    

    即可,无需修改Java源码。

Spring实现IOC

接下来我们使用Spring实现IOC,Spring内部也有一个容器用来管理对象。

  1. 创建Maven工程,引入依赖

    <dependencies>
      <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-context</artifactId>
        <version>5.3.13</version>
      </dependency>
      <dependency>
        <groupId>junit</groupId>
        <artifactId>junit</artifactId>
        <version>4.12</version>
        <scope>test</scope>
      </dependency>
    </dependencies>
    
    
  2. 创建POJO类、Dao类和接口

    public class Student {
      private int id;
      private String name;
      private String address;
        // 省略getter/setter/构造方法/tostring
    }
    
    
    public interface StudentDao {
      // 根据id查询学生
      Student findById(int id);
    }
    
    
    public class StudentDaoImpl implements StudentDao{
      @Override
      public Student findById(int id) {
        // 模拟从数据库查找出学生
        return new Student(1,"张三","北京");
       }
    }
    
    
  3. 编写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 id="studentDao" class="com.Spring.dao.StudentDaoImpl"></bean>
      
    </beans>
    
    
  4. 测试从Spring容器中获取对象。

    public class TestContainer {
      @Test
      public void t1(){
        // 创建Spring容器
        ApplicationContext ac = new ClassPathXmlApplicationContext("bean.xml");
        // 从容器获取对象
        StudentDao studentDao1 = (StudentDao) ac.getBean("studentDao");
        StudentDao studentDao2 = (StudentDao) ac.getBean("studentDao");
        System.out.println(studentDao1.hashCode());
        System.out.println(studentDao2.hashCode());
        System.out.println(studentDao1.findById(1));
       }
    }
    
    

Spring容器类型

容器接口
  • BeanFactory:BeanFactory是Spring容器中的顶层接口,它可以对Bean对象进行管理。

  • ApplicationContext:ApplicationContext是BeanFactory的子接口。它除了继承 BeanFactory的所有功能外,还添加了对国际化、资源访问、事件传播等方面的良好支持。

    ApplicationContext有以下三个常用实现类:

容器实现类
  • ClassPathXmlApplicationContext:该类可以从项目中读取配置文件
  • FileSystemXmlApplicationContext:该类从磁盘中读取配置文件
  • AnnotationConfigApplicationContext:使用该类不读取配置文件,而是会读取注解
@Test
public void t2(){
  // 创建spring容器
  //     ApplicationContext ac = new ClassPathXmlApplicationContext("bean.xml");
  ApplicationContext ac = new FileSystemXmlApplicationContext
      ("C:\\Users\\a\\IdeaProjects\\spring_demo\\src\\main\\resources\\bean.xml");
  
  // 从容器中获取对象
  StudentDao userDao = (StudentDao) ac.getBean("studentDao");
  System.out.println(userDao);
  System.out.println(userDao.findById(1));
}

对象的创建方式

Spring会帮助我们创建bean,那么它底层是调用什么方法进行创建的呢?

使用构造方法

Spring默认使用类的空参构造方法创建bean:

// 假如类没有空参构造方法,将无法完成bean的创建
public class StudentDaoImpl implements StudentDao{
  public StudentDaoImpl(int a){}
  
  @Override
  public Student findById(int id) {
    // 模拟根据id查询学生
    return new Student(1,"张三","北京");
   }
}
使用工厂类的方法

Spring可以调用工厂类的方法创建bean:

  1. 创建工厂类,工厂类提供创建对象的方法:

    public class StudentDaoFactory {
      public StudentDao getStudentDao(){
        return new StudentDaoImpl(1);
       }
    }
    
  2. 在配置文件中配置创建bean的方式为工厂方式。

    <!--  id:工厂对象的id,class:工厂类  -->
    <bean id="studentDaoFactory" class="com.Spring.dao.StudentDaoFactory"></bean>
    <!--  id:bean对象的id,factory-bean:工厂对象的id,factory-method:工厂方法 -->
    <bean id="studentDao" factory-bean="studentDaoFactory" factory-method="getStudentDao"></bean>
    
  3. 测试

使用工厂类的静态方法

Spring可以调用工厂类的静态方法创建bean:

  1. 创建工厂类,工厂提供创建对象的静态方法。

    public class StudentDaoFactory2 {
      public static StudentDao getStudentDao2() {
        return new StudentDaoImpl();
       }
    }
    
  2. 在配置文件中配置创建bean的方式为工厂静态方法。

    <!-- id:bean的id  class:工厂全类名 factory-method:工厂静态方法  -->
    <bean id="studentDao" class="com.Spring.dao.StudentDaoFactory2" factory-method="getStudentDao2"></bean>
    
  3. 测试

对象的创建策略

Spring通过配置<bean>中的scope属性设置对象的创建策略,共有五种创建策略:

  • singleton:单例,默认策略。整个项目只会创建一个对象,通过<bean>中的lazy-init属性可以设置单例对象的创建时机:

    lazy-init=“false”(默认):立即创建,在容器启动时会创建配置文件中的所有Bean对象。

    lazy-init=“true”:延迟创建,第一次使用Bean对象时才会创建。

    配置单例策略:

    <!--   <bean id="studentDao" class="com.Spring.dao.StudentDaoImpl2" scope="singleton" lazy-init="true"></bean>-->
    <bean id="studentDao" class="com.Spring.dao.StudentDaoImpl2" 
          scope="singleton" lazy-init="false">
    </bean>
    

    测试单例策略:

    // 为Bean对象的类添加构造方法
    public StudentDaoImpl2(){
      System.out.println("创建StudentDao!!!");
    }
    @Test
    public void t2(){
      // 创建Spring容器
      ApplicationContext ac = new ClassPathXmlApplicationContext("bean1.xml");
      // 从容器获取对象
      StudentDao studentDao1 = (StudentDao) ac.getBean("studentDao");
      StudentDao studentDao2 = (StudentDao) ac.getBean("studentDao");
      StudentDao studentDao3 = (StudentDao) ac.getBean("studentDao");
      System.out.println(studentDao1.hashCode());
      System.out.println(studentDao2.hashCode());
      System.out.println(studentDao3.hashCode());
    }
    
    
  • prototype:多例,每次从容器中获取时都会创建对象。

    <!-- 配置多例策略 -->
    <bean id="studentDao"   class="com.Spring.dao.StudentDaoImpl2" scope="prototype"></bean>
    
  • request:每次请求创建一个对象,只在web环境有效。

  • session:每次会话创建一个对象,只在web环境有效。

  • gloabal-session:一次集群环境的会话创建一个对象,只在web环境有效。

对象的销毁时机

对象的创建策略不同,销毁时机也不同:

  • singleton:对象随着容器的销毁而销毁。
  • prototype:使用JAVA垃圾回收机制销毁对象。
  • request:当处理请求结束,bean实例将被销毁。
  • session:当HTTP Session最终被废弃的时候,bean也会被销毁掉。
  • gloabal-session:集群环境下的session销毁,bean实例也将被销毁。

生命周期方法

Bean对象的生命周期包含创建——使用——销毁,Spring可以配置Bean对象在创建和销毁时自动执行的方法:

  1. 定义生命周期方法

    public class StudentDaoImpl2 implements StudentDao{
       // 创建时自动执行的方法
      public void init(){
        System.out.println("创建StudentDao!!!");
       }
    
    
      // 销毁时自动执行的方法
      public void destory(){
        System.out.println("销毁StudentDao!!!");
       }
    }
    
  2. 配置生命周期方法

    <!-- init-method:创建对象时执行的方法  destroy-method:销毁对象时执行的方法  -->
    <bean id="studentDao" class="com.Spring.dao.StudentDaoImpl2" scope="singleton"
       init-method="init" destroy-method="destory"></bean>
    
  3. 测试

    @Test
    public void t3(){
      // 创建Spring容器
      ClassPathXmlApplicationContext ac = new ClassPathXmlApplicationContext("bean1.xml");
    
    
      // 销毁Spring容器,ClassPathXmlApplicationContext才有销毁容器的方法
      ac.close();
    }
    

获取Bean对象的方式

Spring有多种获取容器中对象的方式:

通过id/name获取
  1. 配置文件

    <bean name="studentDao" class="com.Spring.dao.StudentDaoImpl2"></bean>
        ||
    <bean id="studentDao" class="com.Spring.dao.StudentDaoImpl2"></bean>
    
  2. 获取对象

    StudentDao studentDao = (StudentDao) ac.getBean("studentDao");
    
通过类型获取
  1. 配置文件

    <bean name="studentDao" class="com.Spring.dao.StudentDaoImpl2"></bean>
    
  2. 获取对象

    StudentDao studentDao2 = ac.getBean(StudentDao.class);
    

    可以看到使用类型获取不需要强转。

通过类型+id/name获取

虽然使用类型获取不需要强转,但如果在容器中有一个接口的多个实现类对象,则获取时会报错,此时需要使用类型+id/name获取

  1. 配置文件

    <bean name="studentDao" class="com.Spring.dao.StudentDaoImpl2"></bean>
    <bean name="studentDao1" class="com.Spring.dao.StudentDaoImpl"></bean>
    
  2. 获取对象

    StudentDao studentDao2 = ac.getBean("studentDao",StudentDao.class);
    

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1257562.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

眼精星票证识别系统操作教程与技巧

眼精星票证识别系统是一款高效、准确的票证识别工具&#xff0c;可以帮助用户快速将各种票证识别成结构化数据&#xff0c;并支持批量合并导出Excel。该系统的使用非常简单&#xff0c;只需要按照以下步骤进行操作即可&#xff1a; 1. 下载并安装眼精星票证识别系统 来百度AP…

SSM图书捐赠网站系统开发mysql数据库web结构java编程计算机网页源码eclipse项目

一、源码特点 SSM 图书捐赠网站系统是一套完善的信息系统&#xff0c;结合springMVC框架完成本系统&#xff0c;对理解JSP java编程开发语言有帮助系统采用SSM框架&#xff08;MVC模式开发&#xff09;&#xff0c;系统具有完整的源代码和数据库 &#xff0c;系统主要采用B/…

深入理解 Docker 核心原理:Namespace、Cgroups 和 Rootfs

来自&#xff1a;探索云原生 https://www.lixueduan.com 原文&#xff1a;https://www.lixueduan.com/posts/docker/03-container-core/ 通过这篇文章你可以了解到 Docker 容器的核心实现原理&#xff0c;包括 Namespace、Cgroups、Rootfs 等三个核心功能。 后续文章会演示如…

k8s中安装consul集群

一、准备知识 headless services一般结合StatefulSet来部署有状态的应用&#xff0c;比如kafka集群&#xff0c;mysql集群&#xff0c;zk集群等&#xff0c;也包括本文要部署的consul集群。 0、consul集群 consul集群的分布式协议算法采用的是raft协议&#xff0c;这意味着必…

UI彩虹外链网盘系统整站源码/PHP网盘与外链分享程序/整站+模版文件

源码简介&#xff1a; 全新UI彩虹外链网盘系统源码&#xff0c;它是PHP网盘与外链分享程序&#xff0c;提供了整站模版文件&#xff0c;前后端美化模板。 彩虹外链网盘美化模板是一款专为PHP网盘和外链分享程序设计的模板。它具备多种功能&#xff0c;包括支持所有格式文件的…

记录本地与服务器之间数据传输方法(上传、下载文件)

文章目录 一、使用scp命令实现参数说明示例说明 二、使用工具实现windows系统苹果系统如有启发&#xff0c;可点赞收藏哟~ 一、使用scp命令实现 scp 是 secure copy &#xff08;安全复制&#xff09;的缩写, scp 是基于 ssh 登陆进行安全的远程文件拷贝命令。相当于 cp 命令 …

ChatGPT进阶:提示工程的神秘面纱与实战指南

文章目录 一、提示工程的概念与原理二、提示工程的实践方法三、提示工程的挑战与展望四、实战案例分析总结《ChatGPT进阶&#xff1a;提示工程入门》内容简介作者简介陈颢鹏&#xff1a;李子菡&#xff1a; 目录获取方式 在人工智能领域&#xff0c;对话系统已经成为了一个热门…

数智赋能 锦江汽车携手苏州金龙打造高质量盛会服务

作为一家老牌客运公司&#xff0c;成立于1956年的上海锦江汽车服务有限公司&#xff08;以下简称锦江汽车&#xff09;&#xff0c;拥有1200多辆大巴和5000多辆轿车&#xff0c;是上海乃至长三角地区规模最大的专业旅游客运公司。面对客运市场的持续萎缩&#xff0c;锦江汽车坚…

不用再羡慕“SpaceX“ 民营星际荣耀打造国产可回收火箭双曲线二号

大家好&#xff0c;我是极智视界&#xff0c;欢迎关注我的公众号&#xff0c;获取我的更多前沿科技分享 邀您加入我的知识星球「极智视界」&#xff0c;星球内有超多好玩的项目实战源码和资源下载&#xff0c;链接&#xff1a;https://t.zsxq.com/0aiNxERDq 大家似乎习惯了对马…

PDF不小心关闭没保存?这4个方法教你拯救文件!

“我刚刚在查看一个PDF文件&#xff0c;但是一不小心我就把它关闭了&#xff0c;而且我还忘记保存了。这可怎么办呢&#xff1f;PDF不小心关闭没保存应该怎么解决呢&#xff1f;还有办法恢复吗&#xff1f;” PDF文档是我们日常工作和学习中常用的文件格式&#xff0c;但有时候…

QQ录屏保存到哪了?教你快速找到保存位置

qq录屏是许多用户常用的屏幕录制工具&#xff0c;可是一旦录制结束&#xff0c;许多人不清楚如何找到和管理录制的视频文件。那么&#xff0c;您知道qq录屏保存到哪了吗&#xff1f;本文将深入研究qq录制视频功能&#xff0c;以帮助您了解如何存储和管理这些重要的录制视频文件…

【Android Gradle】之一小时 Gradle及 wrapper 入门

&#x1f604;作者简介&#xff1a; 小曾同学.com,一个致力于测试开发的博主⛽️&#xff0c;主要职责&#xff1a;测试开发、CI/CD 如果文章知识点有错误的地方&#xff0c;还请大家指正&#xff0c;让我们一起学习&#xff0c;一起进步。 &#x1f60a; 座右铭&#xff1a;不…

【算法萌新闯力扣】:合并两个有序链表

力扣题目&#xff1a;合并两个有序链表 开篇 今天是备战蓝桥杯的第24天及算法村开营第2天。根据算法村的讲义&#xff0c;来刷链表的相关题目。今天要分享的是合并两个有序链表。 题目链接: 21.合并两个有序链表 题目描述 代码思路 通过创建一个新链表&#xff0c;然后遍历…

DEM分析

一、实验名称&#xff1a; DEM分析 二、实验目的&#xff1a; 通过本实验练习&#xff0c;掌握DEM的建立与应用基本方法。 三、实验内容和要求&#xff1a; 实验内容&#xff1a; 利用ARCGIS软件相关分析工具及实验数据&#xff0c;创建DEM&#xff0c;并计算相应坡度的区…

Android 单元测试初体验

Android 单元测试初体验 前言一、单元测试是什么&#xff1f;二、简单使用1.依赖2.单元测试代码简单模版及解释 总结 前言 当初在学校学安卓的时候&#xff0c;老师敢教学进度&#xff0c;翻到单元测试这一章节的时候提了两句&#xff0c;没有把单元测试当重点讲&#xff0c;只…

文章解读与仿真程序复现思路——电力自动化设备EI\CSCD\北大核心《考虑碳排放分摊的综合能源服务商交易策略》

这篇文章的标题表明它将讨论一个关于综合能源服务商交易策略的主题&#xff0c;而在这个策略中&#xff0c;特别考虑了碳排放分摊的因素。以下是对标题中各关键词的解读&#xff1a; 综合能源服务商&#xff1a; 这指的是在能源领域提供多种服务的企业或组织&#xff0c;可能涵…

https到底把什么加密了?

首先直接说结论&#xff0c; https安全通信模式&#xff0c;是使用TLS加密传输所有的http协议。再重复一遍&#xff0c;是所有&#xff01; 通常将TLS加密传输http这个通信过程称为https&#xff0c;如果使用协议封装的逻辑结构来表达就是&#xff1a; IP TCP TLS 【 HTTP 】…

算法效率的度量

算法效率的度量通常是通过时间复杂度和空间复杂度来描述的。 一、时间复杂度 算法中所有语句的执行次数之和为T(n)&#xff0c;它是算法问题规模n的函数&#xff0c;时间复杂度主要分析T(n)的数量级。 分类 1. 最好时间复杂度&#xff1a;最好情况下&#xff0c;算法的时间…

python subprocess

查看python官方文档&#xff1a;最全 p subprocess.Popen([rpng2bdf.exe,[r-o .\tst\myfont.bdf -f myfont -e 65 tst\*.png]],stdoutsubprocess.PIPE,stderr subprocess.PIPE) out,err p.communicate() print(out) 注意&#xff0c;如何将shell命令分解为参数序列可能并…

HT97226 免输出电容立体声耳机放大器的应用与曲线

HT97226应用&#xff1a; ・耳机 ・多媒体音频接口 ・机顶盒 ・ 蓝光/DVD播放器 ・LCD电视 ・音频消费电子产品 HT97226应用图于曲线&#xff1a; HT97226是一款差分输入/单端输入、可直接输出驱动的耳机放大器。5V供…