spring第一天学习

目录

1. spring概述

1.1 spring的体系结构

2. IoC 的概念和作用

2.1 什么是程序的耦合

2.2 控制反转-IOC(Inversion Of Control)

3. spring基于 xml的IOC细节

3.1. spring中工厂的类结构图

3.2. BeanFactory 和 ApplicationContext 的区别

3.3.ApplicationContext 接口的实现类

3.4.IOC 中 bean 标签和管理对象细节

3.4.1 bean 标签

3.4.2 bean 的作用范围和生命周期

3.4.3 实例化 Bean 的三种方式

3.5 spring 的依赖注入

3.5.1 依赖注入的概念

3.5.2 构造函数注入

3.5.3 set 方法注入

3.5.4 注入集合属性


 

1. spring概述

1.1 spring的体系结构

2. IoC 的概念和作用

2.1 什么是程序的耦合

耦合性(Coupling),也叫耦合度,是对模块间关联程度的度量。耦合的强弱取决于模块间接口的复杂性、调 用模块的方式以及通过界面传送数据的多少。模块间的耦合度是指模块之间的依赖关系,包括控制关系、调用关 系、数据传递关系。模块间联系越多,其耦合性越强,同时表明其独立性越差( 降低耦合性,可以提高其独立 性)。耦合性存在于各个领域,而非软件设计中独有的,但是我们只讨论软件工程中的耦合。

在软件工程中,耦合指的就是就是对象之间的依赖性。对象之间的耦合越高,维护成本越高。因此对象的设计 应使类和构件之间的耦合最小。软件设计中通常用耦合度和内聚度作为衡量模块独立程度的标准。划分模块的准则就是高内聚低耦合。

 

总结:
 

耦合是影响软件复杂程度和设计质量的一个重要因素,在设计上我们应采用以下原则:如果模块间必须 存在耦合,就尽量使用数据耦合,少用控制耦合,限制公共耦合的范围,尽量避免使用内容耦合。

 
内聚与耦合
 

内聚标志一个模块内各个元素彼此结合的紧密程度,它是信息隐蔽和局部化概念的自然扩展。内聚是从 功能角度来度量模块内的联系,一个好的内聚模块应当恰好做一件事。它描述的是模块内的功能联系。耦合是软件结构中各模块之间相互连接的一种度量,耦合强弱取决于模块间接口的复杂程度、进入或访问一个模块的点以及通 过接口的数据。 程序讲究的是低耦合,高内聚。就是同一个模块内的各个元素之间要高度紧密,但是各个模块之 间的相互依存度却要不那么紧密。

内聚和耦合是密切相关的,同其他模块存在高耦合的模块意味着低内聚,而高内聚的模块意味着该模块同其他模块之间是低耦合。在进行软件设计时,应力争做到高内聚,低耦合

这里说的耦合度高,咱们在代码中可以理解为一个类中调用了另一个类,也就是创建了别的类的对象,使得模块间的联系紧了。spring框架很好的解决了这个问题。

2.2 控制反转-IOC(Inversion Of Control)

先解释一下什么是工厂:

工厂就是负责给我们从容器中获取指定对象的类。这时候我们获取对象的方式发生了改变。

原来: 我们在获取对象时,都是采用 new 的方式。是主动的。

 

现在: 我们获取对象时,同时跟工厂要,有工厂为我们查找或者创建对象。是被动的。

 

               

 

明确 ioc 的作用: 削减计算机程序的耦合(解除我们代码中的依赖关系)

 

关于IOC和DI的进一步解释,可参考另一篇博客:

https://blog.csdn.net/hello_cmy/article/details/103159816

3. spring基于 xml的IOC细节

3.1. spring中工厂的类结构图

3.2. BeanFactory ApplicationContext 的区别

BeanFactory 才是 Spring 容器中的顶层接口。
ApplicationContext 是它的子接口。
BeanFactory ApplicationContext 的区别:
         创建对象的时间点不一样。
                     ApplicationContext :只要一读取配置文件,默认情况下就会创建对象。
                     BeanFactory :什么使用什么时候创建对象。

3.3.ApplicationContext 接口的实现类

ClassPathXmlApplicationContext
                它是从类的根路径下加载配置文件      推荐使用这种
FileSystemXmlApplicationContext
                它是从磁盘路径上加载配置文件,配置文件可以在磁盘的任意位置。
AnnotationConfigApplicationContext:
                当我们使用注解配置容器对象时,需要使用此类来创建 spring 容器。它用来读取注解。
 

3.4.IOC bean 标签和管理对象细节

3.4.1 bean 标签

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

3.4.2 bean 的作用范围和生命周期

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

3.4.3 实例化 Bean 的三种方式

第一种方式:使用默认无参构造函数
<!--在默认情况下:
    它会根据默认无参构造函数来创建类对象。如果 bean 中没有默认无参构造函数,将会创建失败。--> 

<bean id="accountService" class="com.itheima.service.impl.AccountServiceImpl"/>
第二种方式:spring 管理静态工厂-使用静态工厂的方法创建对象
/**
* 模拟一个静态工厂,创建业务层实现类
*/
public class StaticFactory {
    public static IAccountService createAccountService(){
        return new AccountServiceImpl();
} }

<!-- 此种方式是:
    使用 StaticFactory 类中的静态方法 createAccountService 创建对象,并存入 spring 容器
    id 属性:指定 bean 的 id,用于从容器中获取
    class 属性:指定静态工厂的全限定类名
    factory-method 属性:指定生产对象的静态方法
--> 

<bean id="accountService"
 class="com.itheima.factory.StaticFactory"
 factory-method="createAccountService"></bean>
第三种方式:spring 管理实例工厂-使用实例工厂的方法创建对象
/**
* 模拟一个实例工厂,创建业务层实现类
* 此工厂创建对象,必须现有工厂实例对象,再调用方法
*/
public class InstanceFactory {
    public IAccountService createAccountService(){
        return new AccountServiceImpl();
} }
<!-- 此种方式是:
    先把工厂的创建交给 spring 来管理。
    然后在使用工厂的 bean 来调用里面的方法
    factory-bean 属性:用于指定实例工厂 bean 的 id。
    factory-method 属性:用于指定实例工厂中创建对象的方法。
--> 

<bean id="instancFactory" class="com.itheima.factory.InstanceFactory"></bean> 

<bean id="accountService"
 factory-bean="instancFactory"
 factory-method="createAccountService"></bean>

3.5 spring 的依赖注入

3.5.1 依赖注入的概念

依赖注入:Dependency Injection。它是 spring 框架核心 ioc 的具体实现。
我们的程序在编写时,通过控制反转,把对象的创建交给了 spring,但是代码中不可能出现没有依赖的情况。
ioc 解耦只是降低他们的依赖关系,但不会消除。例如:我们的业务层仍会调用持久层的方法。
那这种业务层和持久层的依赖关系,在使用 spring 之后,就让 spring 来维护了。
简单的说,就是坐等框架把持久层对象传入业务层,而不用我们自己去获取。

3.5.2 构造函数注入

顾名思义,就是使用类中的构造函数,给成员变量赋值。
注意,赋值的操作不是我们自己做的,而是通过配置
的方式,让 spring 框架来为我们注入。具体代码如下:
/**
*/
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; 
    }

    @Override
    public void saveAccount() {
        System.out.println(name+","+age+","+birthday);
    } 
}
<!-- 使用构造函数的方式,给 service 中的属性传值
    要求:
        类中需要提供一个对应参数列表的构造函数。
        涉及的标签:
            constructor-arg
                属性:
                    index:指定参数在构造函数参数列表的索引位置
                    type:指定参数在构造函数中的数据类型
                    name:指定参数在构造函数中的名称 用这个找给谁赋值
            =======上面三个都是找给谁赋值,下面两个指的是赋什么值的==============
                    value:它能赋的值是基本数据类型和 String 类型
                    ref:它能赋的值是其他 bean 类型,也就是说,必须得是在配置文件中配置过的 bean
-->


<bean id="accountService" class="com.itheima.service.impl.AccountServiceImpl">                         
    <constructor-arg name="name" value="张三"></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>

3.5.3 set 方法注入

顾名思义,就是在类中提供需要注入成员的 set 方法。具体代码如下:

public class AccountServiceImpl 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; 
    }

    @Override
    public void saveAccount() {
        System.out.println(name+","+age+","+birthday);
      } 
 }
<!-- 通过配置文件给 bean 中的属性传值:使用 set 方法的方式
    涉及的标签:
        property
            属性:
                name:找的是类中 set 方法后面的部分
                ref:给属性赋值是其他 bean 类型的
                value:给属性赋值是基本数据类型和 string 类型的
            实际开发中,此种方式用的较多。
--> 

<bean id="accountService" class="com.itheima.service.impl.AccountServiceImpl"> 
    <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>

还有一个是p命名空间,这里就不介绍了。它的本质还是set方法。

3.5.4 注入集合属性

顾名思义,就是给类中的集合成员传值,它用的也是set方法注入的方式,只不过变量的数据类型都是集合。
我们这里介绍注入数组,List,Set,Map,Properties。具体代码如下:

public class AccountServiceImpl 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;
    }

    @Override
    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);
    }
<!-- 注入集合数据
    List 结构的:
        array,list,set
    Map 结构的
        map,entry,props,prop
-->
    <bean id="accountService" class="com.itheima.service.impl.AccountServiceImpl">
        <!-- 在注入集合数据时,只要结构相同,标签可以互换 -->
        <!-- 给数组注入数据 -->
        <property name="myStrs">
            <set>
                <value>AAA</value>
                <value>BBB</value>
                <value>CCC</value>
            </set>
        </property>
        <!-- 注入 list 集合数据 -->
        <property name="myList">
            <array>
                <value>AAA</value>
                <value>BBB</value>
                <value>CCC</value>
            </array>
        </property>
        <!-- 注入 set 集合数据 -->
        <property name="mySet">
            <list>
                <value>AAA</value>
                <value>BBB</value>
                <value>CCC</value>
            </list>
        </property>
        <!-- 注入 Map 数据 -->
        <property name="myMap">
            <props>
                <prop key="testA">aaa</prop>
                <prop key="testB">bbb</prop>
            </props>
        </property>
        <!-- 注入 properties 数据 -->
        <property name="myProps">
            <map>
                <entry key="testA" value="aaa"></entry>
                <entry key="testB">
                    <value>bbb</value>
                </entry>
            </map>
        </property>
    </bean>

 

 

 

 

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值