写点什么

深入学习微框架:Spring Boot

  • 2014-05-12
  • 本文字数:18407 字

    阅读完需:约 60 分钟

Spring Boot 是由 Pivotal 团队提供的全新框架,其设计目的是用来简化新 Spring 应用的初始搭建以及开发过程。该框架使用了特定的方式来进行配置,从而使开发人员不再需要定义样板化的配置。通过这种方式,Boot 致力于在蓬勃发展的快速应用开发领域(rapid application development)成为领导者。

多年以来, Spring IO 平台饱受非议的一点就是大量的 XML 配置以及复杂的依赖管理。在去年的 SpringOne 2GX 会议上,Pivotal 的 CTO Adrian Colyer回应了这些批评,并且特别提到该平台将来的目标之一就是实现免XML 配置的开发体验。Boot 所实现的功能超出了这个任务的描述,开发人员不仅不再需要编写XML,而且在一些场景中甚至不需要编写繁琐的import 语句。在对外公开的beta 版本刚刚发布之时,Boot 描述了如何使用该框架在140 个字符内实现可运行的web 应用,从而获得了极大的关注度,该样例发表在 tweet 上。

然而,Spring Boot 并不是要成为 Spring IO 平台里面众多“Foundation”层项目的替代者。Spring Boot 的目标不在于为已解决的问题域提供新的解决方案,而是为平台带来另一种开发体验,从而简化对这些已有技术的使用。对于已经熟悉Spring 生态系统的开发人员来说,Boot 是一个很理想的选择,不过对于采用Spring 技术的新人来说,Boot 提供一种更简洁的方式来使用这些技术。

在追求开发体验的提升方面,Spring Boot,甚至可以说整个Spring 生态系统都使用到了 Groovy 编程语言。Boot 所提供的众多便捷功能,都是借助于 Groovy 强大的 MetaObject 协议、可插拔的 AST 转换过程以及内置的依赖解决方案引擎所实现的。在其核心的编译模型之中,Boot 使用 Groovy 来构建工程文件,所以它可以使用通用的导入和样板方法(如类的 main 方法)对类所生成的字节码进行装饰(decorate)。这样使用 Boot 编写的应用就能保持非常简洁,却依然可以提供众多的功能。

安装 Boot

从最根本上来讲,Spring Boot 就是一些库的集合,它能够被任意项目的构建系统所使用。简便起见,该框架也提供了命令行界面,它可以用来运行和测试 Boot 应用。框架的发布版本,包括集成的 CLI(命令行界面),可以在 Spring 仓库中手动下载和安装。一种更为简便的方式是使用 Groovy 环境管理器(Groovy enVironment Manager,GVM),它会处理 Boot 版本的安装和管理。Boot 及其 CLI 可以通过 GVM 的命令行gvm install springboot进行安装。在 OS X 上安装 Boot 可以使用 Homebrew 包管理器。为了完成安装,首先要使用brew tap pivotal/tap切换到 Pivotal 仓库中,然后执行brew install springboot命令。

要进行打包和分发的工程会依赖于像 Maven Gradle 这样的构建系统。为了简化依赖图,Boot 的功能是模块化的,通过导入 Boot 所谓的“starter”模块,可以将许多的依赖添加到工程之中。为了更容易地管理依赖版本和使用默认配置,框架提供了一个 parent POM,工程可以继承它。Spring Boot 工程的样例 POM 文件定义如程序清单 1 所示。

程序清单 1

复制代码
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>com.example</groupId>
<artifactId>myproject</artifactId>
<version>1.0.0-SNAPSHOT</version>
<!-- Inherit defaults from Spring Boot -->
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>1.0.0.RC1</version>
</parent>
<!-- Add typical dependencies for a web application -->
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-actuator</artifactId>
</dependency>
</dependencies>
<repositories>
<repository>
<id>spring-snapshots</id>
<url>http://repo.spring.io/libs-snapshot</url>
</repository>
</repositories>
<pluginRepositories>
<pluginRepository>
<id>spring-snapshots</id>
<url>http://repo.spring.io/libs-snapshot</url>
</pluginRepository>
</pluginRepositories>
<build>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
</plugin>
</plugins>
</build>
</project>

为了实现更为简单的构建配置,开发人员可以使用 Gradle 构建系统中简洁的 Groovy DSL,如程序清单 1.1 所示。

程序清单 1.1

复制代码
buildscript {
repositories {
maven { url "http://repo.spring.io/libs-snapshot" }
mavenCentral()
}
dependencies {
classpath("org.springframework.boot:spring-boot-gradle-plugin:1.0.0.RC1")
}
}
apply plugin: 'java'
apply plugin: 'spring-boot'
repositories {
mavenCentral()
maven { url "http://repo.spring.io/libs-snapshot" }
}
dependencies {
compile 'org.springframework.boot:spring-boot-starter-actuator:1.0.0.RC1'
}

为了快速地搭建和运行 Boot 工程,Pivotal 提供了称之为“Spring Initializr” 的web 界面,用于下载预先定义好的Maven 或Gradle 构建配置。我们也可以使用 Lazybones 模板实现快速起步,在执行lazybones create spring-boot-actuator my-app命令后,它会为 Boot 应用创建必要的工程结构以及 gradle 构建文件。

开发 Spring Boot 应用

Spring Boot 在刚刚公开宣布之后就将一个样例发布到了 Twitter 上,它目前成为了最流行的一个应用样例。它的全部描述如程序清单 1.2 所示,一个非常简单的 Groovy 文件可以生成功能强大的以 Spring 为后端的 web 应用。

程序清单 1.2

复制代码
@RestController
class App {
@RequestMapping("/")
String home() {
"hello"
}
}

这个应用可以通过spring run App.groovy命令在 Spring Boot CLI 中运行。Boot 会分析文件并根据各种“编译器自动配置(compiler auto-configuration)”标示符来确定其意图是生成 Web 应用。然后,它会在一个嵌入式的 Tomcat 中启动 Spring 应用上下文,并且使用默认的 8080 端口。打开浏览器并导航到给定的 URL,随后将会加载一个页面并展现简单的文本响应:“hello”。提供默认应用上下文以及嵌入式容器的这些过程,能够让开发人员更加关注于开发应用以及业务逻辑,从而不用再关心繁琐的样板式配置。

Boot 能够自动确定类所需的功能,这一点使其成为了强大的快速应用开发工具。当应用在 Boot CLI 中执行时,它们在使用内部的 Groovy 编译器进行构建,这个编译器可以在字节码生成的时候以编码的方式探查并修改类。通过这种方式,使用 CLI 的开发人员不仅可以省去定义默认配置,在一定程度上甚至可以不用定义特定的导入语句,它们可以在编译的过程中识别出来并自动进行添加。除此之外,当应用在 CLI 中运行时,Groovy 内置的依赖管理器,“Grape”,将会解析编译期和运行时的类路径依赖,与Boot 编译器的自动配置机制类似。这种方式不仅使得框架更加对用户友好,而且能够让不同版本的Spring Boot 与特定版本的来自于Spring IO 平台的库相匹配,这样一来开发人员就不用关心如何管理复杂的依赖图和版本结构了。另外,它还有助于快速原型的开发并生成概念原型的工程代码。

对于不是使用CLI 构建的工程,Boot 提供了许多的“starter”模块,它们定义了一组依赖,这些依赖能够添加到构建系统之中,从而解析框架及其父平台所需的特定类库。例如, spring-boot-starter-actuator依赖会引入一组基本的 Spring 项目,从而实现应用的快速配置和即时可用。关于这种依赖,值得强调的一点就是当开发 Web 应用,尤其是 RESTful Web 服务的时候,如果包含了spring-boot-starter-web依赖,它就会为你提供启动嵌入式 Tomcat 容器的自动化配置,并且提供对微服务应用有价值的端点信息,如服务器信息、应用指标(metrics)以及环境详情。除此之外,如果引入spring-boot-starter-security模块的话,actuator 会自动配置 Spring Security ,从而为应用提供基本的认证以及其他高级的安全特性。它还会为应用结构引入一个内部的审计框架,这个框架可以用来生成报告或其他的用途,比如开发认证失败的锁定策略。

为了阐述在 Java Maven 工程中,如何快速地使 Spring Web 工程准备就绪,考虑一下程序清单 1.3 中的应用程序代码。

程序清单 1.3

复制代码
package com.infoq.springboot;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.EnableAutoConfiguration;
import org.springframework.web.bind.annotation.*;
@RestController
@EnableAutoConfiguration
public class Application {
@RequestMapping("/")
public String home() {
return "Hello";
}
public static void main(String[] args) {
SpringApplication.run(Application.class, args);
}
}

Application类上的@EnableAutoConfiguration注解会告知 Boot 要采用一种特定的方式来对应用进行配置。这种方法会将其他样板式的配置均假设为框架默认的约定,因此能够聚焦于如何尽快地使应用准备就绪以便运行起来。Application类是可运行的,因此,当我们以 Java Application 的方式运行这个类时,就能启动该应用及其嵌入式的容器,这样也能实现即时地开发。

为了发布版本而构建工程时,Boot 的 Maven 和 Gradle 插件可以嵌入(hook)到这些构建系统的打包过程中,以生成可执行的“胖 jar 包(fat jar)”,这种 jar 包含了工程的所有依赖并且能够以可运行 jar 的方式执行。使用 Maven 打包 Boot 应用只需运行mvn package命令,与之类似,使用 Gradle 时,执行gradle build命令将会在构建的目标地址下生成可运行的 jar。

开发微服务

Boot 对 Spring 应用的开发进行了简化,提供了模块化方式导入依赖的能力,强调了开发 RESTful Web 服务的功能并提供了生成可运行 jar 的能力,这一切都清晰地表明在开发可部署的微服务方面 Boot 框架是一个强大的工具。正如前面的例子所示,借助于 Boot,让一个 RESTful Web 工程运行起来是一件很容易的事情;不过,为了了解 Boot 所有潜在的功能,我们会阐述在开发完整功能的微服务时,会遇到的所有繁琐的事情。在企业级基础设施领域,微服务是一种越来越流行的应用架构,因为它能够实现快速开发、更小的代码库、企业级集成以及模块化部署。有众多的框架致力于该领域的开发,该章节将会讨论使用 Boot 如何简化这一过程。

数据访问

我们可以基于各种目的来构建微服务,但有一点是肯定的,那就是大多数都需要读取和写入数据库的能力。Spring Boot 使数据库集成变成了一项非常简单的任务,因为它具有自动配置 Spring Data 以访问数据库的能力。只需在你的工程中将spring-boot-starter-data-jpa包含进来,Boot 的自动配置引擎就能探测到你的工程需要数据访问功能,并且会在 Spring 应用上下文中创建必要的 Bean,这样你就可以使用 Repository 和服务了。为了更具体地阐述这一点,请参见程序清单 1.4 中的 Gradle 构建文件,它列出了一个基于 Groovy 的微服务 web 应用的构建结构,该应用使用了 Spring Data 对 JPA 的支持来实现数据访问。

程序清单 1.4

复制代码
buildscript {
repositories {
maven { url "http://repo.spring.io/libs-snapshot" }
mavenCentral()
}
dependencies {
classpath("org.springframework.boot:spring-boot-gradle-plugin:1.0.0.RC1")
}
}
apply plugin: 'groovy'
apply plugin: 'spring-boot'
repositories {
mavenCentral()
maven { url "http://repo.spring.io/libs-snapshot" }
}
ext {
springBootVersion = "1.0.0.RC1"
}
dependencies {
compile 'org.codehaus.groovy:groovy-all:2.2.1'
compile "org.springframework.boot:spring-boot-starter-web:$springBootVersion"
compile "org.springframework.boot:spring-boot-starter-data-jpa:$springBootVersion"
compile "org.springframework.boot:spring-boot-starter-actuator:$springBootVersion"
}

在这个配置中,Boot 的actuator模块提供了对hsqldb的依赖,这会搭建所有必要的依赖——包括模式的创建——因此 Spring Data 可以使用这个内存数据库作为数据源。这种简便的方式能够让开发人员免于在开发期创建和管理复杂的 XML 配置,进而能够快速地开发数据库驱动的微服务。如果在 classpath 中有 H2 或 Derby 数据库的话,这种自动化配置也会生效。Boot 所提供的另一个便利之处就是能够快速简便地使用相关数据启动应用的数据库模式。这在开发期是非常有用的,此时数据库可能是在内存中或者是不稳定的,开发人员需要保证的是在应用启动的时候能够访问到这些特定的数据。为了阐述这一点,考虑一下程序清单 1.5 中的示例 JPA 实体,它代表了微服务所提供的“User”数据结构。

程序清单 1.5

复制代码
@Entity
class User {
@Id
@GeneratedValue
Long id
String username
String firstName
String lastName
Date createdDate
Date lastAccessed
Boolean isActive = Boolean.TRUE
}

为了启用代表User对象的通用数据,我们只需创建一个名为schema.sqldata.sql的文件,并将其包含在 classpath 之中。这个文件会在模式创建完成之后执行,所以基于程序清单 1.5 所给出的实体,我们可以使用 SQL 语句启用一个用户账号,如程序清单 1.6 所示。

程序清单 1.6

复制代码
insert into user(username, first_name, last_name, created_date) values ('danveloper', 'Dan', 'Woods', now())

在启动的时候,我们所提供的 SQL 代码会执行,这样就能确保有一个测试账号可以使用。微服务此时已经具有了数据访问的起始点,程序清单 1.7 展现了如何按照 Spring Data 的开发模式创建Repository接口,该接口会作为User实体的数据访问对象(Data Access Object)。

程序清单 1.7

复制代码
public interface UserRepository extends CrudRepository<User, Long> {
}

CrudRepository提供了一些通用的接口方法来创建、查询、更新以及删除对象和对象集合。应用所需的其他特定功能可以按照 Spring Data 的 Repository 开发约定进行定义。一旦UserRepository接口创建成功,Boot 的spring-data-jpa层会在工程中探测到它,并将其添加到 Spring 应用上下文之中,这样对于 controller 和 sevice 对象来说,它就成为可以进行自动注入的可选对象。这种自动化的配置只有在 Boot 应用要求按照这种方式初始化的时候才生效,这是通过存在@EnableAutoConfiguration注解来标识的。借助程序清单 1.8 中所实现的 controller,微服务现在就可以定义 RESTful 端点了,服务的使用者可以获取到 User 的列表或单个 User。

程序清单 1.8

复制代码
@RestController
@EnableAutoConfiguration
@RequestMapping("/user")
class UserController {
@Autowired
UserRepository repository
@RequestMapping(method=[RequestMethod.GET])
def get(Long id) {
id ? repository.findOne(id) : repository.findAll()
}
public static void main(String[] args) {
SpringApplication.run UserController, args
}
}

在启动的时候,应用将会输出日志,表明 Hibernate 按照User实体的定义创建数据库结构,在应用初始化的最后,Boot 还会从schema.sql文件中导入数据。

在开发微服务应用时,需要特别注意的一点是使用了@RequestMapping注解。这不是 Boot 特定的注解。不过,因为 Boot 安装了自己的端点以监控应用的性能、健康情况以及配置,所以需要确保应用的代码不要与这些内置的提供详情的路径解析相冲突。鉴于此,如果有从请求路径中解析属性的需求(在我们的场景中,也就是 user 的id属性),那么我们需要仔细考虑这个动态的属性解析会对微服务的其他行为产生什么影响。在本例中,只是简单地将 controller 映射到/user端点而不是根上下文,就能允许 Boot 的端点也可以进行访问。

微服务所提供的数据并不一定全部适合关系型结构,针对这一点 Spring Boot 也暴露了一些模块,从而让开发人员可以使用 Spring Data 的 MongoDB 和 Redis 项目,不过依然采取特定的方式来进行配置。Spring Data 用来定义数据访问对象(Data Access Object)的高层框架,这样快速切换 JPA 与非 JPA 数据源会变得非常容易。参见程序清单 1.9,它展现了一个重新定义的UserRepository接口,这个接口设计为使用 MongoDB 取代 JPA。

程序清单 1.9

复制代码
public interface UserRepository extends MongoRepository<User, Long> {
}

MongoRepository接口也扩展了CrudRepository,因此微服务的 Controller 代码,也就是程序清单 1.8 所示并不需要修改。为了实现与 MongoDB 的集成,工程唯一要做的就是在应用的 classpath 中包含spring-boot-starter-data-mongodb。程序清单 1.4 所示的 Gradle 构建文件需要稍微调整一下依赖的部分,如程序清单 1.10 所示。

程序清单 1.10

复制代码
dependencies {
compile 'org.codehaus.groovy:groovy-all:2.2.1'
compile "org.springframework.boot:spring-boot-starter-web:$springBootVersion"
compile "org.springframework.boot:spring-boot-starter-data-mongodb:$springBootVersion"
compile "org.springframework.boot:spring-boot-starter-actuator:$springBootVersion"
}

MongoDB 依赖都置于 classpath 之中以后,Boot 将会自动配置 Spring Data 连接到 localhost 上的数据库,并且默认的数据库名为test。在这个库中,将会自动创建User集合(按照 MongoDB 的标准),微服务现在就能使用 MongoDB 作为后端了。对非 JPA 的数据存储来说,初始化数据比其他的方式更为简单,这主要是因为它不能针对 MongoDB 的文档存储和 Redis 的键值存储运行 SQL 文件。鉴于 Spring Data 会使用这些存储的持久化实例,这就意味着开发期创建的数据需要在重启后保留。为了持久化数据,我们需要修改微服务的 controller,这样服务的使用者就能创建User实例了。我们也可以将微服务的UserController进行修改,使其符合通用的 RESTful API 结构,让 controller 以不同的方式处理不同的 HTTP 方法。程序清单 1.11 展现了为 controller 添加创建新User实例的功能。

程序清单 1.11

复制代码
@RestController
@RequestMapping("/user")
@EnableAutoConfiguration
class UserController {
@Autowired
UserRepository repository
@RequestMapping(method=[RequestMethod.GET])
def get(Long id) {
id ? repository.findOne(id) : repository.findAll()
}
@RequestMapping(method=[RequestMethod.POST])
def create(@RequestBody User user) {
repository.save user
user
}
public static void main(String[] args) {
SpringApplication.run UserController, args
}
}

当微服务的使用者往应用的端点上发送一个 HTTP POST 请求时,Spring 将会把请求体转换为User实例。代码接下来会使用UserRepository将这个对象存储到 MongoDB 集合之中。使用 curl 创建User实例的样例如程序清单 1.12 所示。

程序清单 1.12

复制代码
curl -v -H "Content-Type: application/json" -d "{ \"username\": \"danveloper\", \"firstName\": \"Dan\", \"lastName\": \"Woods\", \"createdDate\": \"2014-02-02T00:00:00\" }" http://localhost:8080/user

按照 Boot 针对 Mongo 数据源的特定配置,新的User实例默认会持久化到本地 Mongo 实例的test数据库的user集合之中。如果我们打开 web 浏览器并对微服务发起一个 HTTP GET 请求,我们就能看到所创建的 user 存在于返回的列表之中。

配置

我们可以很快地重写 Spring Boot 的默认配置。默认情况下,应用的配置可以使用 Java 属性文件来进行定义,这个文件名为application.properties并且位于应用的 classpath 根目录下。不过,一种更好的方式是使用 YAML 配置,它提供了结构化以及嵌套的配置。在应用的运行时类路径之中包含snakeyaml之后,你的工程就可以在application.yml文件中直接定义配置了。为了详述这一点,考虑程序清单 1.13 的示例 YAML 配置,这里列出了应用的嵌入式 HTTP 服务器(默认是 Tomcat,也可选择 Jetty)的各种设置项。

程序清单 1.13

复制代码
# Server settings (ServerProperties)
server:
port: 8080
address: 127.0.0.1
sessionTimeout: 30
contextPath: /
# Tomcat specifics
tomcat:
accessLogEnabled: false
protocolHeader: x-forwarded-proto
remoteIpHeader: x-forwarded-for
basedir:
backgroundProcessorDelay: 30 # secs

允许重写 Boot 的自动化配置,这一点能够使你的应用从原型转化为真正的产品,Boot 使用相同的application.yml文件进行配置,这样就会非常容易。自动化配置的指令被设计的尽可能简短,所以当使用actuator构建微服务时,会安装一个配置属性的端点,也就是/configprops,当确定哪些指令需要重写时可以进行参考。如果我们的微服务要使用持久化数据源,如 MySQL ,那么只需将 MySQL 的 Java 驱动添加到运行时 classpath 中,然后在application.yml中添加必要的配置指令即可,如程序清单 1.14 所示。

程序清单 1.14

复制代码
spring:
datasource:
driverClassName: com.mysql.jdbc.Driver
url: jdbc:mysql://localhost:3306/proddb
username: root
password

在一些场景下你可能需要更为灵活的配置,Boot 允许你通过 Java 的系统属性(System properties)重写很多它的默认配置。例如,如果你的应用需要在部署到产品化环境中使用不同的数据库用户,那么 username 配置指令可以通过标准的 Java 系统属性传入到应用之中,而这需要切换到命令行中执行-Dspring.datasource.username=user。关于这一点更为现实的场景是云部署环境,如 Cloud Foundry 或 Heroku,这些平台需要应用启动特定的 HTTP 端口,这一点通过操作系统的环境变量可以实现。Boot 能够从系统属性继承得到配置,这样你的应用就可以在命令行中使用-Dserver.port=$PORT来得到 HTTP 端口。在开发微服务时,这是一种相当有用的特性,因为它可以让微服务应用运行在各种环境配置之中。

外部化配置

微服务必须要支持的很重要的一点就是 _ 外部化 _ 配置。这种配置可以包含任何的内容,从占位符信息到数据库配置等等,在初始规划和构建应用原型时,这是必须要考虑的架构内容。在 Spring IO 平台中,已经存在各种导入配置的策略,但是应用能够以多种方式使用配置所造成的后果往往是产生冗长的编码性耦合。

Boot 一个很棒的特性在于它能管理外部化的配置并将其转换为对象结构,这个对象可以在整个应用上下文中使用。创建一个简单老式的 Java/Groovy 对象(Plain Old Java/Groovy Object),并使用@ConfigurationProperties注解,那么这个对象就能使用 Boot 配置结构中预先定义的name名下的配置项。更具体一点来讲,考虑一下程序清单 1.15 中的 POGO,它能够得到application.key 下的配置指令。

程序清单 1.15

复制代码
@ConfigurationProperties(name = "application")
class ApplicationProperties {
String name
String version
}

ApplicationProperties对象在 Spring 上下文中创建完成之后,Boot 将会识别出它是一个配置对象,并且会按照运行时 classpath 之中application.propertiesapplication.yml文件中的配置指令填充它的属性。因此,如果我们在微服务的application.yml文件中添加application内容区的话,如程序清单 1.16 所示,那么我们就可以在应用的其他部分以编程的方式访问这些配置指令。

程序清单 1.16

复制代码
application:
name: sb-ms-custdepl
version: 0.1-CUSTOMER

这些配置指令可以有各种用途,要访问这些指令的唯一要求就是代表它们的 POJO/POGO 必须是 Spring 应用上下文的成员。Boot 能够将一个 controller 作为 Spring Java 配置对象,这样就能很容易地管理配置 bean 与应用上下文的集成,如程序清单 1.17 所示。

程序清单 1.17

复制代码
@RestController
@Configuration
@RequestMapping("/appinfo")
@EnableAutoConfiguration
class AppInfoController {
@Autowired
ApplicationProperties applicationProperties
@RequestMapping(method=[RequestMethod.GET])
def get() {
[
name: applicationProperties.name,
version: applicationProperties.version
]
}
@Bean
ApplicationProperties applicationProperties() {
new ApplicationProperties()
}
public static void main(String[] args) {
SpringApplication.run UserController, args
}
}

程序清单 1.17 中的样例代码可能有些牵强,不过,即便是在更为复杂的场景下,如何使用 Boot 来访问应用特定配置的原则是相同的。配置类也支持嵌套式的对象图,这样来自于配置中的深层数据就能更便利地进行访问,也有了更好的语义。例如,如果我们想要得到的配置指令是application.根下的那些 metrics key,那么可以在ApplicationProperties POGO 中添加一个嵌套对象来表示这些值,如程序清单 1.18 所示。

程序清单 1.18

复制代码
@ConfigurationProperties(name = "application")
class ApplicationProperties {
String name
String version
final Metrics metrics = new Metrics()
static class Metrics {
String dbExecutionTimeKey
}
}

现在,我们的application.yml文件可以如程序清单 1.19 所示,它在application.代码块中包含了metrics配置。

程序清单 1.19

复制代码
application:
name: sb-ms-custdepl
version: 0.1-CUSTOMER
metrics:
dbExecutionTimeKey: user.get.db.time

当我们需要访问application.metrics.dbExecutionTimeKey的值时,能够以编程的方式通过ApplicationProperties对象来进行访问。

为了在整个应用之中使用application.propertiesapplication.yml文件中的这些配置指令,我们并不是必须要将其转换为对象图。Boot 也为 Spring 应用上下文提供了PropertySourcesPlaceholderConfiguration,这样的话,来自于application.propertiesapplication.yml文件的指令或者来自于 Java 系统的重写属性都可以作为 Spring 属性占位符来使用。Spring 的这种机制能够让你以一种特定的语法来为属性定义占位符值,如果 Spring 发现了占位符配置的话,就会用这个配置来进行填充。作为示例,我们可以在 controller 中使用@Value注解来直接访问application.metrics.dbExecutionTimeKey,如程序清单 1.20 所示。

程序清单 1.20

复制代码
@RestController
@RequestMapping("/user")
@EnableAutoConfiguration
class UserController {
@Autowired
UserRepository repository
@Autowired
GaugeService gaugeService
@Value('${application.metrics.dbExecutionTimeKey}')
String dbExecutionKey
@RequestMapping(method=[RequestMethod.GET])
def get(Long id) {
def start = new Date().time
def result = id ? repository.findOne(id) : repository.findAll()
gaugeService.submit dbExecutionKey, new Date().time - start
result
}
public static void main(String[] args) {
SpringApplication.run UserController, args
}
}

关于应用指标的报告,后面会有更为详细的介绍,但现在重要的一点在于,理解@Value注解如何与 Spring 属性占位符一起使用,使 Boot 能够自动注入值,从而满足这个微服务的特定配置需求。

安全

在微服务的开发中,对于完备安全场景的需求会持续增长。为了满足这种需求,Boot 引入了强大完整的 Spring Security,并且提供了自动配置的功能,以快速简便地启用安全层。只需在应用的 classpath 中包含spring-boot-starter-security模块就能使 Boot 引入一些安全特性,如跨站脚本防护(cross-site scripting protection)并且会添加头信息以防止点击劫持(click-jacking)。除此之外,添加一条简单的配置指令就能启用基本认证来保护你的应用,如程序清单 1.21 所示。

程序清单 1.21

复制代码
security:
basic:
enabled: true

Boot 会为你提供一个默认的用户账号user和默认角色USER,并且会在应用启动的时候在控制台上输出随机生成的密码。就像 Boot 的其他功能那样,对于内置的user账号,我们可以很容易地指定不同的用户名和密码(分别为“secured”和“foo”),这需要通过明确定义的配置指令来实现,如程序清单 1.22 所示。

程序清单 1.22

复制代码
security:
basic:
enabled: true
user:
name: secured
password: foo

对于简单的内部应用或开发原型来说,Boot 内置的基础设施能够快速地在微服务中启用基本认证,这是非常有用的。随着需求的演化,你的应用毫无疑问会需要更细粒度的安全特性,如保护端点只能由特定的角色访问。从这个角度来看,我们可能希望具有USER角色的调用者只能读取数据(即GET请求),而对具有ADMIN角色的调用者可以读取和写入数据(即POST请求)。为了做到这一点,我们需要在工程的application.yml文件中禁用 Boot 的基本认证自动配置功能,并且定义我们自己的useradmin账号以及对应的角色。当你的需求超过 Boot 所提供的默认功能时,它通常很快就能实现,这可以作为佐证这一点的又一个例子。为了更具体地阐述这一点,考虑一下程序清单 1.23 中的代码。这个样例可以阐述如何发挥 Spring Security 所有潜在的功能以及更为复杂的认证策略,如基于 JDBC 后端、OpenID 或单点登录(Single-Sign On)。

程序清单 1.23

复制代码
@RestController
@RequestMapping("/user")
@Configuration
@EnableGlobalMethodSecurity(securedEnabled = true)
@EnableAutoConfiguration
class UserController extends WebSecurityConfigurerAdapter {
@Autowired
UserRepository repository
@RequestMapping(method = [GET])
@Secured(['ROLE_USER'])
def get(Long id) {
id ? repository.findOne(id) : repository.findAll()
}
@RequestMapping(method = [POST])
@Secured(['ROLE_ADMIN'])
def create(@RequestBody User user) {
repository.save user
user
}
@Override
void configure(AuthenticationManagerBuilder auth) {
auth
.inMemoryAuthentication()
.withUser "user" password "password" roles "USER" and() withUser "admin" password "password" roles "USER", "ADMIN"
}
@Override
void configure(HttpSecurity http) throws Exception {
BasicAuthenticationEntryPoint entryPoint = new BasicAuthenticationEntryPoint()
entryPoint.realmName = "Spring Boot"
http.exceptionHandling().authenticationEntryPoint(entryPoint)
http.requestMatchers().antMatchers("/**").anyRequest()
.and().httpBasic().and().anonymous().disable().csrf().disable()
}
public static void main(String[] args) {
SpringApplication.run UserController, args
}
}

在程序清单 1.23 的样例之中,应用现在被明确地配置为要基于useradmin用户账号进行访问,它们的密码都是password,具有的角色分别是USERADMIN。微服务的GETPOST端点分别通过USERADMIN角色进行保护,这就意味着普通用户可以访问只读的数据,而执行读取 - 写入操作的话,需要admin用户凭证。

对于微服务来说,基本认证是很好的一个选择,因为它遵循了很实用且广泛使用的认证协议。换句话说,很多的 API 调用者,包括移动应用,能够很容易地使用这一点来访问你的微服务。当你的认证需求超过了基本认证的功能时(如 OpenID 或 OAuth),微服务可以使用 Spring Security 的全部功能来满足你的需求。

消息集成

在任何的应用中,消息(messaging)都是一种很强大的工具,在一点上,微服务当然也不能例外。使用消息驱动架构开发的应用能够更好地支持可重用性和扩展性。Spring Boot 能够让开发人员在编写微服务时将消息作为架构的核心组成部分,它使用到了 Spring IO 平台的企业集成模式(Enterprise Integration Patterns)实现,即 Spring Integration。Spring Integration 提供了开发消息驱动架构的基本结构,以及与分布式企业平台集成的模块。这种能力使得微服务可以使用来自抽象消息源的业务对象,这些消息源可以在应用内部,也可能是组织机构内部的其他服务所提供的。

尽管 Boot 并没有提供明确的 Spring 上下文自动化配置,但是它为 Spring Integration 提供了一个 starter 模块,它会负责引入 Spring Integration 项目的一系列依赖。这些依赖包括 Spring Integration 的核心库(Core library)、HTTP 模块(用来进行面向 HTTP 的企业集成)、IP 模块(用来进行基于 Socket 的集成操作)、File 模块(用于进行文件系统集成)以及 Stream 模块(用于支持使用 Stream 的操作,如 stdin 和 stdout)。这个 starter 模块为开发人员提供了健壮的消息功能的工具集,可以使已有的基础设施适应微服务 API。

除了 starter 模块,Boot 也为通过 CLI 构建的应用提供了编译器自动配置的功能。对于需要快速构建微服务原型并验证可行性的开发者来说,这种方式提供了一些捷径。使用企业级平台的应用可以快速地进行开发,在转移到正式的工程和构建系统之前,就能确认其价值。使用 Spring Boot 和 Spring Integration 使一个消息驱动的微服务运行起来非常简单,如程序清单 1.24 的样例代码所示。

程序清单 1.24

复制代码
@RestController
@EnableIntegrationPatterns
class App {
@Bean
def userLookupChannel() {
new DirectChannel()
}
@Bean
def userTemplate() {
new MessagingTemplate(userLookupChannel())
}
@RequestMapping(method=[RequestMethod.GET])
def get(@RequestParam(required=false) Long id) {
userTemplate().convertSendAndReceive( id ? id : "")
}
}
class User {
Long id
}
@MessageEndpoint
class UserLookupObject {
@ServiceActivator(inputChannel="userLookupChannel")
def get(Long id) {
id ? new User(id:id) : new User()
}
}

使用消息驱动的方式来进行微服务的开发能提供很大的代码可重用性,并且能够与底层的服务提供者实现相解耦。在更为正式的场景之中,程序清单 1.24 的代码可能会负责组合数据,这些数据可能来自于数据库调用和企业组织中某个外部的服务集成。Spring Integration 具有内置的设施用来进行负载路由(payload routing)和处理器链(handler chaining),这对于组合不同的数据来说,是一个很有吸引力的方案,我们的微服务可以作为一个数据的提供者(provider)。

提供度量指标

微服务最重要的一个特性可能就是为报表终端(reporting agent)提供度量指标。不像那些功能完备的 Web 应用,微服务是轻量级的,设计时可能就不会规划提供报表界面或完备的接口来分析服务的活动。这种类型的操作最好是留给专门进行数据聚合和分析的应用,这些数据能够用来进行稳定性、性能以及商务智能的监控。基于这样的前提,微服务应该为这些工具提供端点,从而更加容易地获取有关该服务活动的数据。而报表工具负责将数据聚合到一个视图或报告中,对于关心数据的人这才是有意义的。

微服务的一些指标如稳定性和性能,对所有的应用都是通用的,但是与业务操作相关的指标必须由应用本身来具体进行管理。针对这一点,Spring Boot 的actuator模块为开发人员提供了一种机制,允许开发人员通过/metrics端点以编码的方式暴露微服务状态的细节。Boot 将指标拆分为“counter”和“gauge”两种类别:counter 是所有以 Number 类型来展现的指标,而 gauge 是衡量双精度计算的指标。为了让微服务的开发人员更加容易地使用指标,Boot 暴露了CounterServiceGaugeService,它们可以自动织入到应用上下文之中。请参见程序清单 1.25 的样例,它阐述了如何通过CounterService对外暴露点击数。

程序清单 1.25

复制代码
@RestController
@RequestMapping("/user")
@EnableAutoConfiguration
class UserController {
@Autowired
UserRepository repository
@Autowired
CounterService counterService
@RequestMapping(method = [GET])
def get() {
get(null)
}
@RequestMapping(value="/{id}", method = [GET])
def get(@PathVariable Long id) {
counterService.increment id ? "queries.by.id.$id" : "queries.without.id"
id ? repository.findOne(id) : repository.findAll()
}
}

在点击/user端点时,有可能提供 ID 也有可能不提供 ID,/metrics端点都会在counter.父节点下记录新的 key。例如,如果我们只是查询/user端点而不带有 ID 的话,那么就会注册counter.queries.without.id指标。类似的,如果我们带有 ID 的话,那么就会看到有一个counter.queries.by.id.<id>的 key,它能用来标记对于给定的 ID 已经进行了多少次查询。这些指标可能会有助于掌握最经常访问的User对象并指导要采取的行为,如缓存或数据库索引。类似于递增指标的数值,CounterService也允许将指标的值将为零。这对于跟踪打开的连接数或其他频率分布(histographic)的测量都是很有用处的。

gauge 是稍微有所不同的一种类型指标,它会进行探索性的计算或基于请求来确定值。如GaugeService的 JavaDocs 所述,“gauge”可以测量任意的值,从方法执行的次数到会议室的温度。当需要为报表工具暴露细节时,这种类型的测量尤其适合于使用GaugeService。gauge 的指标会在/metrics端点之下进行访问,并且带有gauge.前缀。它们的注册方式与 counter 有些差别,如程序清单 1.26 所示。

程序清单 1.26

复制代码
@RestController
@RequestMapping("/user")
@EnableAutoConfiguration
class UserController {
@Autowired
UserRepository repository
@Autowired
CounterService counterService
@RequestMapping(method = [GET])
def get() {
get(null)
}
@RequestMapping(value="/{id}", method = [GET])
def get(@PathVariable Long id) {
def start = new Date().time
def result = id ? repository.findOne(id) : repository.findAll()
def time = new Date().time - start
gaugeService.submit("user.get.db.time", time.doubleValue())
result
}
}

默认情况下,指标会存储在一个易失的内存数据库之中,但 Boot 同时也为应用上下文提供了MetricsRepository实现,它能支持更为持久化的行为。Boot 自带了一个RedisMetricsRepository,它能够自动织入进来,从而将指标存储到 Redis 值存储之中,另外,可以编写自定义的实现将指标存储到任意的数据存储形式之中。

Boot 还提供了对 Coda Hale Metrics 库的支持,它会将以特定名称开头的指标强制转换为对应的 Metrics 类型。例如,如果有一个指标是以histogram.开头,那么这个值将会作为Histogram对象类型。这种自动化的强制转换对于meter.timer.key 也是有效的,而普通的指标将会作为Gauge类型。

一旦微服务的指标在 Boot 中进行了注册,那么报表工具就可以通过/metrics端点来检索它们。已命名的指标可以通过/metrics端点获取,只需将指标的 key 名作为查询字符串的一部分即可。例如,如果 _ 只是 _ 访问 gauge 指标下的“user.get.db.time”,报表工具可以针对/metrics/gauge.user.get.db.time进行查询。

打包 Boot 应用

正如前面所讨论的,Boot 提供了 Maven 和 Gradle 插件,它为构建系统的打包阶段提供了一种钩子(hook),以产生所谓的“胖 jar”,在这种 jar 中包含了工程的所有依赖。当这个胖 jar 包执行时,应用将会运行在与工程开发期相同的嵌入式容器之中。这种简便的方式能够让开发人员省去很多麻烦,因为他们的部署包在开发期和运行时环境之中具有相同的依赖结构。这也能够缓解运维团队的焦虑,他们不用担心部署的场景,因为在部署时一个错误配置的运行时容器可能会带有某个特定的依赖,而在项目的开发期所依赖的可能是另外一个。

为了在 Maven 下执行打包,只需执行mvn package命令。Spring Boot 的插件会备份工程所创建的原始 jar 并且在文件名上添加“.original”。在这里,能够得到可运行的 jar,文件符合 Maven artifact 的命名约定,它可以按照工程最合适的方式进行部署。使用 Gradle 构建 Boot 工程同样很简单,只需执行标准的gradle build命令即可。类似于 Maven,Boot 插件在原有的打包任务之后使用 Gradle 安装了一个生命周期事件,并且会在build/libs目录下创建胖 jar 包。对所生成的胖 jar 包进行检查的一种方式就是所有依赖的 jar 都会位于归档文件的lib/目录下。

打包完成之后,胖 jar 包就能够像其他可运行的 jar 文件那样在命令行中执行了,也就是使用$JAVA_HOME/bin/java -jar path/to/myproject.jar命令。启动后,Boot 应用的日志将会显示在控制台上。

对于需要部署到传统 servlet 容器之中的应用,Boot 提供了一种方式以编码的方式初始化 Web 配置。为了使用这一点,Boot 提供了可选的WebApplicationInitializer,它会使用 servlet 容器来注册应用,这会通过 Servlet 3.0 API 以编码的方式注册 servlet 并且会用到ServletContext。通过提供SpringBootServletInitializer的子类,Boot 应用能够使用嵌入的 Spring 上下文来注册配置,这个 Spring 上下文是在容器初始化的时候创建的。为了阐述这个功能,考虑程序清单 1.27 中的示例代码。

程序清单 1.27

复制代码
@RestController
@EnableAutoConfiguration
class Application extends SpringBootServletInitializer {
@RequestMapping(method = RequestMethod.GET)
String get() {
"home"
}
static void main(String[] args) {
SpringApplication.run this, args
}
@Override
SpringApplicationBuilder configure(SpringApplicationBuilder application) {
application.sources Application
}
}

Application类中被重写的configure方法就是使用嵌入式的 Spring 上下文注册应用的地方。在更为正式的场景之中,这个方法可能会用来注册 Spring Java 配置类,它会定义应用中所有 controller 和服务的 bean。

当将应用打包部署到 servlet 容器之中时,工程要构建为一个 war 文件。在 Maven 工程中,为了适应这一点,需要移除 Boot 插件,并且 packaging 需要明确定义为“war”类型,如程序清单 1.28 所示。

程序清单 1.28

复制代码
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>com.example</groupId>
<artifactId>myproject</artifactId>
<version>1.0.0-SNAPSHOT</version>
<packaging>war</packaging>
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>1.0.0.RC1</version>
</parent>
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-actuator</artifactId>
</dependency>
</dependencies>
<repositories>
<repository>
<id>spring-snapshots</id>
<url>http://repo.spring.io/libs-snapshot</url>
</repository>
</repositories>
</project>

对这个工程执行mvn install命令会在target目录下生成myproject-1.0.0-SNAPSHOT.war文件。使用 Gradle 构建的工程可以使用 Gradle War Plugin,它为构建 war 文件暴露了一个war任务。类似于 Maven 的配置,Boot Gradle 工程也需要移除所包含的 Boot 插件。产生 war 文件的示例 Gradle 构建脚本如程序清单 1.29 所示。

程序清单 1.29

复制代码
apply plugin: 'java'
apply plugin: 'war'
repositories {
mavenCentral()
maven { url "http://repo.spring.io/snapshot" }
maven { url "http://repo.spring.io/milestone" }
}
ext {
springBootVersion = '1.0.0.BUILD-SNAPSHOT'
}
dependencies {
compile "org.springframework.boot:spring-boot-starter-web:${springBootVersion}"
compile "org.springframework.boot:spring-boot-starter-actuator:${springBootVersion}"
}

对于 Boot 工程,使用这个构建脚本运行 Gradle 的war任务将会在build/libs目录下产生 war 文件。

不管是 Maven 还是 Gradle 的配置,一旦 war 文件产生,它就可以部署到任意兼容 Servlet 3.0 的应用服务器之中。部分兼容的容器包括 Tomcat 7+、Jetty 8、Glassfish 3.x、JBoss AS 6.x/7.x 以及 Websphere 8.0。

延伸阅读

Spring Boot 团队已经编写了完整的指导和样例来阐述框架的功能。Blog 文章、参考资料以及API 文档都可以在 Spring.IO 网站上找到。项目的GitHub 页面上可以找到示例的工程,更为具体的细节可以阅读 Spring Boot 的参考手册。SpringSourceDev YouTube 频道有一个关于Spring Boot 的webinar ,它概述了这个项目的目标和功能。在去年在伦敦举行的Groovy & Grails Exchange 上,David Dawson 做了一个使用Spring Boot开发微服务的演讲

关于作者

Daniel Woods是 Netflix 的高级软件工程师,负责开发持续交付和云部署工具。他擅长 JVM 栈相关的技术,活跃在 Groovy、Grails 和 Spring 社区。可以通过电子邮件地址 danielpwoods@gmail.com 或 Twitter @danveloper 联系到 Daniel。

关注 IT 趋势,承载前沿、深入、有温度的内容。感兴趣的读者可以搜索 ID:laocuixiabian,或者扫描下方二维码加关注。

查看原文链接: Exploring Micro-frameworks: Spring Boot

2014-05-12 12:56547551

评论

发布
暂无评论
发现更多内容

2021阿里Java高级面试题总结,Dubbo高频面试题+解析

Java 程序员 后端

21年Java面经分享,Java面试知识点总结宝典助你通关

Java 程序员 后端

35岁老年程序员的绝地翻身之路,Java面试重点问题

Java 程序员 后端

2021金九银十面试季,java零基础入门视频教程,成功入职腾讯

Java 程序员 后端

区块链上升为国家战略两周年后 看浪潮下企业如何创新数字化应用

CECBC

推荐两款工具给爱做实验的人

Java 开源 编程 架构

25K大牛甩出的超详细面试总结,给班出身的程序员一些建议

Java 程序员 后端

2面技术+HR面+offer,成功入职头条月薪35K

Java 程序员 后端

35岁程序员的人生感悟,mongodb入门教程,阿里Java高级工程师面试题

Java 程序员 后端

3分钟就能完成的Redis主从复制搭建,10天拿到阿里Java岗offer

Java 程序员 后端

4面技术5面HR附加笔试面,初级Java面试题大全

Java 程序员 后端

4面技术5面HR附加笔试面,面试的时候突然遇到答不上的问题怎么办

Java 程序员 后端

4面阿里拿到P7Offer,SpringSecurity如何实现加密和解码

Java 程序员 后端

60分钟快速掌握RabbitMQ,Java基础全套视频教程

Java 程序员 后端

先行一步,7大技术创新和突破,阿里云把 Serverless 领域的这些难题都给解了

阿里巴巴云原生

阿里云 Serverless 云原生 云栖大会

35岁技术人如何转型做管理,mybatis使用教程,Java全套视频

Java 程序员 后端

35岁技术人如何转型做管理?牛客网中级项目笔记,Java高级工程师必备知识

Java 程序员 后端

38岁的中年失业者怎么活下去,Java中级工程师面试题及答案

Java 程序员 后端

不要让孩子在12岁之前接触手机游戏

石云升

育儿 10月月更

5年crud经验,【微信小程序】

Java 程序员 后端

数字货币能改变国际货币体系吗?

CECBC

2面技术+HR面+offer,从头到尾,都是精华

Java 程序员 后端

30岁以后搞Java已经没有前途,Java经典排序算法

Java 程序员 后端

30岁以后搞Java已经没有前途,java自学编程入门教程,大V推荐

Java 程序员 后端

阿里技术官终于把这份万字面试手册整理出来了,在Github上获赞89.7K

Java 编程 程序员 架构 面试

3年内被辞退5次,35岁程序员该何去何从,Java工程师必备知识

Java 程序员 后端

4个改变你编程技能的小技巧,附答案解析

Java 程序员 后端

4面字节跳动拿到Offer,尚学堂java视频下载,初级Java面试题大全

Java 程序员 后端

云栖掠影|回首开源十年,RocketMQ 焕发新生

阿里巴巴云原生

阿里云 RocketMQ 云原生

46道面试题带你了解高级Java面试,linux教程视频合集

Java 程序员 后端

4个改变你编程技能的小技巧,非科班生金九银十求职经历

Java 程序员 后端

深入学习微框架:Spring Boot_Java_Dan Woods_InfoQ精选文章