关于spring,总结了一篇上万字的图文笔记,不管你工作几年都应该看看。

逻辑灵珀探
• 阅读 1713

spring bean的实例化

构造器实例化

 <!--
        无参构造器实例化
    -->
    <bean id="student1" class="com.qfedu.entity.Student">
        <property name="id" value="100"></property>
        <property name="name" value="xiaoming"></property>
    </bean>

静态工厂实例化

容器创建对象,不直接调用对象构造方法,而是调用静态工厂的创建对象的方法
好处:便于我们定制化创建对象,对象的初始化,需要访问网络中的数据

/**
 * 静态工厂
 *    静态方法创建对象
 */
public class StaticFactory {


    /**
     * 静态方法 创建对象
     * @return
     */
    public static Student createStudent(){

        Student student = new Student();

        
        // 好处就是 程序员可以 自定义 初始化对象,并交给spring容器创建 
        student.setId(123);
        student.setName("小明");

        return  student;
    }

}
<!--
       告诉容器使用静态工厂 创建对象
        class="com.qfedu.factory.StaticFactory" 静态工厂
        factory-method="createStudent" 调用静态工厂的方法名
    -->
    <bean id="student2" class="com.qfedu.factory.StaticFactory" factory-method="createStudent">

    </bean>

个人整理了一些资料,有需要的朋友可以直接点击领取。

Java基础知识大全

22本Java架构师核心书籍

从0到1Java学习路线和资料

1000+道2021年最新面试题

实例工厂实例化

/**
 * 实例工厂 创建对象
 */
public class Factory {


    /**
     * 实例方法 创建对象
     * @return
     */
    public  Student createStudent(){

        Student student = new Student();


        // 好处就是 程序员可以 自定义 初始化对象,并交给spring容器创建
        student.setId(123);
        student.setName("小明");

        return  student;
    }
}
<?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="factory" class="com.qfedu.factory.Factory">

    </bean>
    <!--
            目标创建 student  使用实例工厂
                factory-bean="factory" 引用实例工厂
                factory-method="createStudent" 调用实例工厂中的实例 方法
    -->
    <bean id="student" factory-bean="factory" factory-method="createStudent">

    </bean>

</beans>

工厂实例化的使用场景

工厂实例化作用:便于程序员自定义创建对象
使用场景:初始化数据库数据时,对数据库密码进行解密,将数据源放置到容器中,提高安全性

bean的作用域

作用域:就是bean 在容器中生存的范围
常用的:单例,原型
关于spring,总结了一篇上万字的图文笔记,不管你工作几年都应该看看。

单例模式

<!--
       让容器创建一个student
       默认 该bean 作用域 是单例singleton  scope="singleton"
             单例singleton 无论用户是否获取该bean,容器在启动创建该bean,而且只创建一个 对应bean id为 student
   -->
   <bean id="student" class="com.qfedu.entity.Student" scope="singleton">
       <property name="id" value="100"></property>
       <property name="name" value="xiaoming"></property>
   </bean>

原型测试


    <!--
        告诉容器创建一个 原型 bean  id="student1"
        什么时候创建? 容器启动不创将该bean,在用户获取时创建,而且每获取一次创建一个bean,容器只负责创建,
        不负责持有,管理该bean
    -->
    <bean id="student1" class="com.qfedu.entity.Student" scope="prototype">
        <property name="id" value="100"></property>
        <property name="name" value="xiaoming"></property>
    </bean>

测试

public class ScopeTest {
    public static void main(String[] args) {

        ApplicationContext applicationContext = new ClassPathXmlApplicationContext("scope/scope_bean.xml");

        System.out.println("单例测试");
        //单例的bena无论获取多少次都是单例
        Student student1 = (Student) applicationContext.getBean("student");

        Student student2 = (Student) applicationContext.getBean("student");

        System.out.println("student1 == student2:" + (student1 == student2));

        System.out.println("*********");
        System.out.println("原型测试");
        // 获取 原型bean, 每次获取都是一个新的对象,容器只负责创建,不负责持有,维护
        Student student3 = (Student) applicationContext.getBean("student1");

        Student student4 = (Student) applicationContext.getBean("student1");

        System.out.println("student3 == student4:" + (student3 == student4));
    }
}

bean的生命周期

bean:交给容器维护,容器管理bean
bean的生命周期 的bean 的作用域相关

singleton

Spring容器可以管理singleton作用域的Bean的生命周期,在此作用域下,Spring能够精确的知道该Bean何时被创建,何时初始化完成,以及何时被销毁。

容器创建bean,管理bean
容器初始化时创建 单例的 bean,并且持有bean, 容器销毁时 ,销毁bean

prototype

prototype作用域的Bean,Spring只负责创建,当容器创建了Bean实例后,Bean的实例就交给客户端代码来管理,Spring容器将不再跟踪其生命周期。

容器只创建bean不管理
容器初始化时,不会创建bean,只有在获取时才会创建bean,spring 只负责创建,不持有bean,也不负责销毁,还可以为bean 配置 init()初始化方法,destroy() 销毁方法

public class Student implements Serializable {

    private int id;

    private String name;

    /**
     * 初始化 资源
     */
    public void  init(){

        System.out.println("Student 初始化方法");
    }

    /**
     * 释放资源
     */
    public void destroy(){
        System.out.println("Student 销毁方法");
    }
。。。。
}
<?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 scope="singleton"

              init-method="init" bean 在创建时调用 初始化方法
              destroy-method="destroy" 容器销毁 方法
    -->

    <bean id="student1" class="com.qfedu.entity.Student" scope="singleton"
          init-method="init" destroy-method="destroy">
        <property name="id" value="100"></property>
        <property name="name" value="xiaoming"></property>
    </bean>



    <!--
    原型bean scope="prototype"

          init-method="init" bean 在创建时调用 初始化方法
          destroy-method="destroy" 容器销毁 方法
-->
    <bean id="student2" class="com.qfedu.entity.Student" scope="prototype"
          init-method="init" destroy-method="destroy">
        <property name="id" value="100"></property>
        <property name="name" value="张三"></property>
    </bean>
</beans>

测试:

/**
 *bena的生命周期
 */
public class LifeTest {
    public static void main(String[] args) {
        ClassPathXmlApplicationContext classPathXmlApplicationContext  = new ClassPathXmlApplicationContext("life/life_bean.xml");

        Student student1 = (Student) classPathXmlApplicationContext.getBean("student1");

        System.out.println(student1);
        System.out.println("容器销毁");

        // 明示销毁容器,此时会调用容器中所有bean destroy() 方法
        // 单例bean 调用 destroy()
        // 原型bean  不会调用 destroy()  因为容器不持有该bean
        classPathXmlApplicationContext.destroy();
    }
}

Bean的装配

什么Bean的装配?
就是bean 属性的设置,以及bean 之间依赖关系的配置

基于XML的装配

property和构造方法设置值(无参和有参)

<?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">


    <!--
         通过无参构造 使用setter 初始化bean
    -->
    <bean id="student1" class="com.qfedu.entity.Student">
        <property name="id" value="100"></property>
        <property name="name" value="xiaoming"></property>
        <property name="sex" value="F"></property>

        <property name="course" >

            <list>
                <value>Java</value>
                <value>UI</value>
                <value>H5</value>
                <value>php</value>
            </list>
        </property>
    </bean>

    <!--
        通过有参构造创建  bean
        public Student(int id, String name, String sex, List<String> course)
                            0        1               2          3
                            
          <constructor-arg index="0" value="111"></constructor-arg> 为每一个构造方法的参数设置
          属性                     
     -->
    <bean id="student1" class="com.qfedu.entity.Student">

        <constructor-arg index="0" value="111"></constructor-arg>
        <constructor-arg index="1" value="zhangsan"></constructor-arg>
        <constructor-arg index="2" value="F"></constructor-arg>
        <constructor-arg index="3" >
            
            <list>
                <value>Java</value>
                <value>UI</value>
                <value>H5</value>
                <value>php</value>
            </list>
        </constructor-arg>
    </bean>

</beans>

测试

public class XmlTest {
    public static void main(String[] args) {

        ApplicationContext applicationContext = new ClassPathXmlApplicationContext("xml/xml_bean.xml");

        Student student1 = (Student) applicationContext.getBean("student1");
        System.out.println("student1 = " + student1);

        Student student2 = (Student) applicationContext.getBean("student2");
        System.out.println("student2 = " + student2);
    }
}

基于注解的装配

注解:就是一个标记,记号
通过注解创建bean 并管理bean 之间的依赖关系

注入的注解
一个实例中的属性可以通过以下注解从容器获取对应的bean,并注入进来:

@Autowired:用于对Bean的属性变量、属性的setter方法及构造方法进行标注,配合对应的注解处理器完成Bean的自动配置工作

@Qualifier:与@Autowired注解配合使用,会将默认的按Bean类型装配修改为按Bean的实例名称装配,Bean的实例名称由@Qualifier注解的参数指定。

@Resource:其作用与Autowired一样。@Resource中有两个重要属性:name和type。Spring将name属性解析为Bean实例名称,type属性解析为Bean实例类型。

@Autowired + @Qualifier

激活注入的注解:

 <!--

        这是一个开关,激活 @Autowired    @ Resource 、@ PostConstruct、@ PreDestroy
                       让他们生效
    -->
    <context:annotation-config></context:annotation-config>

加入注解到容器中:

<!--
       将StudentDaoImpl加入到容器中StudentDaoImpl
   -->
    <bean id="studentDao1" class="com.qfedu.dao.StudentDaoImpl"></bean>
    <bean id="studentDao2" class="com.qfedu.dao.StudentDaoImpl"></bean>

    <!--
       将StudentServiceImpl加入到容器中
   -->
    <bean id="studentService" class="com.qfedu.service.StudentServiceImpl"></bean>

装配:

public class StudentServiceImpl implements StudentService{


    /*@Autowired 不需要 setter方法支持
     * 意义: 1.首先根据注解的类型 去容器中查找 如果只有一个,则设置
     *       2.如果按照类型查找到 多个 ,则使用 变量名( private StudentDao studentDao) 作为id 去容器中查找
     *       3.如果按照变量名找不到,可以使用@Qualifier("studentDao2") 配置,按照传递的参数作为iD 查找
     */

    // 去容器中查找 StudentDao 对应实例,并将当前 属性引用
    @Autowired
    @Qualifier("studentDao2")//如果找到多个使用  @Qualifier区分
    private StudentDao studentDao;

//    public void setStudentDao(StudentDao studentDao){
//        this.studentDao = studentDao;
//    }

    public Student findStudentById(int id) {
        return studentDao.findStudentById(id);
    }
}

@Resource

激活注入的注解:

 <!--

        这是一个开关,激活 @Autowired    @ Resource 、@ PostConstruct、@ PreDestroy
                       让他们生效
    -->
    <context:annotation-config></context:annotation-config>

加入注解到容器中:


    <!--
       将StudentDaoImpl加入到容器中StudentDaoImpl
   -->
    <bean id="studentDao1" class="com.qfedu.dao.StudentDaoImpl"></bean>
    <bean id="studentDao2" class="com.qfedu.dao.StudentDaoImpl"></bean>

    <!--
       将StudentServiceImpl加入到容器中
   -->
    <bean id="studentService" class="com.qfedu.service.StudentServiceImpl"></bean>

装配:

public class StudentServiceImpl implements StudentService{


    /*
     *  @Resource  也是 将容器中的bean 注入到当前对象中
     * 意义:
     *      1.首先按照声明 属性名作为id 去容器中查找(private StudentDao studentDao ),
     *      2.如果没有找到 按照类型查找 ,如果查找到一个则设置值, 如果查到多个,则进入第三步
     *      3.@Resource(name = "studentDao2"): 如果找到多个类型的bean 必须传入name 作为id 进行限定
     *
     * @Autowired    @Resource 的区别?
     *
     * 1. @Resource(name = "studentDao2") 等价于 @Autowired  @Qualifier
     * 2.意义
     * 3.@Resource 是jdk 注解 @Autowired 是spring 注解
    * */

    @Resource(name = "studentDao2")
    private StudentDao studentDao;

//    public void setStudentDao(StudentDao studentDao){
//        this.studentDao = studentDao;
//    }

    public Student findStudentById(int id) {
        return studentDao.findStudentById(id);
    }
}

总结

@Autowired 不需要 setter方法支持

  • 意义: 1.首先根据注解的类型 去容器中查找 如果只有一个,则设置
  • 2.如果按照类型查找到 多个 ,则使用 变量名( private StudentDao studentDao) 作为id 去容器中查找
  • 3.如果按照变量 名找不到,可以使用@Qualifier(“studentDao2”) 配置,按照传递的参数作为iD 查找
    *
  • @Resource 也是 将容器中的bean 注入到当前对象中
  • 意义:
  • 1.首先按照声明 属性名作为id 去容器中查找(private StudentDao studentDao ),
  • 2.如果没有找到 按照类型查找 ,如果查找到一个设置值 如果多个,则进入第三部
  • 3.@Resource(name = “studentDao2”) 如果找到多个类型的bean 必须传入name 作为id 进行限定
    *
  • @Autowired @Resource 的区别?
    *
    1. @Resource(name = “studentDao2”) 等价于 @Autowired @Qualifier
  • 2.意义
  • 3.@Resource 是jdk 注解 @Autowired spring 注解
    *

    生成bean的注解

    @Component
    //@Component // 在容器中加入bean 默认id studentServiceImpl
    @Component(“studentService”) // 在容器中加入bean id studentService
    以下三个用法和 @Component 一样是@Component 的子注解
    @Service 用于service
    @Controller 用于控制层
    @Repository 用于持久层 dao
    关于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"
       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 https://www.springframework.org/schema/context/spring-context.xsd">


    <!--
        包扫描器 :
        激活 注入注解   @Autowired    @ Resource
        激活生成bean 的注解
            @Component
                    //@Component // 在容器中加入bean 默认id  studentServiceImpl
                    @Component("studentService")   // 在容器中加入bean id  studentService
            以下三个用法和 @Component 一样是@Component 的子注解
            @Service   用于service
            @Controller 用于控制层
            @Repository  用于持久层 dao
    -->
    <context:component-scan base-package="com.qfedu"></context:component-scan>


</beans>
//@Component // 在容器中加入bean 默认id  studentServiceImpl
//@Component("studentService")   // 在容器中加入bean id  studentService
@Service("studentService")// 加入到容器  id  studentService
public class StudentServiceImpl implements StudentService {

    // 去容器中查找 StudentDao 对应实例,并将当前 属性引用
    /**
     *  @Autowired 不需要 setter方法支持
     * 意义: 1.首先根据注解的类型 去容器中查找 如果只有一个,则设置
     *       2.如果按照类型查找到 多个 ,则使用 变量名( private StudentDao studentDao) 作为id 去容器中查找
     *       3.如果按照变量 名找不到,可以使用@Qualifier("studentDao2") 配置,按照传递的参数作为iD 查找
     *
     * @Resource  也是 将容器中的bean 注入到当前对象中
     * 意义:
     *      1.首先按照声明 属性名作为id 去容器中查找(private StudentDao studentDao ),
     *      2.如果没有找到 按照类型查找 ,如果查找到一个设置值 如果多个,则进入第三部
     *      3.@Resource(name = "studentDao2") 如果找到多个类型的bean 必须传入name 作为id 进行限定
     *
     * @Autowired    @Resource 的区别?
     *
     * 1. @Resource(name = "studentDao2") 等价于 @Autowired  @Qualifier
     * 2.意义
     * 3.@Resource 是jdk 注解 @Autowired spring 注解
     *
     */
//    @Autowired
//    @Qualifier("studentDao2")//如果找到多个使用  @Qualifier区分
    @Resource
    private StudentDao studentDao ;

//    public void setStudentDao(StudentDao studentDao) {
//        this.studentDao = studentDao;
//    }

    public Student findStudentById(int id) {
        return studentDao.findStudentById(id);
    }
}
//@Component// 将当前类 创建一个bean 加入到容器中 id studentDaoImpl
@Repository // 加入到容器 id studentDaoImpl
public class StudentDaoImpl implements StudentDao{
    public Student findStudentById(int id) {

        // 模拟取数据库查询
        Student student = new Student();
        student.setId(id);
        student.setName("XXXXX");

        return student;
    }
}

测试:

/**
 * 自动转配
 */
public class AnnootaionTest {


    public static void main(String[] args) {

        ApplicationContext applicationContext = new ClassPathXmlApplicationContext("annotation/auto_annotation.xml");


        StudentService studentService = (StudentService) applicationContext.getBean("studentService");


        Student student =  studentService.findStudentById(12);
        System.out.println("student"+student);
    }

}

基于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 http://www.springframework.org/schema/beans/spring-beans.xsd">



    <bean id="studentDao1" class="com.qfedu.dao.StudentDaoImpl"></bean>
    <bean id="studentDao" class="com.qfedu.dao.StudentDaoImpl"></bean>

    <!--
        加入到容器中  StudentServiceImpl

         autowire="byName" 容器中的属性 会根据属性名作为id 去容器中查找
         autowire="byType"  容器中的属性 会根据属性类型   去容器中查找
          autowire="constructor" 根据构造方法设置 属性 
    -->
    <bean id="studentService" class="com.qfedu.service.StudentServiceImpl" autowire="byName">

    </bean>

</beans>

最后

感谢大佬们能看到这里,来都来了不妨点个赞再走呗

点赞
收藏
评论区
推荐文章
blmius blmius
3年前
MySQL:[Err] 1292 - Incorrect datetime value: ‘0000-00-00 00:00:00‘ for column ‘CREATE_TIME‘ at row 1
文章目录问题用navicat导入数据时,报错:原因这是因为当前的MySQL不支持datetime为0的情况。解决修改sql\mode:sql\mode:SQLMode定义了MySQL应支持的SQL语法、数据校验等,这样可以更容易地在不同的环境中使用MySQL。全局s
皕杰报表之UUID
​在我们用皕杰报表工具设计填报报表时,如何在新增行里自动增加id呢?能新增整数排序id吗?目前可以在新增行里自动增加id,但只能用uuid函数增加UUID编码,不能新增整数排序id。uuid函数说明:获取一个UUID,可以在填报表中用来创建数据ID语法:uuid()或uuid(sep)参数说明:sep布尔值,生成的uuid中是否包含分隔符'',缺省为
Jacquelyn38 Jacquelyn38
4年前
2020年前端实用代码段,为你的工作保驾护航
有空的时候,自己总结了几个代码段,在开发中也经常使用,谢谢。1、使用解构获取json数据let jsonData  id: 1,status: "OK",data: 'a', 'b';let  id, status, data: number   jsonData;console.log(id, status, number )
Wesley13 Wesley13
3年前
FLV文件格式
1.        FLV文件对齐方式FLV文件以大端对齐方式存放多字节整型。如存放数字无符号16位的数字300(0x012C),那么在FLV文件中存放的顺序是:|0x01|0x2C|。如果是无符号32位数字300(0x0000012C),那么在FLV文件中的存放顺序是:|0x00|0x00|0x00|0x01|0x2C。2.  
Wesley13 Wesley13
3年前
mysql设置时区
mysql设置时区mysql\_query("SETtime\_zone'8:00'")ordie('时区设置失败,请联系管理员!');中国在东8区所以加8方法二:selectcount(user\_id)asdevice,CONVERT\_TZ(FROM\_UNIXTIME(reg\_time),'08:00','0
Wesley13 Wesley13
3年前
PHP创建多级树型结构
<!lang:php<?php$areaarray(array('id'1,'pid'0,'name''中国'),array('id'5,'pid'0,'name''美国'),array('id'2,'pid'1,'name''吉林'),array('id'4,'pid'2,'n
Wesley13 Wesley13
3年前
Java日期时间API系列36
  十二时辰,古代劳动人民把一昼夜划分成十二个时段,每一个时段叫一个时辰。二十四小时和十二时辰对照表:时辰时间24时制子时深夜11:00凌晨01:0023:0001:00丑时上午01:00上午03:0001:0003:00寅时上午03:00上午0
Wesley13 Wesley13
3年前
00:Java简单了解
浅谈Java之概述Java是SUN(StanfordUniversityNetwork),斯坦福大学网络公司)1995年推出的一门高级编程语言。Java是一种面向Internet的编程语言。随着Java技术在web方面的不断成熟,已经成为Web应用程序的首选开发语言。Java是简单易学,完全面向对象,安全可靠,与平台无关的编程语言。
Wesley13 Wesley13
3年前
MySQL部分从库上面因为大量的临时表tmp_table造成慢查询
背景描述Time:20190124T00:08:14.70572408:00User@Host:@Id:Schema:sentrymetaLast_errno:0Killed:0Query_time:0.315758Lock_
Python进阶者 Python进阶者
1年前
Excel中这日期老是出来00:00:00,怎么用Pandas把这个去除
大家好,我是皮皮。一、前言前几天在Python白银交流群【上海新年人】问了一个Pandas数据筛选的问题。问题如下:这日期老是出来00:00:00,怎么把这个去除。二、实现过程后来【论草莓如何成为冻干莓】给了一个思路和代码如下:pd.toexcel之前把这
美凌格栋栋酱 美凌格栋栋酱
5个月前
Oracle 分组与拼接字符串同时使用
SELECTT.,ROWNUMIDFROM(SELECTT.EMPLID,T.NAME,T.BU,T.REALDEPART,T.FORMATDATE,SUM(T.S0)S0,MAX(UPDATETIME)CREATETIME,LISTAGG(TOCHAR(