(这是一篇迟来的文章,从3月份计划到成文花了5个月多……之后须要避免这样的低效率。)java
以前写第一篇文章时,只是想试试在Spring中使用Scala。但如今随着工做的须要,已经决定在应用层基于Spring boot进行开发。后面的数据服务和数据整合部分将采用Akka。做者是一个Scala粉,但不脑残。鉴于团队、招人及社区生态多方面考虑,总体使用Scala技术栈仍是比较困难的。以前就有考虑过把Spring和Scala结合起来。后来了解到挖财的技术选型,他们就是基于Spring和Scala的,还开源了不少不错的Spring Boot加强插件。这坚决了我以前的想法,也有了我5个月后续写第2篇的能量。git
对于Scala还不熟悉的朋友能够先看看《写给Java程序员的Scala入门教程》,好对Scala有个初步映像。程序员
第一篇文章是基于Maven作项目配置的,如今换成了Gradle。缘由?Spring官方默认都是基于Gradle了,并且如今不少大型的Java项目都是基于Gradle进行构建了。如:Android、Kafka(Linkdin总体采用Gradle)。再加上我是一个比较爱折腾的人,既然如今有时间,为何不试试Gradle呢?github
代码在这里:https://github.com/yangbajing/spring-boot-scala,此次不但把构建工具换成了Gradle,还一步到位使用了多项目的构建方式,这样更符合真实开发的场景。 **注意:在build.gradle配置中,须要从新设置Scala和Java源码的搜索路径,把Java源码路径移动Scala的搜索路径来。否则编译时会遇到Java代码找不到Scala代码符号问题 **web
sourceSets { main { scala { srcDirs = ['src/main/scala', 'src/main/java'] } java { srcDirs = [] } } test { scala { srcDirs = ['src/test/scala', 'src/test/java'] } java { srcDirs = [] } } }
Spring Boot默承认以自动转换JSON数据格式到Java类型或反之,但怎样支持Scala数据类型呢?其实很简单,只须要加入jackson-module-scala
依赖:spring
compile("com.fasterxml.jackson.module:jackson-module-scala_$scalaLibVersion:2.8.0.rc2")
并添加jacksonModuleScala
Bean 便可:json
@Bean public Module jacksonModuleScala() { return new DefaultScalaModule(); }
如今,咱们就能够在Spring中自由的使用case class
、Scala Collection
、Option
等类型和数据结构,甚至还能够和Java类型混合使用。好比咱们把Java类型嵌入到Scala的case class里。api
User.java数组
public class User { private String name; private String nickname; public String getName() { return name; } public void setName(String name) { this.name = name; } public String getNickname() { return nickname; } public void setNickname(String nickname) { this.nickname = nickname; } }
Message.scala数据结构
case class Message(name: String, age: Int, user: User, status: Option[Boolean]) { @BeanProperty val createdAt: LocalDateTime = LocalDateTime.now() }
Scala控制器 (ApiController.scala):
@RequestMapping(path = Array("message"), method = Array(RequestMethod.POST)) def message(@RequestBody reqMsg: Message): Seq[Message] = { List( reqMsg, reqMsg.copy(age = reqMsg.age + 1, status = Some(true)) ) }
使用Scala编写Spring控制器方法,有些和Java不同的地方和Scala的惯用法:
@RequestMapping
注解的path
发型是一个数组类型,在Scala中须要显示传入一个数组类型的参数:Array("message")
。return
显示返回数据,也不推荐使用return
。另外,若在Java代码中使用Scala的数据类型。如:case class。在Java中必需使用new
关键字进行实例化,像Scala那样直接经过类名实例化是不支持的。
Java控制器(WebController.java):
@RequestMapping(path = "message", method = RequestMethod.POST) public Message message(@RequestBody User user) { return new Message("Yang Jing", 30, user, new Some(false)); }
测试效果以下:
$ curl -XPOST -H 'content-type: application/json;utf8' -d '{"user":"杨景","nickname":"羊八井"}' http://localhost:18080/web/message {"name":"Yang Jing","age":30,"user":{"name":null,"nickname":"羊八井"},"status":false,"createdAt":"2016-08-25T17:22:50.841"} $ curl -XPOST -H 'content-type: application/json;utf8' -d '{"name":"yangbajing","age":30,"user":{"name":"杨景","nickname":"羊八井"}}' http://localhost:18080/api/message [{"name":"yangbajing","age":30,"user":{"name":"杨景","nickname":"羊八井"},"status":null,"createdAt":"2016-08-25T17:26:03.352"},{"name":"yangbajing","age":31,"user":{"name":"杨景","nickname":"羊八井"},"status":true,"createdAt":"2016-08-25T17:26:03.352"}]
Java 8开始,支持Lambda函数。可是Java的Lambda函数与Scala的函数类型是不兼容的(好消息是,从Scala 2.12开始,将兼容Java Lambda函数)。咱们可使用scala-java8-compat
这个库来还算优雅的解决这个问题。
首先添加scala-java8-comat
依赖:
compile("org.scala-lang.modules:scala-java8-compat_$scalaLibVersion:0.7.0")
在Scala中访问Java8 Function,可使用以下方式:
import scala.compat.java8.FunctionConverters._ def(@RequestParam name: Optional[String], ... name.orElseGet(asJavaSupplier(() => reqMsg.name))
除了显示的使用asJavaSupplier
来转换特定的Java8 Function,还可使用asJava
隐式转换来自动转换:
name.orElseGet((() => reqMsg.name).asJava)
也许你并不喜欢Scala,也不须要在Spring中使用Scala,Java 8也足够。但我但愿能为你打开了一扇门,在JVM平台上还有如此有意思的语言。