java框架SSM学习——用XML实现IoC操作

在进行IoC操作的时候,要清楚的了解到控制反转的原理。而在代码中去使用,是为什么干什么?是为了降低耦合性,那如何去降低耦合性呢?其实如果过多的去自己新建对象,那么类与类之间的耦合性就会变强,那么IoC解决的就是让spring去帮我们新建对象,而我们只要通过xml/注解的方式去告诉spring,我们需要什么对象。

案例一:无DI的Ioc操作(XML)

首先我们先用在上一篇的案例的代码,我们稍微修改一下,因为避免有人没有看,我还是将代码贴出来。

新建IAccountDao接口

package com.dao;

//账户的持久层接口
public interface IAccountDao {
    //模拟保存账户
    void saveAccount();
}

新建IAccountDao接口实现类

package com.dao.impl;

import com.dao.IAccountDao;

//账户的持久层实现类
public class AccountDaoImpl  implements IAccountDao {
    public void saveAccount() {
        System.out.println("保存成功");
    }
}

新建IAccountService接口

package com.service;

//账户业务层的接口
public interface IAccountService {
    //模拟保存账户
    void saveAccount();
}

新建IAccountService接口实现类

package com.service.impl;

import com.dao.IAccountDao;
import com.dao.impl.AccountDaoImpl;
import com.service.IAccountService;

//账户的业务层实现类
public class AccountServiceImpl implements IAccountService {

    private IAccountDao accountDao = new AccountDaoImpl();

    public AccountServiceImpl(){
        System.out.println("对象创建成功");
    }

    public void saveAccount() {
        accountDao.saveAccount();
    }
}

新建bean.xml配置文件

注意,xml里的约束需要你自己去找,进入spring.io网站中,按ctrl+f在其中输入xmlns,显示的约束即是我们需要的:

<?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
        https://www.springframework.org/schema/beans/spring-beans.xsd">

然后将需要创建的对象通过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
        https://www.springframework.org/schema/beans/spring-beans.xsd">
    <!--把对象的创建交给spring来管理-->
    <bean id="accountService" class="com.service.impl.AccountServiceImpl"></bean>
    <bean id="accountDao" class="com.dao.impl.AccountDaoImpl"></bean>
</beans>

在这里,我们需要创建dao层的对象去操作方法,而也需要sevice层的对象去调用dao层对象。所以我们需要创建两个实现类的对象,以上的配置即可让spring自动创建。

测试类

package com.ui;

import com.dao.IAccountDao;
import com.service.IAccountService;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

//模拟一个表现层,用于调用业务层
//核心容器的两个接口引发出的问题
//ApplicationContext:(单例对象适用)   实际开发中采用此接口
// 它在构建核心容器时,创建对象采取的策略是采用立即加载的方式。也就是说,只要一读完配置文件马上就创建配置文件中配置的对象
//BeanFactory:(多例对象适用)
// 它在构建核心容器时,创建对象才去的策略是采用延迟加载的方式。也就是说,什么时候根据id获取对象了,什么时候才真正的创建对象
public class Client {
    public static void main(String[] args) {
        //获取spring容器的Ioc的核心容器,并根据id获取对象
        //1.获取核心容器对象
        ApplicationContext ac = new ClassPathXmlApplicationContext("bean.xml");
        //ApplicationContext ac = new FileSystemXmlApplicationContext("C:\\Users\\0hasee\\Desktop\\bean.xml");
        //2.根据id获取bean对象
        IAccountService as = (IAccountService)ac.getBean("accountService");
        IAccountDao adao = ac.getBean("accountDao" , IAccountDao.class);
        System.out.println(as);
        System.out.println(adao);
        as.saveAccount();
    }
}
注意:IoC的用法,首先需要获取装有我们配置对象的IoC容器通过以下代码:
 //方法一(它可以加载类路径下的配置文件,要求配置文件必须在类路径下,不在的加载不了。(实际开发中更常用))
 ApplicationContext ac = new ClassPathXmlApplicationContext("bean.xml");
 //方法二(它可以加载磁盘任意路径下的配置文件(必须要有访问权限))
 ApplicationContext ac = new FileSystemXmlApplicationContext("C:\\Users\\0hasee\\Desktop\\bean.xml");
 //方法三(它是用于读取注解创建容器的 , 在此不作过多解释)
 AnnotationConfigApplicationContext

普及知识点:
①因为Spring IoC容器的设计主要是基于BeanFactroyApplicationContext两个接口,其中ApplicationContextBeanFactory的子接口之一,而BeanFactory是Spring IoC最底层的接口,而ApplicationContext是其高级接口之一,并且对BeanFactory功能做了许多有用的扩展,所以我们都会使用ApplicationContext作为IoC容器。
ClassPathXmlApplicationContextApplicationContext的一个子类,我们可以使用ClassPathXmlApplicationContext容器就可以将Spring IoC容器初始化,然后我们就能从IoC容器中获取需要的资源了。对于FileSystemXmlApplicationContext也是其子类之一。

然后我们利用容器中的getBean()方法来获取对象
 //2.根据id获取bean对象
 IAccountService as = (IAccountService)ac.getBean("accountService");

在第一篇中我们需要自己new来新建AccountServiceImpl的对象来调用里面的方法,而现在我交给spring来新建,我只要告诉spring,我要拿的对象的id是什么即可。这样我们就实现了简单的spring IoC操作。

案例二:有DI的Ioc操作(XML)

在看完上一篇的标题,估计又有人要懵了,DI又是什么?
DI:Dependency Injection 中文名:依赖注入 ,那么注入什么?是创建此对象时,需要的的外部资源注入,等下以案例来讲解会更加的好。
依赖注入的三种方式:构造器注入、set方法注入(常用)、接口注入(了解)

还是利用前面那个案例的IAccountService接口

package com.service;

//账户业务层的接口
public interface IAccountService {
    //模拟保存账户
    void saveAccount();
}

新建IAccountService接口的实现类1(构造器注入)

package com.service.impl;

import com.service.IAccountService;

import java.util.Date;

//账户的业务层实现类
public class AccountServiceImpl1 implements IAccountService {

    //如果是经常变化的数据,并不适用于注入的方式
    private String name;
    private Integer age;
    private Date birthday;

    public AccountServiceImpl(String name , Integer age , Date birthday){
        this.name = name;
        this.age = age;
        this.birthday = birthday;
    }

    public void saveAccount() {
        System.out.println("service中的saveAccount方法执行了" + name + "," + age + "," + birthday);
    }
}

新建IAccountService接口的实现类2(set方法注入)

package com.service.impl;

import com.service.IAccountService;

import java.util.Date;

//账户的业务层实现类
public class AccountServiceImpl2 implements IAccountService {

    //如果是经常变化的数据,并不适用于注入的方式
    private String name;
    private Integer age;
    private Date birthday;

    public void setName(String name) {
        this.name = name;
    }

    public void setAge(Integer age) {
        this.age = age;
    }

    public void setBirthday(Date birthday) {
        this.birthday = birthday;
    }

    public void saveAccount() {
        System.out.println("service中的saveAccount方法执行了" + name + "," + age + "," + birthday);
    }
}

新建IAccountService接口的实现类3(复杂类型注入)

package com.service.impl;

import com.service.IAccountService;

import java.util.*;

//账户的业务层实现类
public class AccountServiceImpl3 implements IAccountService {

    private String myStrs[];
    private List<String> myList;
    private Set<String> mySet;
    private Map<String , String> myMap;
    private Properties myProps;

    public void setMyStrs(String[] myStrs) {
        this.myStrs = myStrs;
    }

    public void setMyList(List<String> myList) {
        this.myList = myList;
    }

    public void setMySet(Set<String> mySet) {
        this.mySet = mySet;
    }

    public void setMyMap(Map<String, String> myMap) {
        this.myMap = myMap;
    }

    public void setMyProps(Properties myProps) {
        this.myProps = myProps;
    }

    public void saveAccount() {
        System.out.println(Arrays.toString(myStrs));
        System.out.println(myList);
        System.out.println(mySet);
        System.out.println(myMap);
        System.out.println(myProps);
    }
}

bean.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
        https://www.springframework.org/schema/beans/spring-beans.xsd">
    <!--构造函数注入:-->
    <bean id="accountService" class="com.service.impl.AccountServiceImpl">
        <constructor-arg name="name" value="test"></constructor-arg>
        <constructor-arg name="age" value="18"></constructor-arg>
        <constructor-arg name="birthday" ref="now"></constructor-arg>
    </bean>

    <bean id="now" class="java.util.Date"></bean>

    <!--set方法注入 (更常用)-->
    <bean id="accountService2" class="com.service.impl.AccountServiceImpl2">
        <property name="name" value="Test"></property>
        <property name="age" value="21"></property>
        <property name="birthday" ref="now"></property>
    </bean>

    <!--复杂类型的注入/集合类型的注入-->
    <bean id="accountService3" class="com.service.impl.AccountServiceImpl3">
        <property name="myStrs">
            <array>
                <value>aaa</value>
                <value>bbb</value>
                <value>ccc</value>
            </array>
        </property>

        <property name="myList">
            <list>
                <value>aaa</value>
                <value>bbb</value>
                <value>ccc</value>
            </list>
        </property>

        <property name="mySet">
            <set>
                <value>aaa</value>
                <value>bbb</value>
                <value>ccc</value>
            </set>
        </property>

        <property name="myMap">
            <map>
                <entry key="testA" value="aaa"></entry>
                <entry key="testB">
                    <value>bbb</value>
                </entry>
            </map>
        </property>

        <property name="myProps">
            <props>
                <prop key="testC">ccc</prop>
                <prop key="testD">ddd</prop>
            </props>
        </property>
    </bean>
</beans>

测试类

package com.itheima.ui;

import com.service.IAccountService;
import com.service.impl.AccountServiceImpl;
import javafx.application.Application;
import org.springframework.beans.factory.BeanFactory;
import org.springframework.beans.factory.xml.XmlBeanFactory;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
import org.springframework.context.support.FileSystemXmlApplicationContext;
import org.springframework.core.io.ClassPathResource;
import org.springframework.core.io.Resource;
public class Client {
    public static void main(String[] args) {
//      获取spring容器的Ioc的核心容器,并根据id获取对象
//      1.获取核心容器对象
//        ApplicationContext ac = new FileSystemXmlApplicationContext("C:\\Users\\0hasee\\Desktop\\bean.xml");
          ApplicationContext ac = new ClassPathXmlApplicationContext("bean.xml");
//      2.根据id获取bean对象
        IAccountService as1 = (IAccountService)ac.getBean("accountService1");
        as1.saveAccount();

        IAccountService as2 = (IAccountService)ac.getBean("accountService2");
        as2.saveAccount();

        IAccountService as3 = (IAccountService)ac.getBean("accountService3");
        as3.saveAccount();
    }
}

构造器注入测试结果:

在这里插入图片描述
我们再来看看在bean.xml中的配置:

<!--构造函数注入:-->
    <bean id="accountService" class="com.service.impl.AccountServiceImpl">
        <constructor-arg name="name" value="test"></constructor-arg>
        <constructor-arg name="age" value="18"></constructor-arg>
        <constructor-arg name="birthday" ref="now"></constructor-arg>
    </bean>

    <bean id="now" class="java.util.Date"></bean>

在AccountServiceImpl1中可以看到,有一个有参的构造器,而我们如果要新建这个类的对象,并不能使用默认的构造器,所以我们一定要满足有参构造器中的参数的要求。所以我们在告诉spring新建这个对象的时候,不给spring足够的参数,那么就会报错,为什么?因为当创建这个类对象的时候,就要使用有参构造器去创建,而参数也需要你去告诉spring,而这些参数就是用这个标签去告诉spring,这就是依赖参数的注入才能创建对象,所以这叫做构造器注入。

构造器注入总结:

使用的标签:constructor-arg
标签出现的位置:bean标签的内部
标签中的属性:
type:用于指定要注入的数据的数据类型,该数据类型也是构造函数中某个或某些参数的类型
index:用于指定要注入的数据给构造函数中指定索引位置的参数赋值,索引的位置是从0开始
name:用于指定给构造函数中指定名称测参数赋值(常用的)
value:用于提供基本类型和String类型的数据
ref:用于指定其他的bean类型数据。指得就是在spring的Ioc核心容器中出现过的bean对象

    优势:
        在获取bean对象时,注入数据是必须的操作,否则对象无法创建成功
    弊端:
        改变了bean对象的实例化方式,使我们在创建对象时,如果用不到这些数据,也必须提供

set方法注入测试结果:

在这里插入图片描述
我们再来看看bean.xml中的配置:

<bean id="accountService2" class="com.itheima.service.impl.AccountServiceImpl2">
        <property name="name" value="Test"></property>
        <property name="age" value="21"></property>
        <property name="birthday" ref="now"></property>
</bean>
<bean id="now" class="java.util.Date"></bean>

在AccountServiceImpl2中可以看到,我们用的是set方法去将需要的数据注入,所以叫做set方法注入,这个方法是最常用的,
set方法注入总结:
涉及的标签:property
出现的位置:bean标签的内部
标签的属性:
name:用于指定注入时所调用的set方法名称
value:用于提供基本类型和String类型的数据
ref:用于指定其他的bean类型数据。指得就是在spring的Ioc核心容器中出现过的bean对象

	  优势:
        创建对象时没有明确的限制,可以直接使用默认构造函数
      弊端:
        如果有某个成员必须有值,则获取对象是有可能set方法没有执行

复杂类型的注入/集合类型的注入测试结果:

在这里插入图片描述
我们再来看看bean.xml中的配置:

<!--复杂类型的注入/集合类型的注入-->
    <bean id="accountService3" class="com.service.impl.AccountServiceImpl3">
        <property name="myStrs">
            <array>
                <value>aaa</value>
                <value>bbb</value>
                <value>ccc</value>
            </array>
        </property>

        <property name="myList">
            <list>
                <value>aaa</value>
                <value>bbb</value>
                <value>ccc</value>
            </list>
        </property>

        <property name="mySet">
            <set>
                <value>aaa</value>
                <value>bbb</value>
                <value>ccc</value>
            </set>
        </property>

        <property name="myMap">
            <map>
                <entry key="testA" value="aaa"></entry>
                <entry key="testB">
                    <value>bbb</value>
                </entry>
            </map>
        </property>

        <property name="myProps">
            <props>
                <prop key="testC">ccc</prop>
                <prop key="testD">ddd</prop>
            </props>
        </property>
    </bean>

复杂类型的注入/集合类型的注入
用于给List结构集合注入的标签:
list array set
用于给Map结构集合注入的标签:
map props
结构相同,标签可以互换

接口注入

接口注入模式因为历史较为悠久,在很多容器中都已经得到应用。但由于其在灵活性、易用性上不如其他两种注入模式,所以在这也就不介绍了,知道有这个依赖注入的方式即可。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值