Play! Framework 系列(三):依赖注入

Play! Framework 系列(二)中我们介绍了 Play 的项目结构。在日常处理业务逻辑的时候,我们都会用到依赖注入,本文将介绍一下 Play! 中的依赖注入以及如何合理地去使用她。

为什么要使用「依赖注入」

在许多 Java 框架中,「依赖注入」早已不是一个陌生的技术,Play 框架从 2.4 开始推荐使用 Guice 来作为依赖注入。

采用依赖注入最大的好处就是为了「解耦」,举个栗子:

上一篇文章的例子中,我们实现了一个 EmployeeService 用来对公司的员工进行操作:

1

2

3

4

5

6

7

package services

import models._

classEmployeeSerivce{

...

}

在之前的实现中,我们没有加入数据库的操作,那么现在我们想要引入一个数据库连接的类:DatabaseAccessService 来对数据库进行连接以便我们对相应的数据库表进行操作,则:

新建一个数据库连接操作的 Service

1

2

3

package services

classDatabaseAccessService{}

EmployeeSerivce 需要依赖 DatabaseAccessService

1

2

3

4

5

6

7

package services

import models._

classEmployeeSerivce(db: DBService){

...

}

好了,现在我们需要在 EmployeeController 中使用 EmployeeSerivce,如果不采用依赖注入,则:

1

2

3

4

5

6

7

8

9

10

11

classEmployeeController@Inject() (

cc: ControllerComponents

) extends AbstractController(cc) {

val db = newDatabaseAccessService()

val employeeSerivce = newEmployeeSerivce(db)

def employeeList = Action { implicit request: Request[AnyContent] =>

val employees = employeeSerivce.getEmployees()

Ok(views.html.employeeList(employees))

}

}

可以看到,为了实例化 EmployeeSerivceDatabaseAccessService 也需要实例化,如果随着需求的增加,EmployeeSerivce 所需要依赖的东西增加,那么我们每次实例化 EmployeeSerivce 的时候都需要将她的依赖也实例化一遍,而且她的依赖也有可能会依赖其他东西,这样就使得我们的代码变得非常冗余,也极难维护。

为了解决这一问题,我们引入了依赖注入,Play支持两种方式的依赖注入,分别是:「运行时依赖注入」以及「编译时依赖注入」,接下来我们就通过这两种依赖注入来解决我们上面提出的问题。

运行时依赖注入(runtime dependency

Play 的运行时依赖注入默认采用 Guice,关于 Guice,我们后面的文章当中会介绍,这里只需要知道她。为了支持 Guice 以及其他的运行时依赖注入框架,Play 提供了大量的内置组件。详见 play.api.inject

那么在 Play 中我们将如何使用这种依赖注入呢?回到我们文章刚开始讲的那个栗子中,现在我们通过依赖注入的方式来重新组织我们的代码:

首先 EmployeeSerivce 需要依赖 DatabaseAccessService,这里其实就存在一个「依赖注入」,那我们这样去实现:

1

2

3

4

5

6

7

8

package services

import models._

import javax.inject._

classEmployeeSerivce@Inject() (db: DBService){

...

}

在上面的代码中,我们引入了 import javax.inject._,并且可以看到多了一个 @Inject() 注解,我们实现运行时依赖注入就采用该注解。

那么在 EmployeeController 中,我们的代码就变成了:

1

2

3

4

5

6

7

8

9

classEmployeeController@Inject() (

employeeSerivce: EmployeeSerivce,

cc: ControllerComponents

) extends AbstractController(cc) {

def employeeList = Action { implicit request: Request[AnyContent] =>

val employees = employeeSerivce.getEmployees()

Ok(views.html.employeeList(employees))

}

}

可以看到我们不需要再去写那么多的实例了,我们只要在需要某种依赖的地方声明一下我们需要什么样的依赖, Play 在运行时就会将我们需要的依赖注入到相应的组件中去。

tip@Inject 必须放在类名的后面,构造参数的前面。

「运行时依赖注入」,顾名思义就是在程序运行的时候进行依赖注入,但是她不能在编译时进行校验,为了能让程序在编译时就能实现对依赖注入的校验, Play支持了「编译时依赖注入」。

编译时依赖注入(compile time dependency injection

为了实现编译时依赖注入,我们需要知道 Play 提供的一个特质:ApplicationLoader,该特质中的 load 方法将会在程序启动的时候加载我们的应用程序,在这个过程中,Play 框架本身以及我们自己的程序代码所依赖的东西都会被实例化。

默认情况下,Play 提供了一个 Guice 模块,该模块下的 GuiceApplicationBuilder 会根据 Play 框架给定的 context 去将该程序所依赖的所有组件联系在一起。

如果我们要自定义 ApplicationLoader,我们也需要一个像 GuiceApplicationBuilder 的东西,好在 Play 提供了这么一个东西,那就是:BuiltInComponentsFromContext,我们可以通过继承这个类来实现我们自己的 ApplicationLoader

接下来我们通过相应的代码来作进一步的解释:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

import controllers._

import play.api._

import play.api.routing.Router

import services._

import router.Routes

//自定义 ApplicationLoader

classMyApplicationLoaderextendsApplicationLoader {

def load(context: Context): Application = {

newMyComponents(context).application

}

}

classMyComponents(context: Context)

extendsBuiltInComponentsFromContext(context)

with play.filters.HttpFiltersComponents {

lazyval databaseAccessService = newDatabaseAccessService

lazyval employeeSerivce = newEmployeeSerivce(databaseAccessService)

lazyval employeeController = newEmployeeController(employeeSerivce, controllerComponents)

lazyval router: Router = newRoutes(httpErrorHandler, employeeController)

}

我们通过继承 BuiltInComponentsFromContext 使得程序能够根据 Play 所提供的 context 来加载 Play 框架本身所需要的一些组件。

那么回到我们的「编译时的依赖注入」中来,可以看到在 class MyComponents 中,我们将所有的 service 都实例化了,并且将这些实例注入到相应的依赖她们的模块中:

1

2

3

4

5

6

7

8

//将两个 service 实例化

lazyval databaseAccessService = newDatabaseAccessService

//EmployeeSerivce 依赖 DatabaseAccessService,将实例 databaseAccessService 注入其中

lazyval employeeSerivce = newEmployeeSerivce(databaseAccessService)

// employeeSerivce 注入到 employeeController

lazyval employeeController = newEmployeeController(employeeSerivce, controllerComponents)

使用 BuiltInComponentsFromContext 时,我们需要自己实现一下 router

1

lazyval router: Router = newRoutes(httpErrorHandler, employeeController)

tip:需要注意的是,如果我们实现了自己的 ApplicationLoader,我们需要在 application.conf 文件中声明一下:

1

play.application.loader = MyApplicationLoader

通过自定义 ApplicationLoader 我们就实现了编译时期的依赖注入,那么 EmployeeSerivce 就变成了:

1

2

3

4

5

6

7

package services

import models._

classEmployeeSerivce (db: DBService){

...

}

可以看到,这里就省去了 @Inject() 注解。

同样的,对于 EmployeeController

1

2

3

4

5

6

7

8

9

10

11

12

13

14

package controllers

import play.api._

import play.api.mvc._

import models._

import services._

// 没有了 @Inject() 注解

classEmployeeController (

employeeSerivce: EmployeeSerivce,

cc: ControllerComponents

) extends AbstractController(cc) {

...

}

通过使用编译时期的依赖注入,我们只需要在将所有的依赖实例化一次就够了,并且使用这种方式,我们能够在编译时期就能发现程序的一些异常。同样的,使用该方法也会有一些问题,就是我们需要写许多样板代码。另外本文的编译时期的依赖注入完全是自己手动注入的,看上去也比较繁琐,不是那么直观,如果要使用更优雅的方式,我们可以使用 macwire,这个我们在后面的文章中会详细讲解。

结语

本文简单介绍了一下 Play 支持的两种依赖注入的模式,文中提到的一些第三方依赖注入的框架我们会在后面的文章中详细介绍。本文的例子请戳源码链接

本文转载自:https://scala.cool/2017/11/play-3/

  • 2
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论
MVC应用程序模型 - 7 - app/controllers - 8 - app/models - 8 - app/views - 8 - 请求生命周期 - 8 - 标准应用程序布局layout - 9 - app目录 - 9 - public目录 - 10 - conf目录 - 10 - lib目录 - 11 - 开发生命周期 - 11 - 连接到java调试器 - 12 - 类增强Enhancement - 13 - 02.HTTP路由 - 13 - 关于REST - 14 - routes文件语法 - 14 - HTTP方法 - 15 - URI范示 Pattern - 15 - Java调用定义 - 17 - 把404当作action来用 - 17 - 指派静态参数 - 17 - 变量和脚本 - 18 - 路由优先级 - 18 - 服务器静态资源 - 18 - staticDir: mapping - 18 - staticFile: mapping - 19 - URL 编码 - 19 - 反转路由:用于生成某些URL - 19 - 设置内容风格(CSS) - 20 - HTTP 内容协商 negotiation - 21 - 从http headers开始设置内容类型 - 21 - 定制格式 - 22 - 03.控制器 - 23 - 控制器概览 - 23 - 获取http参数 - 24 - 使用params map - 25 - 还可以从action方法签名实现转换 - 25 - 高级HTTP Java绑定 - 26 - 简单类型 - 26 - Date类型 - 26 - Calendar日历 - 27 - File - 27 - 支持类型的数组或集合 - 28 - POJO对象绑定 - 29 - JPA 对象绑定 - 30 - 定制绑定 - 30 - @play.data.binding.As - 30 - @play.data.binding.NoBinding - 31 - play.data.binding.TypeBinder - 31 - @play.data.binding.Global - 32 - 结果类型 - 32 - 返回一些文本类型的内容 - 33 - 返回一个JSON字符串 - 33 - 返回一个XML字符串F - 34 - 返回二进制内容 - 34 - 作为附件下载文件 - 34 - 执行一个模板 - 35 - 跳转到其他URL - 36 - Action链 - 36 - 定制web编码 - 37 - 拦截器 - 38 - @Before - 38 - @After - 39 - @Catch - 40 - @Finally - 41 - 控制器继承 - 42 - 使用@With注释添加更多的拦截器 - 42 - Because Java does not allow multiple inheritance, it can be very limiting to rely on the Controller hierarchy to apply interceptors. But you can define some interceptors in a totally different class, and link them with any controller using the @With annotation.由于java不允许多继承,通过控制器继承特点来应用拦截器就受到极大的限制。但是我们可以在一个完全不同的类里定义一些拦截器,然后在任何控制器里使用@With注释来链接他们。 - 42 - Session和Flash作用域 - 42 - 04.模板引擎 - 43 - 模板语法 - 43 - Expressions: ${…} - 44 - Template decorators : #{extends /} and #{doLayout /} - 44 - Tags: #{tagName /} - 45 - Actions: @{…} or @@{…} - 46 - Messages: &{…} - 46 - Comment: *{…}* - 46 - Scripts: %{…}% - 46 - Template inheritance继承 - 47 - 定制模板标签 - 48 - 检索tag参数 - 48 - 调用标签体 - 48 - 格式化特定标签 - 49 - 定制java标签 - 49 - 标签命名空间 - 50 - 在模板里的Java对象扩展 - 51 - 创建定制扩展 - 52 - 模板里可以使用的保留对象 - 52 - 05.用play验证http数据

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

学亮编程手记

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值