SpringBoot之分层解耦以及 IOC&DI的详细解析

发布时间:2023年12月20日

### 3.2 分层解耦

刚才我们学习过程序分层思想了,接下来呢,我们来学习下程序的解耦思想。

解耦:解除耦合。

#### 3.2.1 耦合问题

首先需要了解软件开发涉及到的两个概念:内聚和耦合。

- 内聚:软件中各个功能模块内部的功能联系。

- 耦合:衡量软件中各个层/模块之间的依赖、关联的程度。

**软件设计原则:高内聚低耦合。**

> 高内聚指的是:一个模块中各个元素之间的联系的紧密程度,如果各个元素(语句、程序段)之间的联系程度越高,则内聚性越高,即 "高内聚"。
>
> 低耦合指的是:软件中各个层、模块之间的依赖关联程序越低越好。

程序中高内聚的体现:

- EmpServiceA类中只编写了和员工相关的逻辑处理代码

程序中耦合代码的体现:

- 把业务类变为EmpServiceB时,需要修改controller层中的代码

高内聚、低耦合的目的是使程序模块的可重用性、移植性大大增强。

#### 3.2.2 ?解耦思路

之前我们在编写代码时,需要什么对象,就直接new一个就可以了。 这种做法呢,层与层之间代码就耦合了,当service层的实现变了之后, 我们还需要修改controller层的代码。

?那应该怎么解耦呢?

- 首先不能在EmpController中使用new对象。代码如下:

- 此时,就存在另一个问题了,不能new,就意味着没有业务层对象(程序运行就报错),怎么办呢?
? - 我们的解决思路是:
? ? - 提供一个容器,容器中存储一些对象(例:EmpService对象)
? ? - controller程序从容器中获取EmpService类型的对象

我们想要实现上述解耦操作,就涉及到Spring中的两个核心概念:

- **控制反转:** Inversion Of Control,简称IOC。对象的创建控制权由程序自身转移到外部(容器),这种思想称为控制反转。

? > 对象的创建权由程序员主动创建转移到容器(由容器创建、管理对象)。这个容器称为:IOC容器或Spring容器

- **依赖注入:** Dependency Injection,简称DI。容器为应用程序提供运行时,所依赖的资源,称之为依赖注入。

? > 程序运行时需要某个资源,此时容器就为其提供这个资源。
? >
? > 例:EmpController程序运行时需要EmpService对象,Spring容器就为其提供并注入EmpService对象

IOC容器中创建、管理的对象,称之为:bean对象

3.3 IOC&DI

上面我们引出了Spring中IOC和DI的基本概念,下面我们就来具体学习下IOC和DI的代码实现。

3.3.1 IOC&DI入门

任务:完成Controller层、Service层、Dao层的代码解耦

  • 思路:

    1. 删除Controller层、Service层中new对象的代码

    2. Service层及Dao层的实现类,交给IOC容器管理

    3. 为Controller及Service注入运行时依赖的对象

      • Controller程序中注入依赖的Service层对象

      • Service程序中注入依赖的Dao层对象

第1步:删除Controller层、Service层中new对象的代码

第2步:Service层及Dao层的实现类,交给IOC容器管理

  • 使用Spring提供的注解:@Component ,就可以实现类交给IOC容器管理

第3步:为Controller及Service注入运行时依赖的对象

  • 使用Spring提供的注解:@Autowired ,就可以实现程序运行时IOC容器自动注入需要的依赖对象

完整的三层代码:

  • Controller层:

@RestController
public class EmpController {
?
 ? ?@Autowired //运行时,从IOC容器中获取该类型对象,赋值给该变量
 ? ?private EmpService empService ;
?
 ? ?@RequestMapping("/listEmp")
 ? ?public Result list(){
 ? ? ? ?//1. 调用service, 获取数据
 ? ? ? ?List<Emp> empList = empService.listEmp();
?
 ? ? ? ?//3. 响应数据
 ? ? ? ?return Result.success(empList);
 ?  }
}
  • Service层:

@Component //将当前对象交给IOC容器管理,成为IOC容器的bean
public class EmpServiceA implements EmpService {
?
 ? ?@Autowired //运行时,从IOC容器中获取该类型对象,赋值给该变量
 ? ?private EmpDao empDao ;
?
 ? ?@Override
 ? ?public List<Emp> listEmp() {
 ? ? ? ?//1. 调用dao, 获取数据
 ? ? ? ?List<Emp> empList = empDao.listEmp();
?
 ? ? ? ?//2. 对数据进行转换处理 - gender, job
 ? ? ? ?empList.stream().forEach(emp -> {
 ? ? ? ? ? ?//处理 gender 1: 男, 2: 女
 ? ? ? ? ? ?String gender = emp.getGender();
 ? ? ? ? ? ?if("1".equals(gender)){
 ? ? ? ? ? ? ? ?emp.setGender("男");
 ? ? ? ? ?  }else if("2".equals(gender)){
 ? ? ? ? ? ? ? ?emp.setGender("女");
 ? ? ? ? ?  }
?
 ? ? ? ? ? ?//处理job - 1: 讲师, 2: 班主任 , 3: 就业指导
 ? ? ? ? ? ?String job = emp.getJob();
 ? ? ? ? ? ?if("1".equals(job)){
 ? ? ? ? ? ? ? ?emp.setJob("讲师");
 ? ? ? ? ?  }else if("2".equals(job)){
 ? ? ? ? ? ? ? ?emp.setJob("班主任");
 ? ? ? ? ?  }else if("3".equals(job)){
 ? ? ? ? ? ? ? ?emp.setJob("就业指导");
 ? ? ? ? ?  }
 ? ? ?  });
 ? ? ? ?return empList;
 ?  }
}

Dao层:

@Component //将当前对象交给IOC容器管理,成为IOC容器的bean
public class EmpDaoA implements EmpDao {
 ? ?@Override
 ? ?public List<Emp> listEmp() {
 ? ? ? ?//1. 加载并解析emp.xml
 ? ? ? ?String file = this.getClass().getClassLoader().getResource("emp.xml").getFile();
 ? ? ? ?System.out.println(file);
 ? ? ? ?List<Emp> empList = XmlParserUtils.parse(file, Emp.class);
 ? ? ? ?return empList;
 ?  }
}

运行测试:

文章来源:https://blog.csdn.net/qq_69748833/article/details/135095652
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。