01.基于XML Spring--IOC

Spring概述

  • Spring的概念:
      Spring是分层的 Java SE/EE应用 full-stack 轻量级开源框架,以 IoC(Inverse Of Control: 反转控制)AOP(Aspect Oriented Programming:面向切面编程) 为内核,提供了展现层 Spring MVC 和持久层 Spring JDBC 以及业务层事务管理等众多的企业级应用技术,还能整合开源世界众多 著名的第三方框架和类库,逐渐成为使用最多的Java EE 企业应用开源框架
  • Spring的优势:
    1. 方便解耦,简化开发:
      通过 Spring提供的 IoC容器,可以将对象间的依赖关系交由 Spring进行控制,避免硬编码所造 成的过度程序耦合。用户也不必再为单例模式类、属性文件解析等这些很底层的需求编写代码,可以更专注于上层的应用
    2. AOP编程的支持 :
      通过 Spring的 AOP 功能,方便进行面向切面的编程,许多不容易用传统OOP 实现的功能可以通过 AOP 轻松应付
    3. 声明式事务的支持
      可以用非容器依:赖的编程方式进行几乎所有的测试工作,测试不再是昂贵的操作,而是随手可 做的事情
    4. 方便集成各种优秀框架:
      Spring可以降低各种框架的使用难度,提供了对各种优秀框架(Struts、Hibernate、Hessian、Quartz 等)的直接支持
    5. 降低 JavaEE API的使用难度:
      Spring对 JavaEE API(如 JDBC、JavaMail、远程调用等)进行了薄薄的封装层,使这些 API 的 使用难度大为降低
    6. Java源码是经典学习范例
      Spring的源代码设计精妙、结构清晰、匠心独用,处处体现着大师对Java 设计模式灵活运用以 及对 Java技术的高深造诣。它的源代码无意是 Java 技术的最佳实践的范例
  • Spring的体系结构
    在这里插入图片描述

耦合的概念

  • 耦合的概念:
      耦合性(Coupling),也叫耦合度,是对模块间关联程度的度量。耦合的强弱取决于模块间接口的复杂性、调用模块的方式以及通过界面传送数据的多少。模块间的耦合度是指模块之间的依赖关系,包括控制关系调用关系数据传递关系。模块间联系越多,其耦合性越强,同时表明其独立性越差( 降低耦合性,可以提高其独立 性)。耦合性存在于各个领域,而非软件设计中独有的,但是我们只讨论软件工程中的耦合。
      在软件工程中,耦合指的就是就是对象之间的依赖性。对象之间的耦合越高,维护成本越高。因此对象的设计应使类和构件之间的耦合最小。软件设计中通常用耦合度和内聚度作为衡量模块独立程度的标准。划分模块的一个准则就是高内聚低耦合
  • 耦合的分类:
    1. 内容耦合:
        当一个模块直接修改或操作另一个模块的数据时,或一个模块不通过正常入口而转入另 一个模块时,这样的耦合被称为内容耦合。内容耦合是最高程度的耦合,应该避免使用之。
    2. 公共耦合:
        两个或两个以上的模块共同引用一个全局数据项,这种耦合被称为公共耦合。在具有大 量公共耦合的结构中,确定究竟是哪个模块给全局变量赋了一个特定的值是十分困难的。
    3. 外部耦合:
        一组模块都访问同一全局简单变量而不是同一全局数据结构,而且不是通过参数表传 递该全局变量的信息,则称之为外部耦合。
    4. 控制耦合 :
        一个模块通过接口向另一个模块传递一个控制信号,接受信号的模块根据信号值而进 行适当的动作,这种耦合被称为控制耦合。
    5. 标记耦合 :
        若一个模块 A 通过接口向两个模块 B 和 C 传递一个公共参数,那么称模块 B 和 C 之间 存在一个标记耦合。
    6. 数据耦合:
        模块之间通过参数来传递数据,那么被称为数据耦合。数据耦合是最低的一种耦合形 式,系统中一般都存在这种类型的耦合,因为为了完成一些有意义的功能,往往需要将某些模块的输出数据作为另一些模块的输入数据。
    7. 非直接耦合:
        两个模块之间没有直接关系,它们之间的联系完全是通过主模块的控制和调用来实 现的。
  • 总结:
      耦合是影响软件复杂程度和设计质量的一个重要因素,在设计上我们应采用以下原则:如果模块间必须存在耦合,就尽量使用数据耦合,少用控制耦合,限制公共耦合的范围,尽量避免使用内容耦合
  • 耦合的具体案例1:
	public class JdbcDemo1 {
    public static void main(String[] args) throws  Exception{
        // 1.注册驱动
        DriverManager.registerDriver(new com.mysql.jdbc.Driver());
        // 2.获取连接
        Connection conn = DriverManager.getConnection("jdbc:mysql://localhost:3306/springlearn","root","123");
        // 3.获取操作数据库的预处理对象
        PreparedStatement pstm = conn.prepareStatement("select * from account");
        // 4.执行SQL,得到结果集
        ResultSet rs = pstm.executeQuery();
        // 5.遍历结果集
        while(rs.next()){
            System.out.println(rs.getString("name"));
        }
        // 6.释放资源
        rs.close();
        pstm.close();
        conn.close();
    }
}

  通过这个JDBC操作的案例,在第一步注册驱动是就新建了一个mysql驱动对象new com.mysql.jdbc.Driver(),当我们没有mysql驱动的jar包时,程序就会在编译时出错,若我们通过Class.forName(“com.mysql.jdbc.Driver”);来反射注册驱动,程序在编译时就不会出错,只会在运行时出错,进一步来分析,若我们要使用其他的数据库(比如 Oracle),就只能修改原码来修改数据库的驱动,这显然不是我们想要的。

  • 耦合的具体案例2:
public class AccountServiceImpl implements IAccountService {  
		private IAccountDao accountDao = new AccountDaoImpl(); 
	}

  案例2和案例1的问题差不多,但持久层的IAccountDao类没了,那么程序就会在编译时报错,这种编译期依赖关系,应该在我们开发中杜绝。我们需要优化代码解决。

工厂模式解耦

  • 方式:
      在实际开发中我们可以把三层的对象都使用配置文件配置起来,当启动服务器应用加载的时候,让一个类中的方法通过读取配置文件,把这些对象创建出来并存起来。在接下来的使用的时候,直接拿过来用就好了。 那么,这个读取配置文件,创建和获取三层对象的类就是工厂。 (简单来说就是用文件流的形式,把我们写在配置文件中的配置放到一个集合对象中,通过<key,value>的形式存储,到后面有方法有用到这些配置文件时,通过key调用这个类的get方法,get的方法会从集合里面找到这个配置value返回即可)
  • 创建bean.properties配置文件
accountService=com.itheima.service.impl.AccountServiceImpl
accountDao=com.itheima.dao.impl.AccountDaoImpl
  • 创建BeanFactory工厂
/**
 * 一个创建Bean对象的工厂
 *
 * Bean:在计算机英语中,有可重用组件的含义。
 * JavaBean:用java语言编写的可重用组件。
 *      javabean >  实体类
 *
 *   它就是创建我们的service和dao对象的。
 *
 *   第一个:需要一个配置文件来配置我们的service和dao
 *           配置的内容:唯一标识=全限定类名(key=value)
 *   第二个:通过读取配置文件中配置的内容,反射创建对象
 *
 *   我的配置文件可以是xml也可以是properties
 */
public class BeanFactory {
    //定义一个Properties对象
    private static Properties props;
    //定义一个Map,用于存放我们要创建的对象。我们把它称之为容器
    private static Map<String,Object> beans;
    //使用静态代码块为Properties对象赋值
    static {
        try {
            //实例化对象
            props = new Properties();
            //获取properties文件的流对象
            InputStream in = BeanFactory.class.getClassLoader().getResourceAsStream("bean.properties");
            props.load(in);
            //实例化容器
            beans = new HashMap<String,Object>();
            //取出配置文件中所有的Key
            Enumeration keys = props.keys();
            //遍历枚举
            while (keys.hasMoreElements()){
                //取出每个Key
                String key = keys.nextElement().toString();
                //根据key获取value
                String beanPath = props.getProperty(key);
                //反射创建对象
                Object value = Class.forName(beanPath).newInstance();
                //把key和value存入容器中
                beans.put(key,value);
            }
        }catch(Exception e){
            throw new ExceptionInInitializerError("初始化properties失败!");
        }
    }

    /**
     * 根据bean的名称获取对象
     * @param beanName
     * @return
     */
    public static Object getBean(String beanName){
        return beans.get(beanName);
    }
}
  • 需要用到对象时
private IAccountDao accountDao = (IAccountDao)BeanFactory.getBean("accountDao");

IoC的概念和作用

  • IoC的概念:
      控制反转(Inversion of Control,缩写为IoC),是面向对象编程中的一种设计原则,可以用来减低计算机代码之间的耦合度。其中最常见的方式叫做依赖注入(Dependency Injection,简称DI),还有一种方式叫“依赖查找”(Dependency Lookup)。通过控制反转,对象在被创建的时候,由一个调控系统内所有对象的外界实体将其所依赖的对象的引用传递给它。也可以说,依赖被注入到对象中。----百度百科
  • 我的理解:
      Spring帮我名创建一个容器(Map),我们把类放到里面,Spring会帮我们创建一个对象,当某个方法要用时,直接从容器里面查找对象就行,无需我们直接新建一个对象,从而降低代码之间的耦合
  • IoC的作用:
      最主要的作用就是降低代码的耦合
  • 图示(一开始是我们主动创建对象,现在是我们从工厂里拿已经创建好的对象)
    在这里插入图片描述

使用 Spring的 IOC解耦案例 (基于XML)

  • 依赖坐标
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-context</artifactId>
            <version>5.0.2.RELEASE</version>
        </dependency>
  • 编写dao层实现类
/**
 * IAccountDao实现类
 */
public class AccountDaoImpl implements IAccountDao {
    // 模拟存储用户
    public void saveAccount() {
        System.out.println("模拟存储用户,IAccountDao实现类");
    }
}
  • 编写业务层实现类
/**
 * 账户的业务层实现类
 */
public class AccountServiceImpl implements IAccountService {

    private IAccountDao accountDao ;

    public void setAccountDao(IAccountDao accountDao) {
        this.accountDao = accountDao;
    }

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

    public void  saveAccount(){
        accountDao.saveAccount();
    }
}
  • 编写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">

    <!--把对象的创建交给spring来管理-->
    <bean id="accountService" class="com.zsc.service.impl.AccountServiceImpl">
		<property name="accountDao" ref="accountDao"/>
	</bean>

    <bean id="accountDao" class="com.zsc.dao.impl.AccountDaoImpl"></bean>
</beans>
  • 测试
    public static void main(String[] args) {
        /**
         * ApplicationContext:读取配置文件后立即创建对象
         * BeanFactory:延时创建,什么时候用什么时候创建
         * ----------分割线-----------
         * ClassPathXmlApplicationContext:从类的根路径下加载配置文件,推荐这种
         * FileSystemXmlApplicationContext:从磁盘的任意有读取权限的位置加载配置文件
         * AnnotationConfigApplicationContext:当我们使用注解配置容器对象时,需要使用此类来创建 spring 容器。它用来读取注解
         */
        //1.获取核心容器对象
        ApplicationContext ac = new ClassPathXmlApplicationContext("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();
    }
}
  • 结果
    在这里插入图片描述

基于XML的Spring IoC配置

< bean >标签

  • 作用:
      用于配置对象让 spring 来创建的,默认情况下它调用的是类中的无参构造函数,如果没有无参构造函数则不能创建成功
  • 属性:
      id:对象的唯一标识符,用来获取对象
      class:指定类的全限定类名,用于反射创建对象,默认调用无参构造函数创建
       scope:指定对象的作用范围:
        singleton:默认值,单例模式,创建容器是创建,容器销毁时销毁
        prototype:多例模式,使用对象是新建对象,但长时间不用时java会自动将此销毁
        request :WEB 项目中,Spring 创建一个 Bean 的对象,将对象存入到 request 域中
        session :WEB 项目中,Spring 创建一个 Bean 的对象,将对象存入到 session 域中
        global session :WEB 项目中,应用在 Portlet 环境.如果没有 Portlet 环境那么 globalSession 相当于 session.
      init-method:指定类中初始化方法名称
      destory-method:指定类中销毁方法的名称

bean 的作用范围和生命周期

  • 单例对象:scope=“singleton”
      一个应用只有一个对象的实例。它的作用范围就是整个引用
      生命周期:
        对象出生:当应用加载,创建容器时,对象就被创建
        对象活着:只要容器在,对象一直活着
        对象死亡:当应用卸载,销毁容器时,对象就被销毁
  • 多例对象:scope=“prototype”
      每次访问对象时,都会重新创建对象实例
      生命周期:
        对象出生:当使用对象时,创建新的对象实例
        对象活着:只要对象在使用中,就一直活着
         对象死亡:当对象长时间不用时,被 java 的垃圾回收器回收

实例化 Bean 的三种方式

  • 使用默认无参构造函数
      它会根据默认无参构造函数来创建类对象。如果 bean 中没有默认无参构造函数,将会创建失败
	<bean id="accountService" class="com.zsc.service.impl.AccountServiceImpl"/>
  • Spring管理静态工厂-使用静态工厂的方法创建对象
/**
 * 模拟静态工厂,创建业务层实现类对象
 */
public class StaticFactory {
    public static IAccountService createAccountService(){
        return new AccountServiceImpl();
    }
}
    <!--
        使用StaticFactory类中的静态方法createAccountService创建对象,并存入容器
        属性:
            id:bean的id
            class:指定静态工厂的全限定类名
            factory-method:指定生产对象的静态方法
    -->
    <bean id="accountServiceStatic" class="com.zsc.factory.StaticFactory" factory-method="createAccountService"/>
  • spring管理实例工厂-使用实例工厂的方法创建对象
public class InstanceFactory {   
	public IAccountService createAccountService(){   
		return new AccountServiceImpl();  
	} 
}
    <!--
        先把工厂的创建交给spring,然后使用工厂的bean来调用里面所需的方法
        factory-bean:指定实例bean的id
        factory-method:指定实例工厂中创建对象的方法
    -->
    <bean id="staticFactory" class="com.zsc.factory.StaticFactory"/>
    <bean id="newByFactoryCommon" factory-bean="staticFactory" factory-method="createAccountServiceByCommon"/>

Spring 的依赖注入

  • 依赖注入的概念:
      依赖注入:Dependency Injection。(DI)它是 Spring 框架核心 ioc 的具体实现
      我们的程序在编写时,通过控制反转,把对象的创建交给了 Spring,但是代码中不可能出现没有依赖的情况,IoC 解耦只是降低他们的依赖关系,但不会消除。
      例如:我们的业务层仍会调用持久层的方法。 那这种业务层和持久层的依赖关系,在使用 spring 之后,就让 Spring 来维护了。 简单的说,就是坐等框架把持久层对象传入业务层,而不用我们自己去获取。
  • 构造函数注入:
      使用类中的构造函数,给成员变量赋值。注意,赋值的操作不是我们自己做的,而是通过配置的方式,让 spring 框架来为我们注入
  • 注意:
      当类原有给参数赋值、或者调用构造函数或其他方法进行复制、bean注入赋值,这时候最终的赋值为bean的值
/**
 * 账户的业务层实现类
 */
public class AccountServiceImpl 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);
    }
}
    <!--
        使用有参的构造函数创建有属性的对象<constructor-arg>标签
        属性:
            下面三选一,给谁赋值
            index:指定参数在构造函数参数列表的索引位置
            type:指定参数在构造函数中数据类型
            name:指定参数在构造函数的名称    常用
            ==============分割线============
            value:给指定的参数赋值,只能赋基本数据类型和String
            ref:给指定的参数赋值,赋其他bean类型
    -->
    <bean id="accountService" class="com.zsc.service.impl.AccountServiceImpl">
        <constructor-arg name="name" value="test"/>
        <constructor-arg name="age" value="21"/>
        <constructor-arg name="birthday" ref="date"/>
    </bean>
    <bean id="date" class="java.util.Date"/>
  • set 方法注入:
      在类中提供需要注入成员的 set 方法
/**
 * 账户的业务层实现类
 */
public class AccountServiceImpl2 implements IAccountService {

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

    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 setMyList(List<String> myList) {
        this.myList = myList;
    }
    public void setMyMap(Map<String, String> myMap) {
        this.myMap = myMap;
    }
    public void saveAccount() {
        System.out.println(name+","+age+","+birthday);
        System.out.println(myList);
        System.out.println(myMap);
    }
}
    <!--
        使用set方法的方式创建有属性的对象<property>,此方法使用较多
        属性:
            name:指定赋值的名称,方法set后面的名称
            ref:给指定的参数赋值,赋其他bean类型
            value:给指定的参数赋值,只能赋基本数据类型和String
        ===========上面的基本类、String、bean的赋值,下面是集合类型==========
            当集合的结构相同时,类型之间可以自动转换,选一种即可
            List结构:array,list,set
            Map结构:map,entry,props,prop
    -->
    <bean id="accountService" class="com.zsc.service.impl.AccountServiceImpl">
        <property name="name" value="test"/>
        <property name="age" value="21"/>
        <property name="birthday" ref="date"/>
        <property name="myList">
            <list>
                <value>AAA</value>
                <value>BBB</value>
                <value>CCC</value>
            </list>
        </property>
        <property name="myMap">
            <map>
                <!--两种写法的效果都是一样的-->
                <entry key="testA" value="AAA"></entry>
                <entry key="testB">
                    <value>BBB</value>
                </entry>
            </map>
        </property>
    </bean>
    <bean id="date" class="java.util.Date"/>
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值