目录
一、三层架构
在 JavaWeb开发 —— 请求响应 最后案例中我们编写的程序代码都是写在 Controller 当中。
而在我们实际软件设计和开发中,会尽量让每一个接口、类或者方法功能单一,只负责一方面,这就是所谓的单一职责原则。这样就可以使接口、类或者方法复杂度更低、可读性更强。基于此,Web开发才有了三层架构:
- Controller:控制层,接收前端发送的请求,对请求进行处理,并响应数据。
- Service:业务逻辑层,处理具体的业务逻辑。
- Dao:数据访问层(Data Access object)(持久层),负责数据访问操作,包括数据的增、删、改、查。
二、分层解耦
① 内聚:软件电各个功能模块内部的功能联系。
② 耦合:衡量软件中 各个层 / 模块 之间的依赖、关联的程度。
③ 软件设计原则:高内聚低耦合。
Controller层 | Service层 |
那么我们发现 Service层 代码改动,Controller层 也会随之发生发生改动,我们就称 Service层和Controller层 代码发生耦合。那么我们接下来就需要考虑如何解耦合?
因此我们不能再直接去 new Service层实现类,但是运行结果会报错空指针异常。我们可以提供一个容器来存放对象。
但是进而又出现新的问题:
- 对象如何交给容器进行管理? ——> 控制反转
- 容器如何为我们程序提供所依赖的资源? ——> 依赖注入
控制反转: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同级时,再次运行发生报错:
- 前面声明bean的四大注解,要想生效,还需要被组件扫描注解 @ComponentScan 扫描。
- @ComponentScan注解虽然没有显式配置,但是实际上已经包含在了启动类声明注解@SpringBootApplication中,默认扫描的范围是启动类所在包及其子包。
解决方案:
- 手动再次指定包扫描。(不推荐)
@ComponentScan({"dao","com.spring"}) @SpringBootApplication //默认扫描当前包及其子包 public class SpringbootWebReqRespApplication { public static void main(String[] args) { SpringApplication.run(SpringbootWebReqRespApplication.class, args); } }
- 根据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歌认是按照名称注入。