Scala开发者的Spring-Boot快速上手指南 02:Scala惯用法

(这是一篇迟来的文章,从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

第一篇文章是基于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 = []
            }
        }
    }

支持Scala数据类型

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 classScala CollectionOption等类型和数据结构,甚至还能够和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的惯用法:

  • 注解属性为数组时,Scala必需显示使用数组形式传参。如@RequestMapping注解的path发型是一个数组类型,在Scala中须要显示传入一个数组类型的参数:Array("message")
  • Scala中,方法返回值类型是能够自动推导的。但在写Spring控制器方法时推荐显示注明返回类型。
  • Scala的全部表达式都有值,且代码块最后一个表达式的值就是代码块的值。这样,在Scala的函数里不须要使用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 Function 和 Scala Function[N]

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平台上还有如此有意思的语言。

相关文章
相关标签/搜索