spring介绍

IOC(控制反转)

IOC介绍

IOC是什么

IOC(Inversion of Control 即控制反转)将对象交给容器管理
考虑:谁控制谁、控制什么?为了是反转?那些方面反转?
谁控制了谁?是容器控制了对象
控制什么?主要控制了外部资源及生命周期
由容器帮我们查找并注入依赖的对象,对象只能被动的接收依赖对象,依赖对象的获取被反转了。
在这里插入图片描述
spring中提供了一种IOC容器,来控制对象的创建,无论是你创建对象,处理对象之间的依赖关系,对象的创建时间还是对象的创建数量,都是spring提供IOC容器上配置对象的信息就可以了。

IOC能做什么

由IOC容器帮对象找相应的依赖思想并注入,并不是由对象主动去找
资源集中管理,实现资源的可配置和易管理
降低了使用资源双方的依赖程度,松耦合

Spring容器管理对象

1、maven管理依赖

引入spring核心依赖

    <properties>
        <spring.propety>4.1.7.RELEASE</spring.propety>
    </properties>
    <dependencies>
        <!--spring核心依赖-->
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-beans</artifactId>
            <version>${spring.propety}</version>
        </dependency>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-core</artifactId>
            <version>${spring.propety}</version>
        </dependency>
	<dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-context</artifactId>
            <version>${spring.propety}</version>
        </dependency>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-expression</artifactId>
            <version>${spring.propety}</version>
        </dependency>
    </dependencies>

2、给定容器的配置文件

将要交个容器的对象配置在该配置文件中,配置文件名Applicationcontext.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-3.0.xsd">
    </beans> 

3、IOC容器管理对象

将userInfo对象交给容器管理

    <!--使用bean标签来管理对象,id属性必填,用来表示对象,class属性给定对象的全限定名-->
    <bean id="userInfo" class="com.tulun.Spring.IOC.ObjectUse.UserInfo"/>

4、通过容器来获取对象

    public static void main(String[] args) {
        //获取IOC容器
        ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("springcontext1.xml");
        //获取对象
        UserInfo userInfo = (UserInfo)context.getBean("userInfo");
        System.out.println(userInfo);
    }

执行结果:
在这里插入图片描述
通过spring容器获取到了对象。

总结:Spring IOC操作
1、增加一个spring配置文件
2、解析xml的配置文件
3、BeanFactory工厂类
4、在工厂类中使用反射创建bean对象

Spring中IOC容器介绍

在spring中,spring-core包模块中包含IOC技术,IOC容器主要是为了管理对象,以及对象之间的依赖关系。
在这里插入图片描述
容器实现主要是BeanFactory接口和ApplicationContext接口,其继承关系如上图所示:
**BeanFactory:**是Spring IOC的最底层设计,提供了先进的配置机制,使得任何类型的对象配置成为可能。
ApplicationContext:ApplicationContext接口是继承自BeanFactory,实现了许多接口的扩展

BeanFactory的使用和ApplicationContext类似

BeanFactory factory = new xmlBeanFactory(new ClassPathResource("context.xml"));
fatory.getBean("XXX");

BeanFactory和ApplicationContext区别在于:BeanFactory的实现是按需创建,即第一次获取Bean才创建这个Bean,而ApplicationContext会一次性创建所有Bean。
BeanFactory是ApplicationContext父接口,ApplicationContext也提供了一个额外的功能,比如提供支持国际化、事件通知等,通常总是使用ApplicationContext,很少使用BeanFactory。
ApplicationContext接口常见实现类

  • ClassPathXmlApplicationContext

读取的xml配置文件放置在类路径下,优先考虑使用ClassPathXmlApplicationContext

//获取IOC容器
ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext(“springcontext1.xml”);

  • FileSystemXmlApplicationContext

读取的配置文件放置在文件系统路径下,则优先考虑使用FileSystemXmlApplicationContext

FileSystemXmlApplicationContext context1 = new FileSystemXmlApplicationContext(“c://applacation.xml”);

  • XMLWebApplicationContext

需要在web环境下读取配置信息XMLWebApplicationContext

Spring中Bean的实例化方式

spring中装配Bean的方式主要两种
基于xml配置形式
基于注解形式

1、基于XML配置方式装配bean

bean基于配置实例化方式有三种形式

  • 通过无参构造实例化
  • 通过静态功能方式实例化
  • 通过普通工厂实例化

无参构造方式实例化对象
通过类的全限定名找到类的位置,通过无参构造函数创建对象

<bean id="userInfo" class="com.tulun.Spring.IOC.ObjectUse.UserInfo"/>

:如果不指定构造函数,会生成默认的无参构造函数
如果指定有参构造,必须显性的指定一个无参构造函数,否则实例化对象就会抛出异常。
静态工厂方法实例化对象
首先,使用工厂的静态方法返回去一个对象

public class ObjectFactory {
    //提供静态的方法来获取UserInfo对象
    public static UserInfo getBean() {
        return new UserInfo();
    }
}

在配置文件中使用静态方法返回对象

    <!--
    通过静态工厂获取对象
    class属性指向工厂类全限定名
    factory-method属性是工厂类中获取对象的静态方法
    -->
    <bean id="userInfo1" class="com.tulun.Spring.IOC.IOCDemo.ObjectFactory" factory-method="getBean"/>

通过普通工厂方法实例化对象
通过工厂的非静态方法来得到一个对象

public class ObjectFactory {
    //提供普通方法来获取UserInfo对象
    public UserInfo getUserInfo() {
        return new UserInfo();
    }
}

在容器中使用工厂的非静态方法获取对象

    <!--
    通过普通工厂获取对象
    -->
    <!--创建工厂对象-->
    <bean id="factory" class="com.tulun.Spring.IOC.IOCDemo.ObjectFactory"/>
    <bean id="userInfo2" class="com.tulun.Spring.IOC.ObjectUse.UserInfo" factory-bean="factory" factory-method="getUserInfo"/>

2、使用注解方式装配Bean

通过Component注解标记类

@Component(value = "userInfo")
/**
 * 
 * 注解形式类似于配置中bean标签处理
 * 注解中value参数和bean标签中id是相同的作用
 * <bean id="userInfo" class="com.tulun.Spring.IOC.ObjectUse.UserInfo"/>
 */
public class UserInfo {
    private String name;
    private String sex;
    public UserInfo(){}
    } 

配置启动组件扫描注解

<?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 http://www.springframework.org/schema/context/spring-context.xsd">
       <!--开启注解扫描,指定的包路径或者类名:扫描类、方法、属性上是否有注解-->
    <context:component-scan base-package="com.tulun.Spring.IOC.ObjectUse"/>
    <!--扫描属性上的注解(不建议使用)-->
    <context:annotation-config></context:annotation-config>
</beans>

@Component注解是spring最早的标注类要交给容器管理的注解
还有以下注解:
@Component
@Controller 对Controller层类进行标注(主要接受处理URL)
@Service 对Service层的类进行标注(是进行业务处理)
@Repository 对DAO层实现类进行标注(和数据库打交道的类)
后三个都是Component注解衍生出来,功能是一样,可以互换,为了区分被注解标注的类在不同的业务层,使逻辑更清晰。

IOC中属性讲解:
Bean的作用范围@Scope
@Scope取值:
singleton:单例对象,默认是单例
prototype:多例对象

Spring中DI介绍

DI-Dependency Injection:依赖注入
bean对象中需要依赖一些其他组件或者对象,依赖关系由容器在运行时决定
两种方式处理依赖注入:
基于配置形式
基于注解形式

基于配置形式注入依赖

有参构造函数注入依赖

public class User {
    private String name;
    private UserInfo userInfo;
    public User(String name,UserInfo userInfo) {
        this.name = name;
        this.userInfo = userInfo;
    }

set方法注入依赖

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

bean配置:

    <bean id="user1" class="com.tulun.Spring.IOC.ObjectUse.User">
        <property name="name" value="hello"/>
    </bean>

其他类型数据介绍

public class User {
        private String name;
        private UserInfo userInfo;
        //list类型
        private List<String> lists;
        //map类型
        private Map<String,String> maps;
        //set类型
        private Set<String>  sets;
	public void setUserInfo(UserInfo userInfo) {
        this.userInfo = userInfo;
    }
	public void setLists(List <String> lists) {
        this.lists = lists;
    }
    public void setMaps(Map <String, String> maps) {
        this.maps = maps;
    }
    public void setSets(Set <String> sets) {
        this.sets = sets;
    }
<property name="lists" >
            <!--list属性注入-->
            <list >
                <value>张三</value>
                <value>李四</value>
            </list>
        </property>
        <property name="maps">
            <!--map类型数据注入-->
            <map>
                <entry key="k1" value="k11"/>
                <entry key="k2" value="k22"/>
            </map>
        </property>
        <property name="sets">
            <!--set类型数据注入-->
            <set>
                <value>set1</value>
                <value>set2</value>
            </set>
        </property>

基于注解形式注入依赖

@Value() //注入普通类型属性
@Autowired 注入对象类型
@Resource 注入对象类型
都是在属性上添加

@Component
public class User {
@Value("tulun") //针对普通类型的数据
        private String name;
         @Autowired  //注入对象类型
        @Resource  //注入对象类型
        private UserInfo userInfo;

循环依赖的问题:
如果是使用构造函数的方式注入依赖,有可能会无法解决循环依赖的问题。
解决问题:
对于会产生循环依赖的问题的这些类可以通过setter方法注入,可以解决循环依赖。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值