
本文将以Spring Boot为例,介绍下实际工作中的Java/Scala互操作。在提高个人效率、满足自我追求的情况下更多的照顾到团队不同人的实际。同时也是想说明,在同一个工程里混用Java和Scala语言是可能的。
本文源代码在:http://git.oschina.net/hualongdata/spring-starter
对于Scala不熟悉的Java程序员可以先看看:http://www.yangbajing.me/2016/07/24/写给java程序员的scala入门教程/
对于Spring不了解的Scala程序员可以先看看:http://www.yangbajing.me/2016/03/03/scala开发者的spring-boot快速上手指南_01/ 和 http://www.yangbajing.me/2016/08/25/scala开发者的spring-boot快速上手指南-02:scala惯用法/
Java Bean
Java Bean有个特点,就是对于可修改属性都会有对应的getter和setter方法(final属性将只有getter方法)。由Java定义的对象在Scala中可以直接使用,并无二样。而在Scala中定义Java Bean却有些不同。
其实在Scala中可以像Java一样来定义Java Bean:
// Scala中默认为public访问权限,包括属性和方法 class Person { // 下划线在这里是一个占位符,它代码相应属性对应类型的默认值 private var id: Int = _ private var name: String = _; def getId: Int = id; def setId(id: Int) { this.id = id } def getName: String = name; def setName(name: String) { this.name = name; } }
这样写的话,除了语法上与Java有些差别,其实定义的方式是一样的。但其实Scala提供了注解来自动生成setter和setter函数:
import scala.beans.BeanProperty class Person { @BeanProperty var id: Int = _ @BeanProperty var name: String = _ @BeanProperty val createdAt: LocalDateTime = _ }
除了使用传统的class,在Scala中还可以使用case class来定义POJO:
case class SignRequest(@BeanProperty account: String = null, @BeanProperty password: String = null, @BeanProperty captcha: String = null, @BeanProperty var smsCode: String = null)
case class的主构造函数声明的参数将同时做为SignRequest
的履性,且是val
的(类似Java的public final
)。在这里,account
、password
和captcha
将只生成getter函数。而smsCode
将生成getter和setter函数,因为它使用var
来修饰。
这里有一个Java里没有的特性:参数默认值,像C++、Python、ES6+ 一样,Scala的参数是可以设置默认值的。因为Java Bean规范要求类必需有参数为空的默认构造函数,而当case class的主构造函数所有参数都设置默认值后,在实例化这个类时将相当于拥有一个空的默认构造函数。
在Java中调用case class可见:com/hualongdata/springstarter/data/repository/UserRepositoryImpl.java。
基于注解的依赖注入
在Spring开发中,依赖注入是很常用的一个特性。基于属性的注解注入在Java和Scala中都是一样的。但基于构造函数的依赖注入在Scala中有些特别,代码如下:
class SignController @Autowired()(userService: UserService, webUtils: WebUtils, hlTokenComponent: HlTokenComponent) { ...... }
在Scala中,单注解作用于构造函数上时需要类似方法调用的形式:@Autowired()
。又因为Scala中,主构造函数必需定义在类名之后的小括号内,所以注解需要紧跟在类名之号,主构造函数左括号之前。
在Scala中使用主构造函数的注入组件是一个更好的实践,它同时拥有注入的组件为private final访问权限。相同效果的Java代码需要更多:
public SignController { private final UserService userService; private final WebUtils webUtils; private final HlTokenComponent hlTokenComponent; public SignController(UserService userService, WebUtils webUtils, HlTokenComponent hlTokenComponent) { this.userService = userService; this.webUtils = webUtils; this.hlTokenComponent = hlTokenComponent; } }
可以看到,Scala的版本代码量更少,同时看起来更简洁。
完整文章请点击“阅读原文”查看。




