IOC容器实现分层解耦

devtools/2024/11/28 12:10:58/

         文章开始之前,先引入软件开发的两个名词:耦合和内聚。耦合是指:衡量软件中各个层(三层架构)/各个模块的依赖关联程度;内聚是指:软件中各个功能模块内部的功能联系。三层架构中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容器中进行依赖注入:

java">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注解

java">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注解即可:

java">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.ppmy.cn/devtools/137656.html

相关文章

oracle小技巧-解决特殊密码字符而导致的exp错误

在使用oracle数据库的时候&#xff0c;我们经常会利用exp工具对某些表进行导出。但有些时候&#xff0c;因我们用户密码为安全性设有特殊字符&#xff0c;导致exp导出时候报&#xff1a;“EXP-00056和ORA-12154”&#xff0c;今天我们就分享下如何通过设置符号隔离的小技巧解决…

南京移动“智慧+关怀”服务体系助力老年群体生活安全有保障

在数字化浪潮汹涌澎湃的当下&#xff0c;江苏移动南京分公司秉持“人民邮电为人民”的服务理念&#xff0c;推出一系列创新服务举措&#xff0c;为社区老年群体提供贴心、便捷的数字服务&#xff0c;让老人在享受科技发展成果的同时&#xff0c;感受到社会的温暖与关怀。 贴心…

uniapp介入极光推送教程 超级详细

直接按照下面教程操作 一步一步来 很快就能 完成 下面的文章非常详细 &#xff0c;我就不班门弄斧了 直接上原文链接 https://blog.csdn.net/weixin_52830464/article/details/143823231

Java 使用堆外内存(off-heap memory) 作为缓存

我们通常用的 Java 缓存基本可认为是扩展了 HashMap 或 ConcurrentHashMap 的实现&#xff0c;它们各自实现自己的缓存策略&#xff0c;如时间与空间的控制&#xff0c;生命周期管理&#xff0c;是否支持分布式&#xff0c;溢出时能否转储到磁盘。关于 Java 本地缓存的存储分为…

【MATLAB源码-第228期】基于matlab的鼠群优化算法(RSO)无人机三维路径规划,输出做短路径图和适应度曲线.

操作环境&#xff1a; MATLAB 2022a 1、算法描述 鼠群优化算法&#xff08;Rat Swarm Optimization, RSO&#xff09; 简介 鼠群优化算法&#xff08;Rat Swarm Optimization, RSO&#xff09;是一种模仿鼠类群体觅食行为的优化算法。该算法属于群体智能算法&#xff0c;通过…

基于Graph Convolutional Networks (GCN)的知识图谱嵌入详解

在众多的嵌入方法中&#xff0c;基于图神经网络&#xff08;Graph Neural Networks, GNN&#xff09;的嵌入方法近年来备受瞩目。其中&#xff0c;图卷积网络&#xff08;Graph Convolutional Networks, GCN&#xff09;通过捕捉图中节点的邻域信息&#xff0c;能够有效学习节点…

为什么神经网络的可解释性差?

神经网络的可解释性问题是一个复杂的话题&#xff0c;涉及到多个因素。确实&#xff0c;神经网络中的权重和偏置是影响可解释性的因素之一&#xff0c;但还有其他几个关键因素也对神经网络的可解释性产生了影响。以下是一些主要原因&#xff1a; 权重和偏置的复杂性&#xff1…

next build报错bash: next: command not found

相关背景&#xff1a;我有个next.js项目&#xff08;第一次使用next.js框架&#xff09;&#xff0c;想构建一下&#xff0c;发现报错了&#xff0c;没有该命令。 &#xff08;ps&#xff1a;我测试用的是&#xff1a;yarn dev&#xff09; 然后我用everything全局搜了下 next…