Spring-IOC(控制反转)Set注入实现--01 入门(控制反转基本实现原理)
目录
Set方法注入
在service层利用set方法给dao层接口进行赋值,把具体的dao层对象通过set方法赋值 给dao层接口。这也是IOC(控制反转)的基本实现方式,也是后面要分享的通过xml去对类进行实例化和赋值的基本原理。
配置文件(提前解密,这是下篇正式进入IOCxml配置文件方式使用的配置文件格式)
<?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来创建对象,在spring这些都称为Bean-->
    <bean id="hello" class="com.kuang.pojo.Hello" >
        <property name="name" value="ksh"/> <!--给属性赋值-->
    </bean>
</beans>
可以从上面的配置文件里看到,每一个bean都是一个对象。 配置文件可以理解为IOC容器,每次加载配置文件的时候。系统会把这个配置文件里的所有bean给实例化成对象,方便后面从bean里拿指定的对象使用。 这种方式比较符合开闭原则,但是底层还是依赖set方法进行注入的。所以set方法注入是最基本和最原始的一种依赖注入方法(深刻体会这种方式),然后和xml这种配置文件的方式对比。
优点
对于子类的实现灵活替换,对于dao层方便扩展。只要增加一个实现类,在client(客户端)改变下具体的子类对象就可以了。
缺点
不符合开闭原则,因为dao层每次添加扩展类。client(客户端)端都要去类里修改具体的dao层的实现类,
能干什么
使用set注入可以灵活替换子类实现,下面以service业务类里替换dao层的数据持久层类为例。把dao层的接口关联到service类里,方便调用dao层的具体实现类。
并且在开始不赋值,在后面使用set方法对 dao层接口进行赋值。
怎么用
代码组织架构
    
使用set前代码
dao
UserDAO
package com.kuang.dao;
/**
 * @program: spring
 * @description: 测试dao
 * @author: 康世行
 * @create: 2021-04-22 10:59
 */
public interface UserDao {
    //获取信息
    void getInforim();
}
mysqlDAo
package com.kuang.dao;
/**
 * @program: spring
 * @description: mysql实现测试
 * @author: 康世行
 * @create: 2021-04-22 11:01
 */
public class mysqlDao implements UserDao {
    @Override
    public void getInforim() {
        System.out.println("mysql实现");
    }
}oracle
package com.kuang.dao;
/**
 * @program: spring
 * @description: 测试
 * @author: 康世行
 * @create: 2021-04-22 11:17
 */
public class oracle implements UserDao {
    @Override
    public void getInforim() {
        System.out.println("oracle实现");
    }
}sqlservDao
package com.kuang.dao;
/**
 * @program: spring
 * @description: sqlerver实现
 * @author: 康世行
 * @create: 2021-04-22 11:01
 */
public class sqlservDao implements UserDao {
    @Override
    public void getInforim() {
        System.out.println("sqlerver实现");
    }
}service
UserService
package com.kuang.service;
/**
 * @program: spring
 * @description: service测试
 * @author: 康世行
 * @create: 2021-04-22 11:02
 */
public interface UserService {
    //获取user信息
    void getUer();
}UserServiceImpl
package com.kuang.service;
import com.kuang.dao.UserDao;
import com.kuang.dao.mysqlDao;
import com.kuang.dao.sqlservDao;
/**
 * @program: spring
 * @description: userService实现类
 * @author: 康世行
 * @create: 2021-04-22 11:04
 */
public class UserServiceImpl  implements UserService{
    //注入dao依赖
    private UserDao userDao=new sqlservDao();
    @Override
    public void getUer() {
        userDao.getInforim();
    }
}
test
import com.kuang.dao.mysqlDao;
import com.kuang.dao.oracle;
import com.kuang.dao.sqlservDao;
import com.kuang.service.UserService;
import com.kuang.service.UserServiceImpl;
import org.junit.Test;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
/**
 * @program: spring
 * @description: 测试程序
 * @author: 康世行
 * @create: 2021-04-22 11:06
 */
public class MyTest {
    public static void main(String[] args) {
        //实例化业务类
       UserService userService=new UserServiceImpl();
        userService.getUer();
    }
    
}
result

使用set后代码
service
UserServiceImpl
/**
 * @program: spring
 * @description: userService实现类
 * @author: 康世行
 * @create: 2021-04-22 11:04
 */
public class UserServiceImpl  implements UserService{
    //注入dao依赖
    private UserDao userDao;
    //实现动态创建对象,不用提前把对象写死,根据客户端的需求去实例化对您的对象。
    public void setUserDao(UserDao userDao) {
        this.userDao = userDao;
    }
    @Override
    public void getUer() {
        userDao.getInforim();
    }
}test
import com.kuang.dao.mysqlDao;
import com.kuang.dao.oracle;
import com.kuang.dao.sqlservDao;
import com.kuang.service.UserService;
import com.kuang.service.UserServiceImpl;
import org.junit.Test;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
/**
 * @program: spring
 * @description: 测试程序
 * @author: 康世行
 * @create: 2021-04-22 11:06
 */
public class MyTest {
    public static void main(String[] args) {
        //实例化业务类
        UserService userService=new UserServiceImpl();
        //把userServicve转换为子类类型,并给业务类set方法赋值
       ((UserServiceImpl) userService).setUserDao(new oracle());
         
        //调用具体的方法
        userService.getUer();
    
    
    }
    
}
result
     
对比总结
从上面的两个实例可以看出,使用set方法后对于dao层具体实现类的替换,相对比较灵活。 还可以看出来,使用set前和使用set后,对于dao层是不用动的。只是把service层和test层就只修改了一点点。仅仅是这点修改,可以让客户端灵活替换不同的dao层的具体实现类。这也是IOC最原始的注入方法,IOC容器注入分为:使用xml文件和注解两种常用的方式。并且这两种方式还分为:手动注入和自动注入,具体的大家可以继续深入了解下这两种的不同实现方法。
个人把spring核心分为三类;
IOC
代理模式
AOP
尤其是在学完IOC 之后,一定要先把代理模式整明白(静态代理,动态代理),静态代理比较好理解,重点是动态代理比较稍微难点。把动态代理整明白之后再学习AOP就会很轻松,因为AOP底层实现使用的就是动态代理,还可以顺便了解下反射这对AOP和动态代理的学习有非常大的帮助。
 
                    
                     
                    
                 
                    
                
 
                
            
         
         浙公网安备 33010602011771号
浙公网安备 33010602011771号