[Spring]关于IOC控制反转,你应该掌握这些

[Spring]关于IOC控制反转,你应该掌握这些

本文是于2020-4-24重构,将文章中关于Spring的理念介绍移至:

你一直在用Spring,但你可能真的不懂它

本文为IOC的实操部分,介绍Spring的一些基本操作

Spring环境搭建

导入依赖包

  • 如果你不使用Maven,你应该导入以下依赖包(不推荐)

  • 如果你使用Maven,你应该在你的Maven配置文件中添加以上依赖包的坐标(推荐这种方式)

    spring的maven仓库

创建配置文件

配置文件内容:

<?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.2.xsd">
</beans>

通过容器获取(创建)bean

我们以Dao类作为演示对象,在Spring容器中指定该类,将其控制权转给Spring的容器

创建dao与daoimpl类:

package daoimpl;

import dao.UserDao;

public class UserDaoImpl implements UserDao {
    @Override
    public void save() {
        System.out.println("user被保存");
    }
}

在配置文件中配置Bean

  • 必须指定id,唯一标识
  • class就是bean的类的全路径

创建spring容器

import dao.UserDao;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

public class Main {

    public static void main(String[] args) {
        //创建spring容器,指定配置文件
        ApplicationContext context = new ClassPathXmlApplicationContext("ApplicationContext.xml");
        //从容器中获得bean
        UserDao userDao = (UserDao) context.getBean("userDao");
        //执行bean的方法
        userDao.save();
   }
}

以上演示创建bean的过程需要被创建的bean类有默认构造器

延迟加载

bean默认情况下,是非延迟加载的,即spring的容器创建的时候就把bean给创建出来了,我们getBean的时候直接从容器中去拿这个Bean就可以了

而延迟加载就是在创建容器的时候,先不创建Bean,用到的时候(getBean)再创建

是否延迟加载由lazy-init来控制,默认是false,如果变成true就在getBean的时候去创建bean(user)

设置非延迟加载,只需要在配置文件中添加如下参数

<bean id="userDao" class="daoimpl.UserDaoImpl" lazy-init="true"></bean>

指定Bean的作用域

默认情况下,bean都是单例的,是容器初始化的时候被创建的,就这么一份

Scope:singleton单例,prototype多例,默认使用singleton

设置非单例:

<bean id="userDao" class="daoimpl.UserDaoImpl" scope="prototype"></bean>

如果是singleton我们可以设置非延迟加载(容器初始化时创建bean)和延迟加载(getBean的时候才创建)方式创建bean

如果是prototype我们没得选择只能是延迟加载方式创建(getBean的时候才创建)

演示Bean的生命周期

我们通过自定义两个方法来验证

Init():是bean被创建的时候被调用,主要做一些准备工作

Destroy():是bean被销毁的时候被调用,做清理工作

Bean的生命周期和容器一致,容器创建bean就被创建,容器销毁bean就被销毁

在bean中定义两个方法:

package daoimpl;

import dao.UserDao;

public class UserDaoImpl implements UserDao {
    @Override
    public void save() {
        System.out.println("user被保存");
    }

    @Override
    public void init() {
        System.out.println("init方法被调用");
    }

    @Override
    public void destroy() {
        System.out.println("destroy方法被调用");
    }

}

在配置文件中指定两个方法:

    <bean id="userDao" class="daoimpl.UserDaoImpl" scope="prototype"
    init-method="init" destroy-method="destroy"
    ></bean>
  • init-method代表初始化执行
  • destroy-method代表bean销毁时执行

在主方法中添加关闭容器方法:

import dao.UserDao;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.AbstractApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
public class Main {

    public static void main(String[] args) {
        //创建spring容器
        ApplicationContext context = new ClassPathXmlApplicationContext("ApplicationContext.xml");
        //从容器中获得bean
        UserDao userDao = (UserDao) context.getBean("userDao");
        //执行bean的方法
        userDao.save();
        //关闭容器
        ((AbstractApplicationContext) context).close();
    }
}

依赖注入[配置文件]

创建UserServiceImpl类,类中需要有一个UserDao属性,UserServiceImpl必须提供set方法

package serviceimpl;

import dao.UserDao;
import service.UserDaoService;

public class UserDaoServiceImpl implements UserDaoService {

    private UserDao userDao ;

    //必须指定set方法
    public void setUserDao(UserDao userDao) {
        this.userDao = userDao;
        userDao.save();
    }
	//service中的方法
    @Override
    public void save() {
        userDao.save();
    }
}

在配置文件中配置Bean并添加属性:

    <bean id="userDao" class="daoimpl.UserDaoImpl" scope="prototype"
    ></bean>

    <bean id="userService" class="serviceimpl.UserDaoServiceImpl">
        <property name="userDao" ref="userDao"></property>
    </bean>

主方法调用:

import com.service.UserDaoService;

import org.springframework.context.ApplicationContext;
import org.springframework.context.support.AbstractApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;


public class Main {

    public static void main(String[] args) {
        //创建spring容器
        ApplicationContext context = new ClassPathXmlApplicationContext("ApplicationContext.xml");
        //从容器中获得bean
        UserDaoService service = (UserService) context.getBean("userDaoServiceImpl");
        //调用方法
        service.save(); //执行的是userDao的sava方法
    }
}

依赖注入[注解方式]

导入注解依赖包:

修改配置文件信息:开启注解驱动

<?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-3.2.xsd
            http://www.springframework.org/schema/context
           http://www.springframework.org/schema/context/spring-context-3.2.xsd"
>
    <!--开启注解驱动-->
    <context:annotation-config></context:annotation-config>


    <bean id="userDao" class="daoimpl.UserDaoImpl"></bean>
    <bean id="userDaoService" class="serviceimpl.UserDaoServiceImpl"></bean>

</beans>

@Resource加在属性上

在bean中定义要注入bean的属性,不需要提供set方法

注意:

@Resource注解并不是spring的注解,是javax下的注解import javax.annotation.Resource;

package serviceimpl;

import dao.UserDao;
import service.UserDaoService;

import javax.annotation.Resource;

public class UserDaoServiceImpl implements UserDaoService {

   @Resource
    private UserDao userDao ;
   
    @Override
    public void save() {
        userDao.save();
    }
}

流程:

不指定name,用变量名与ID匹配的匹配

指定name,则name与ID匹配

  • 如果name与id匹配不上,就按着变量名与class属性下的接口和实现类的关系来匹配

    如果存在一个接口有多个实现类的关系的时候,我们必须指定@Resource的name属性来指定到底注入哪一个bean

加在set方法上

在set方法上加@Resource注解

默认不指定@Resource的name的时候,bean的id和如果和左边的三个框内的任意一个名称能匹配上就能注入依赖,如果三个名字都匹配不上就会按着接口和实现类的关系来匹配,如果遇到两个实现类或以上就会报错,这时要求@Resource的name

如果一旦指定了@Resource的name如果name匹配不上就直接报错,不会再去按着类型去匹配

@Autowired加在属性上

@Autowired是按着接口和实现类的关系来匹配的,如果存在一个接口多个实现类的时候,我们必须要指定name来匹配,要结合@Qulifier的注解来指定bean的id

加在set方法上

依赖注入-注解扫描器管理(推荐)

实际项目如果很大就会出现大量配置,我们使用扫描器,告诉Spring,哪些类要作为Bean被容器接管

扫描器扫描带有@Controller,@Service, @Repository,@Component的类

@Controller:控制层的类

@Service:服务层的类

@Repository:数据层的类

@Component:无法分层的类上

以上注解标注的类的bean的id默认为类名(首字符小写)

添加配置信息:

配置扫描器,指定哪些包下的类纳入扫描器范围

<?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-3.2.xsd
            http://www.springframework.org/schema/context
           http://www.springframework.org/schema/context/spring-context-3.2.xsd"
>
    
    <!--注解扫描器,容器创建时会将以下配置的包下的带有相关注解的类创建出bean-->
    <context:component-scan base-package="com"></context:component-scan>

</beans>


添加@Repository的类

package com.daoimpl;

import com.dao.UserDao;
import org.springframework.stereotype.Repository;

@Repository
public class UserDaoImpl implements UserDao {
    @Override
    public void save() {
        System.out.println("user被保存");
    }

}

添加@service的类

package com.serviceimpl;

import com.dao.UserDao;
import com.service.UserDaoService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Service;

import javax.annotation.Resource;

@Service
public class UserDaoServiceImpl implements UserDaoService {

    
    @Autowired
    @Qualifier("userDaoImpl") //选择要注入的类的类名(首字母小写)
    private UserDao userDao ;

    @Override
    public void save() {
        userDao.save();
    }
}

主函数调用:

import com.service.UserDaoService;

import org.springframework.context.ApplicationContext;
import org.springframework.context.support.AbstractApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;


public class Main {

    public static void main(String[] args) {
        //创建spring容器
        ApplicationContext context = new ClassPathXmlApplicationContext("ApplicationContext.xml");
        //从容器中获得bean
        UserDaoService service = (UserDaoService) context.getBean("userDaoServiceImpl");
        //调用方法
        service.save();
    }
}

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Tephra旨在构建一个稳定、高效、易于集群、快速扩展的JavaEE开发框架。目前,Tephra已经具备了以下特性: 提供类级别的热更新,但仅建议在需要快速修正严重BUG、并且无法立即进行全更新时使用。 提供全冗余方式的缓存,自动在每个节点间同步缓存数据,而每个节点都仅从本地内存中获取缓存数据,从而提供高效的执行效率,并且当部分节点宕机时仍旧能正常提供服务。当然,也允许使用Redis提供统一的中心节点缓存。此特性可用于多节点负载时提供不停服更新。 提供数据库读写分离、及读负载均衡,并且允许配置多个不同的数据库,甚至允许在运行期间增加新的数据库配置,并自动映射ORM。允许执行标准的SQL或存储过,同时提供了一个简单、轻量的ORM工具,并集成Hibernate、MyBatis为复杂ORM需求提供支持。 提供MongoDB工具实现对NoSQL的支持,支持负载均衡。 提供轻量级、快速响应的控制器,允许设置最大并发峰值,以确保在突如其来的并发攻击后能继续正常提供服务;也允许设置单IP最大并发量,确保小量IP并发攻击时仍能正常提供服务。允许发布为HTTP(S)、WebSocket、Socket服务。 提供JavaScript脚本引擎支持,允许JavaScript与JavaBean自由交互,并且可以发布JavaScript为服务;以及使用Javacript作为复杂规则验证器。 提供Hadoop存取支持。 模块化开发,使用注解简化配置,增强代码可读性与维护性。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值