JavaWeb开发 —— 分层解耦

目录

一、三层架构

二、分层解耦

三、IOC & DI 入门

四、IOC控制反转详解

五、DI依赖注入详解


一、三层架构

在 JavaWeb开发 —— 请求响应 最后案例中我们编写的程序代码都是写在 Controller 当中。

JavaWeb开发 —— 分层解耦_第1张图片

 而在我们实际软件设计和开发中,会尽量让每一个接口、类或者方法功能单一,只负责一方面,这就是所谓的单一职责原则。这样就可以使接口、类或者方法复杂度更低、可读性更强。基于此,Web开发才有了三层架构:

  • Controller:控制层,接收前端发送的请求,对请求进行处理,并响应数据。
  • Service:业务逻辑层,处理具体的业务逻辑。
  • Dao:数据访问层(Data Access object)(持久层),负责数据访问操作,包括数据的增、删、改、查。

JavaWeb开发 —— 分层解耦_第2张图片

JavaWeb开发 —— 分层解耦_第3张图片 JavaWeb开发 —— 分层解耦_第4张图片
JavaWeb开发 —— 分层解耦_第5张图片

二、分层解耦

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

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

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

JavaWeb开发 —— 分层解耦_第6张图片 JavaWeb开发 —— 分层解耦_第7张图片
Controller层 Service层

 那么我们发现 Service层 代码改动,Controller层 也会随之发生发生改动,我们就称 Service层和Controller层 代码发生耦合。那么我们接下来就需要考虑如何解耦合

 因此我们不能再直接去 new Service层实现类,但是运行结果会报错空指针异常。我们可以提供一个容器来存放对象。

JavaWeb开发 —— 分层解耦_第8张图片

但是进而又出现新的问题:

  • 对象如何交给容器进行管理?   ——> 控制反转
  • 容器如何为我们程序提供所依赖的资源?  ——> 依赖注入

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

依赖注入:Dependency Injection,简称DI。容器为应用程序提供运行时,所依赖的资源,称之为依赖注入。
Bean对象:IOC容器中创建、管理的对象,称之为bean

三、IOC & DI 入门

分层解耦步骤:

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

为Controller及Service注入运行时,依赖的对象。

运行测试。

//EmpDao.java
@Component  //将当前类交给IOC容器管理,成为IOC容器中的bean
public class EmpDaoA implements EmpDao {
    
    ...
}

//EmpServiceA.java
@Component  //将当前类交给IOC容器管理,成为IOC容器中的bean
public class EmpServiceA extends EmpService {
    @Autowired  //运行时,IOC容器会提供该类型的bean对象,并赋值给该对象 -依赖注入
    private EmpDao empDao ;
    
    ...
}


//EmpController.java
@RestController
public class EmpController {
    //运行时,IOC容器会提供该类型的bean对象,并赋值给该对象 -依赖注入
    private EmpService empService ;
   
    ...
}

四、IOC控制反转详解

① Bean声明:

要把某个对象交给IOC容器管理,需要在对应的类上加上如下注解之一:

注解 说明 位置
@Component 声明bean的基础注解 不属于以下三类时,用此注解
@Controller @Component的衍生注解 标注在控制器类Controller
@Service @Component的衍生注解 标注在业务类Service
@Repository @Component的衍生注解 标注在数据访问类Dao上(由于与mybatis整合,用的少)

注意事项:

  • 声明bean的时候,可以通过value属性指定bean的名字,如果没有指定,默认为类名首字母小写。一般使用默认名即可。
  • 使用以上四个注解都可以声明bean,但是在springboot集成web开发中,声明控制器bean只能用@Controller。

② Bean组件扫描:

当我们将Dao层文件夹放在Java目录下与com.spring同级时,再次运行发生报错:

JavaWeb开发 —— 分层解耦_第9张图片

  • 前面声明bean的四大注解,要想生效,还需要被组件扫描注解 @ComponentScan 扫描。
  • @ComponentScan注解虽然没有显式配置,但是实际上已经包含在了启动类声明注解@SpringBootApplication中,默认扫描的范围是启动类所在包及其子包。

解决方案:

  1. 手动再次指定包扫描。(不推荐)
    @ComponentScan({"dao","com.spring"})
    @SpringBootApplication //默认扫描当前包及其子包
    public class SpringbootWebReqRespApplication {
    
        public static void main(String[] args) {
            SpringApplication.run(SpringbootWebReqRespApplication.class, args);
        }
    
    }
  2. 根据SpringBoot代码规范将所写代码全部放在启动类所在包及其子包下。这样SpringBoot项目在启动时就会自动扫描bean对象。

五、DI依赖注入详解

@Autowired注解,默认是按照类型进行,如果存在多个相同类型的bean,将会报出如下错误:

通过以下几种方案来解决:

  • @Prirmary:
    @Primary
    @Service
    public class EmpServiceA implements EmpService{
      ...
    }
  • @Qualifier:
    @RestController
    public class EmpController{
        @Autowired
        @Qualifier("EmpServiceA")
        private EmpService empService;
        ....
    }
  • @Resource:
@RestController
public class EmpController{
   @Resource(name = "EmpServiceB")
   private EmpService empService;
   ....
} 

@Resource 与 @Autowired区别:

  • @Autavaired是spring框架捉供的注解,而@Resource是JDK提供的注解。
  • @Autowired默认是按照类型注入,而@ReSource歌认是按照名称注入。

你可能感兴趣的:(JavaWeb开发,java,开发语言,spring)