当前位置: 首页 > news >正文

IOC容器实现分层解耦

         文章开始之前,先引入软件开发的两个名词:耦合和内聚。耦合是指:衡量软件中各个层(三层架构)/各个模块的依赖关联程度;内聚是指:软件中各个功能模块内部的功能联系。三层架构中Controller、Service、Dao是相互依赖关联的,是耦合的,并且在各层级中,直接使用new创建的依赖的对象,假如依赖的对象发生了变化,就必须去修改源代码,这是我们不愿意看到的。三层架构的每一层内部的功能都是独立的,都只处理自身的业务逻辑,其他业务逻辑不会定义在该类中,所以说类聚程度很高。软件开发就是需要“高内聚,低耦合”。

        三层架构解耦的思路

        需要依赖其他层完成业务时,在声明对象的时候,不要直接new对象;需要使用一个容器,将需要使用的对象存储起来,当需要使用该对象时,直接在容器中获得即可,避免用new实例化对象,从而解除耦合。那么如何实现这个操作呢?Spring提供了两个思想(方法):控制反转,依赖注入。

        控制反转、依赖注入、Bean对象

        控制反转

        控制反转(Inversion Of Control),简称IOC。其主要思想是:对象的创建控制权由程序自身(new对象)转移到外部(容器),这个容器也可以称为IOC容器或者Spring容器(一般不用这个名字)。

        依赖注入

        依赖注入(Dependency Injection),简称DI。其主要思想是:IOC容器为应用程序提供运行时所依赖的资源(需要使用的实体类)。

        Bean对象

        Bean对象是IOC容器中创建和管理的对象。

        具体实现

        因为Controller层依赖了Service层、Service层依赖了Dao层,所以说我们要将Dao及Service层的实现类交给IOC容器管理,并且为Controller及Service层注入运行时所依赖的对象(DI)。Spring框架提供了两个注解实现IOC和DI,分别是:@Component,将该类交给IOC容器管理,这个注解一般使用在实现类上面;@AutoWired,从IOC容器中自动寻找依赖的类并注入,使用在类的声明上。

        Controller层

        没有哪一层需要依赖Controller层,所以说Controller不需要交给IOC容器管理,但是Controller层依赖了Service层,所以说在声明Service对象的时候,要在声明上添加@AutoWired注解,表示从IOC容器中进行依赖注入:

package com.wzb.controller;import cn.hutool.json.JSONConfig;
import cn.hutool.json.JSONUtil;
import com.wzb.pojo.User;
import com.wzb.service.UserService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;import java.util.List;@RestController
public class UserController {@Autowiredprivate UserService userService;// 获取用户数据的代码不能写在此处,类的成员变量初始化是在类的实例化阶段进行的,此时可能@AutoWired注入还未完成,导致Null// private final List<User> userList = userService.findUser();@RequestMapping("/list")public String list() {List<User> userList = userService.findUser();return JSONUtil.toJsonStr(userList, JSONConfig.create().setDateFormat("yyyy-MM-dd HH:mm:ss"));}}

        特别提醒:在不使用IOC容器的时候,可以将获取数据当作成员变量书写,因为事先已经完成了UserService的初始化,不会为null;但是使用之后,获取数据必须在方法中定义,因为类的成员变量初始化是在类的实例化阶段进行的,此时可能@AutoWired注入还未完成,导致Null。

        Service层

        Service层作为Controller层的依赖,所以说在类名上必须加上@Component注解,代表将该类交给IOC容器管理;Service层还依赖了Dao层,所以说在声明的时候需要使用AutoWired注解

package com.wzb.service.impl;
import com.wzb.dao.UserDao;
import com.wzb.pojo.User;
import com.wzb.service.UserService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;import java.time.LocalDateTime;
import java.time.format.DateTimeFormatter;
import java.util.List;
import java.util.stream.Collectors;/*** Service层实现类**/
@Component
public class UserServiceImpl implements UserService {@Autowiredprivate UserDao userDao;// 获取用户数据的代码不能写在此处,类的成员变量初始化是在类的实例化阶段进行的,此时可能@AutoWired注入还未完成,导致Null// private final List<String> lines = userDao.findUser();public List<User> findUser() {List<String> lines = userDao.findUser();List<User> userList = lines.stream().map(line -> {String[] parts = line.split(",");Integer id = Integer.parseInt(parts[0]);String username = parts[1];String password = parts[2];String name = parts[3];Integer age = Integer.parseInt(parts[4]);LocalDateTime updateTime = LocalDateTime.parse(parts[5], DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss"));return new User(id, username, password, name, age, updateTime);}).collect(Collectors.toList());return userList;}
}

        Dao层

        Dao层不需要依赖其他层,但是Service依赖了Dao层,所以说只需要在类名上添加@Component注解即可:

package com.wzb.dao.impl;import cn.hutool.core.io.IoUtil;
import com.wzb.dao.UserDao;
import org.springframework.stereotype.Component;import java.io.InputStream;
import java.nio.charset.StandardCharsets;
import java.util.ArrayList;
import java.util.List;/*** Dao层的实现类**/
@Component
public class UserDaoImpl implements UserDao {@Overridepublic List<String> findUser() {InputStream in = this.getClass().getClassLoader().getResourceAsStream("user.txt");return IoUtil.readLines(in, StandardCharsets.UTF_8, new ArrayList<>());}
}

        通过Spring框架提供的IOC和DI,简单的实现了分层解耦,降低了模块之间的耦合度,但是也并不是大量使用IOC和DI就是完美的,这其实破坏类的不可变属性:字段注入使得类的依赖对象可以在类实例化后被外部(Spring 容器)随意修改。所以说使用还是要合理。

 

 

 

http://www.lryc.cn/news/492615.html

相关文章:

  • Flutter 共性元素动画
  • K8s内存溢出问题剖析:排查与解决方案
  • 乌班图单机(不访问外网)部署docker和服务的方法
  • 使用 pycharm 新建使用 conda 虚拟 python 环境的工程
  • Docker的save和export命令的区别,load和import的区别 笔记241124
  • 通俗理解人工智能、机器学习和深度学习的关系
  • 使用 pycharm 新建不使用 python 虚拟环境( venv、conda )的工程
  • 【大数据学习 | Spark-SQL】SparkSQL读写数据
  • AI赋能公共服务转型升级 | 第十届中国行业互联网大会暨腾讯云TVP行业大使三周年庆典公共服务专场圆满举办!
  • 关于按天切割Tomcat的catalina.out日志文件的配置
  • 【人工智能】深入解析GPT、BERT与Transformer模型|从原理到应用的完整教程
  • 彻底理解如何保证ElasticSearch和数据库数据一致性问题
  • 2024-2025热门留学趋势
  • 寻找视频特效素材的优质网站推荐 轻松提升作品魅力
  • 【英特尔IA-32架构软件开发者开发手册第3卷:系统编程指南】2001年版翻译,2-36
  • 信息安全实验--密码学实验工具:CrypTool
  • python的class 类创建、方法调用以及属性赋值
  • Angular v19 (二):响应式当红实现signal的详细介绍:它擅长做什么、不能做什么?以及与vue、svelte、react等框架的响应式实现对比
  • IMX 平台UART驱动情景分析:write篇--从 TTY 层到硬件驱动的写操作流程解析
  • 网络安全拟态防御技术
  • 灵活开源低代码平台——Microi吾码(一)
  • frida_hook_libart(简单解释)
  • 计算机网络八股整理(二)
  • 强化学习off-policy进化之路(PPO->DPO->KTO->ODPO->ORPO->simPO)
  • Linux 如何创建逻辑卷并使用
  • java实现将图片插入word文档
  • 初识java(3)
  • coqui-ai TTS 初步使用
  • matlab代码--卷积神经网络的手写数字识别
  • Scala—Map用法详解