• SSM框架--Spring配置文件


    Spring配置文件

    1.Bean标签的基本配置

    在这里插入图片描述
    用于配置对象交由Spring来创建。
    默认情况下它调动的是类中的无参构造函数,如果没有无参构造函数则不能创建成功。

    基本属性:

    • id:作用是唯一标识,也就是说在配置文件内部,不允许重复
    • class:Bean的全限定类名

    2.Bean标签的范围配置

    scope:指对象的作用范围,取值如下:
    在这里插入图片描述

    测试singleton和prototype的区别:

    第一种情况:
    在Spring配置文件applicationContext.xml中,class后面添加scope为singleton:

    在这里插入图片描述

    为了方便测试,在test文件下创建一个包“com.xy.test”
    创建一个测试类,“SpringTest”:

    在这里插入图片描述

    在pom.xml文件中添加坐标,代码如下:

     <dependency>
                <groupId>junit</groupId>
                <artifactId>junit</artifactId>
                <version>4.12</version>
                <scope>test</scope>
            </dependency>
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6

    前提:需要先下载IDEA插件junit
    IDEA安装插件方式:搜索安装JunitGenerator V2.0插件
    Junit的jar包下载
    配置Junit测试环境

    在这里插入图片描述

    编写测试方法,代码如下:

    public class SpringTest {
        @Test
    //    测试scope属性
        public void test1(){
    //        获取客户端代码
            ApplicationContext app=new ClassPathXmlApplicationContext("applicationContext.xml");
    //        调动getBean方法
            UserDao userDao1 = (UserDao) app.getBean("userDao");
            UserDao userDao2 = (UserDao) app.getBean("userDao");
            System.out.println(userDao1);
            System.out.println(userDao2);
        }
    }
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14

    运行结果:两个地址一样,说明在Spring容器当中userDao的Bean存在一个

    在这里插入图片描述

    第二种情况:
    在Spring配置文件applicationContext.xml中,class后面添加scope为prototype:

    在这里插入图片描述

    运行结果不一致,说明Spring容器中userDao是多个

    测试对象Bean的创建时机:
    由于scope配置成singleton和prototype的不同会导致Bean的创建时机是不一样的

    第一种情况(singleton):
    applicationContext.xml中:

    <bean id="userDao" class="com.xy.dao.impl.UserDaoImpl" scope="singleton"></bean>
    
    • 1

    默认情况下,上面的这种配置表示,找的是无参构造创建对象, 所以在UserDaoImpl内部去复写它的无参构造:

    //    无参构造,快捷键fn+ctrl+insert
        public UserDaoImpl() {
            System.out.println("UserDaoImpl创建...");
        }
    
    • 1
    • 2
    • 3
    • 4

    这句话执行一次,控制台打印一次,证明无参方法调用一次,无参构造调用一次就代表对象创建一次

    在这里插入图片描述

    在测试类的第一句代码前打一个断点:

    在这里插入图片描述

    单击断点上面的绿色图标,选“Debug”:

    在这里插入图片描述

    点“下一步”跳入,单步执行:

    在这里插入图片描述

    运行结果,创建了UserDaoImpl,但未获得对象:

    在这里插入图片描述

    多次单步执行的运行结果:

    在这里插入图片描述

    说明Bean的创建时机,是在加载配置文件创建Spring容器时

    第二种情况(prototype):
    applicationContext.xml中:

    <bean id="userDao" class="com.xy.dao.impl.UserDaoImpl" scope="prototype"></bean>
    
    • 1

    操作同上:

    单击一次单步执行,创建了Spring配置文件,但Bean对象没有被创建

    在这里插入图片描述

    再单击一次执行,说明Bean被创建了

    在这里插入图片描述

    再按一次,又打印一行,
    说明Bean对象又创建了一个

    在这里插入图片描述

    由此说明,Prototype时Bean的创建时机,是在每次getBean时创建一个

    总结:
    在这里插入图片描述

    3.Bean生命周期配置

    • init-method:指定类中的初始化方法名称
    • destroy-method:指定类中销毁方法名称

    在UserDaoImpl中创建初始化方法和销毁方法:

    //    对象创建完毕后的初始化方法
        public void init(){
            System.out.println("初始化方法...");
        }
    
        //    对象销毁之前的销毁方法
        public void destroy() {
            System.out.println("销毁方法...");
        }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9

    在applicationContext配置文件中配置init-method和destroy-method这两个属性:

        <bean id="userDao" class="com.xy.dao.impl.UserDaoImpl" init-method="init" destroy-method="destroy" ></bean>
    
    • 1

    在测试类中手动关闭Spring容器,让代码能够来得及执行销毁方法

        public void test1() {
    //        获取客户端代码
            ClassPathXmlApplicationContext app = new ClassPathXmlApplicationContext("applicationContext.xml");
    //        调动getBean方法
            UserDao userDao1 = (UserDao) app.getBean("userDao");
            System.out.println(userDao1);
            app.close();
    
        }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9

    运行结果:

    在这里插入图片描述

    4.Bean实例化三种方式

    • 无参构造方法实例化(最常用,上面已用到过)
    • 工厂静态方法实例化
    • 工厂实例方法实例化

    工厂静态方法实例化:

    创建一个新的类,静态工厂类:
    在这里插入图片描述

    在静态方法类中写一个静态方法:

    package com.xy.factory;
    
    import com.xy.dao.UserDao;
    import com.xy.dao.impl.UserDaoImpl;
    
    public class StaticFactory {
        public static UserDao  getUserDao(){
            return new UserDaoImpl();
        }
    }
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11

    把配置文件的class修改为静态工厂:
    选中静态工厂,右键复制全限定名
    将配置文件的class属性替换掉
    class后再补充一个factory-method属性,指定静态工厂中的工厂方法

    在这里插入图片描述

        <bean id="userDao" class="com.xy.factory.StaticFactory" factory-method="getUserDao" ></bean>
    
    
    • 1
    • 2

    测试方法不用变,把close注释掉,运行结果:

    在这里插入图片描述
    工厂实例方法实例化:

    在factory文件内创建一个实例工厂:

    在这里插入图片描述

    在实例方法中创建方法:

    package com.xy.factory;
    
    import com.xy.dao.UserDao;
    import com.xy.dao.impl.UserDaoImpl;
    
    public class DynamicFactory {
            public  UserDao getUserDao(){
                return new UserDaoImpl();
            }
        
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11

    配置文件中需要先有把工厂对象Spring容器产生,再调工厂内部的方法:

        <bean id="factory" class="com.xy.factory.DynamicFactory" ></bean>
        <bean id="userDao" factory-bean="factory" factory-method="getUserDao"></bean>
    
    
    • 1
    • 2
    • 3

    运行结果:

    在这里插入图片描述

    5.Bean的依赖注入分析

    创建一个service接口:

    在这里插入图片描述

    在接口中简单写个方法:

    public interface UserService {
        public void save();
    }
    
    • 1
    • 2
    • 3

    创建接口实现:

    在这里插入图片描述

    实现接口方法:

    public class UserServiceImpl implements UserService {
    
        public void save() {
            ApplicationContext app=new ClassPathXmlApplicationContext("applicationContext.xml");
            UserDao userDao = (UserDao) app.getBean("userDao");
            userDao.save();
        }
    }
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9

    创建一个充当web层的测试:
    在这里插入图片描述
    写一个main方法:

    public class UserController {
        public static void main(String[] args) {
            UserService userService=new UserServiceImpl();
            userService.save();
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6

    运行结果:

    在这里插入图片描述

    在配置文件中配置:

        <bean id="userDao" class="com.xy.dao.impl.UserDaoImpl"  ></bean>
        <bean id="userService" class="com.xy.service.impl.UserServiceImpl"></bean>
    
    
    • 1
    • 2
    • 3

    让Spring容器产生:
    在UserController中删掉原来的,重写代码

    public class UserController {
        public static void main(String[] args) {
            ApplicationContext app=new ClassPathXmlApplicationContext("applicationContext.xml");
            UserService userService = (UserService) app.getBean("userService");
            userService.save();
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7

    搭建完环境后,运行结果:

    在这里插入图片描述

    上面搭建的环境有一定问题:
    在这里插入图片描述
    修改设想:
    在这里插入图片描述
    而在Spring容器内部将UserDao注入UserService的方法有两种:有参构造set方法

    6.Bean的依赖注入概念

    依赖注入(Dependency Injection):它是Spring框架核心IOC的具体实现。

    Service内需要Dao,即Service需要Dao的依赖注入

    在编写程序时,通过控制反转,把对象的创建交给了Spring,但是代码中不可能出现没有依赖的情况。IOC解耦只是降低他们的依赖关系,但不会消除。例如:业务层仍然会调用持久层的方法。

    那这种业务层和持久层的依赖关系,在使用Spring之后,就让Spring来维护了。
    简单的说,就是坐等框架把持久层对象传入业务层,而不用我们自己去获取。

    怎么将UserDao注入到UserService内部呢?

    • 构造方法
    • set方法

    构造方法:

    在UserServiceImpl中添加构造方法:

        private UserDao userDao;
    //    有参构造
        public UserServiceImpl(UserDao userDao) {
            this.userDao = userDao;
        }
    //    无参构造
        public UserServiceImpl() {
        }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8

    在配置文件中配置:

        <bean id="userDao" class="com.xy.dao.impl.UserDaoImpl"  ></bean>
           <bean id="userService" class="com.xy.service.impl.UserServiceImpl">
                <constructor-arg name="userDao" ref="userDao"></constructor-arg>
           </bean>
    <!--    name中的userDao是构造内部的参数名-->
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6

    运行结果:

    在这里插入图片描述

    set方法:

    在UserServiceImpl中添加set方法:

        private UserDao userDao;
    //    快捷键Ctrl+Insert+fn-->setter
        public void setUserDao(UserDao userDao) {
            this.userDao = userDao;
        }
    
        public void save() {
            userDao.save();
        }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9

    在配置文件中配置,告诉Spring容器要把容器中的Dao注入(通过set方法注入)给Service:

        <bean id="userDao" class="com.xy.dao.impl.UserDaoImpl"  ></bean>
        <bean id="userService" class="com.xy.service.impl.UserServiceImpl">
            <property name="userDao" ref="userDao"></property>
        </bean>
    <!--name指UserServiceImpl中的setUserDao方法set后面的UserDao,并把开头字母小写-->
    <!--ref中的userDao 是指id的UserDao-->
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6

    运行结果:

    在这里插入图片描述

    P命名空间注入:
    本质也是set方法注入,但比起上述的set方法注入更加方便,主要体现在配置文件中:

    首先,需要引入P命名空间:

    xmlns:p="http://www.springframework.org/schema/p"
    
    • 1

    在这里插入图片描述

    其次,修改注入方式:

        <bean id="userService" class="com.xy.service.impl.UserServiceImpl" p:userDao-ref="userDao"/>
    
    • 1

    运行结果:

    在这里插入图片描述

    7.Bean的依赖注入的数据类型

    上面的操作,都是注入的引用Bean,除了对象的引用可以注入,普通数据类型,集合等都可以在容器中进行注入。

    注入数据的三种数据类型:

    • 普通数据类型
    • 引用数据类型
    • 集合数据类型

    普通数据类型:

    在userDaoImpl中修改代码为:

    public class UserDaoImpl implements UserDao {
        private String username;
        private int age;
    //生成setter方法
        public void setUsername(String username) {
            this.username = username;
        }
    
        public void setAge(int age) {
            this.age = age;
        }
    
        public void save() {
            System.out.println(username+"===="+age);
            System.out.println("save running...");
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17

    配置文件中:

        <bean id="userDao" class="com.xy.dao.impl.UserDaoImpl"  >
            <property name="username" value="zhangsan"/>
            <property name="age" value="18"/>
        </bean>
    <!--    ref是指引用类型的注入,value是普通类型的注入,这里用value-->
    
    • 1
    • 2
    • 3
    • 4
    • 5

    在UserController测试类中执行代码,运行结果如下:
    在这里插入图片描述

    集合数据类型:

    创建一个User类:

    在这里插入图片描述
    在User类中写几个属性和get、set、tostring方法:

    public class User {
        
        private String name;
        private String addr;
    
        public String getName() {
            return name;
        }
    
        public void setName(String name) {
            this.name = name;
        }
    
        public String getAddr() {
            return addr;
        }
    
        public void setAddr(String addr) {
            this.addr = addr;
        }
        
        @Override
        public String toString() {
            return "User{" +
                    "name='" + name + '\'' +
                    ", addr='" + addr + '\'' +
                    '}';
        }
    
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30

    在UserDaoImpl中创建几个集合和set方法,以及输出,代码如下:

    //    集合类型
        private List<String> strList;
        private Map<String, User> userMap;
        private Properties properties;
    //三个set方法
        public void setStrList(List<String> strList) {
            this.strList = strList;
        }
    
        public void setUserMap(Map<String, User> userMap) {
            this.userMap = userMap;
        }
    
        public void setProperties(Properties properties) {
            this.properties = properties;
        }
        public void save() {
    //        System.out.println(username+"===="+age);
            System.out.println(strList);
            System.out.println(userMap);
            System.out.println(properties);
            System.out.println("save running...");
        }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23

    配置文件配置:

            <bean id="userDao" class="com.xy.dao.impl.UserDaoImpl"  >
            <property name="strList" >
                <list>
                    <value>aaa</value>
                    <value>bbb</value>
                    <value>ccc</value>
                </list>
            </property>
            <property name="userMap">
                <map>
                    <entry key="u1" value-ref="user1"></entry>
                    <entry key="u2" value-ref="user2"></entry>
                </map>
            </property>
            <property name="properties">
                <props>
                    <prop key="p1">ppp1</prop>
                    <prop key="p2">ppp2</prop>
                    <prop key="p3">ppp3</prop>
                </props>
            </property>
        </bean>
    
    
        <bean id="user1" class="com.xy.domain.User">
            <property name="name" value="tom"></property>
            <property name="addr" value="beijing"></property>
        </bean>
        <bean id="user2" class="com.xy.domain.User">
            <property name="name" value="lucy"></property>
            <property name="addr" value="tianjing"></property>
        </bean>
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32

    运行结果:

    在这里插入图片描述

    8.引入其他配置文件(分模块开发)

    实际开发中,Spring的配置内容非常多,这就导致Spring配置很繁杂且体积很大,所以,可以将部分配置拆解到其他配置文件中,而在Spring主配置文件中,而在Spring主配置文件通过import标签进行加载

    创建两个配置文件:
    在这里插入图片描述
    可以在主配置文件中通过import加载其他的配置文件:

        <import resource="applicationContext-user.xml"/>
        <import resource="applicationContext-product.xml"/>
    
    • 1
    • 2

    Spring的重点配置

    在这里插入图片描述

  • 相关阅读:
    2022年Android面试之Jetpack(AAC框架)篇
    golang入门
    【漏洞通告】CVE-2022-31690 Spring Security Oauth2 Client权限提升漏洞
    Educational Codeforces Round 138 (Rated for Div. 2)-赛后总结
    【Qt】边学边写之Qt教程(零基础)
    c++视觉----使用多边形包围轮廓
    rtsp转webrtc的其他几个项目
    k8s Metallb簡單入門示例
    内核对设备树的处理__dtb的修改原理
    Android NDK篇-C++虚函数、纯虚函数、完全纯虚函数与模板函数
  • 原文地址:https://blog.csdn.net/weixin_47678894/article/details/126077240