UP | HOME

Table of Contents

Spring Framework

Table of ContentsClose

1 简介

Spring Framework 是一个开源框架,是为了解决企业应用程序开发复杂性而创建的。框 架的主要优势之一就是其分层架构,分层架构允许您选择使用哪一个组件,同时为 J2EE 应用程序开发提供集成的框架。

spring-projects.png

2 Spring 的技术点和优点

作为 Java Web 开发的龙头老大, Spring 的优点是不需要打广告了,Spring Framework 官网上给出了一下技术要点

  1. 核心技术:依赖注入,事件,资源,i18n,验证,数据绑定,类型转换,SpEL,AOP
  2. 测试:模拟对象,TestContext 框架,Spring MVC 测试,WebTestClient
  3. 数据访问:事务,DAO 支持,JDBC,ORM,编组 XML
  4. Spring MVC 和 SpringWebFlux Web 框架
  5. 整合:远程处理,JMS,JCA,JMX,电子邮件,任务,调度,缓存
  6. 语言:Kotlin,Groovy,动态语言

springmvc-documentation.png

Spring 的技术点很多,被很多人在工作中应用。这显然是离不开 Spring 团队优秀的设 计,以下是 Spring Framework 的几个比较突出的优点:

  1. 轻量级:Spring 在大小和透明性方面绝对属于轻量级的,基础版本的 Spring 框架 大约只有 2MB
  2. 控制反转(IoC):Spring 使用控制反转技术实现了松耦合。依赖被注入到对象,而不 是创建或寻找依赖对象
  3. 面向切面编程(AOP): Spring 支持面向切面编程,同时把应用的业务逻辑与系统的 服务分离开来
  4. 容器:Spring 包含并管理应用程序对象的配置及生命周期
  5. MVC 框架:Spring 的 Web 框架是一个设计优良的 Web MVC 框架,很好的取代了一 些 Web 框架
  6. 事务管理:Spring 对下至本地业务上至全局业务(JAT)提供了统一的事务管理接口
  7. 异常处理:Spring 提供一个方便的 API 将特定技术的异常(由 JDBC, Hibernate,或 JDO 抛出)转化为一致的、Unchecked 异常

3 IoC (Inversion of Control)

  • IoC 也被称为 DI (dependency injection),是实现 bean 的依赖关系,即在初始化 一个 IoC 容器时,先将其依赖的容器一并初始化,并注入的新的容器中。
  • IoC 的好处是可以自动管理 Java 类的示例,提升开发速度
  • Spring 管类的实例叫 Bean,Spring 提供了一系列方法来管理 Bean,其中包括:
    • BeanFactory Bean 的工厂,ApplicationContext 应用运行的上下文。注意: ApplicationContext 和 BeanFactory 都能实例化 Bean,但是一般尽量使用 ApplicationContext,因为 ApplicationContext 还额外提供了一些其他功能,如: 继承生命周期管理,BeanPostProcessor 自动注册,BeanFactoryPostProcessor 等
    • 还有一些 XML 配置类的解析,XML 配置相当于 Spring 构建 Bean 的蓝图,早期 Spring 都用 XML 来配置类,但是由于太复杂,现在已经被 Java 的注解所代替

4 创建 Spring 项目

创建一个 Maven 项目项目工程

mvn archetype:generate \
    -DarchetypeArtifactId=maven-archetype-quickstart \
    -DinteractiveMode=false -DgroupId=io.github.jeanhwea -DartifactId=mapp

添加 spring-context 的项目依赖到 pom.xml 文件中

<dependency>
  <groupId>org.springframework</groupId>
  <artifactId>spring-context</artifactId>
  <version>5.2.4.RELEASE</version>
</dependency>

使用命令行的方式运行主类

mvn exec:java -Dexec.mainClass="io.github.jeanhwea.App" -Dexec.classpathScope=runtime

5 容器 Container 和 Bean

Spring Framework 提供了管理 Bean 的接口 ApplicationContext,也就是说 Bean 的 初始化都是在 ApplicationContext 的实现中处理的,比如: ClassPathXmlApplicationContext 和 FileSystemXmlApplicationContext 都对 ApplicationContext 进行实现,这些 ApplicationContext 是装在 Bean 的,所以被称 为容器。

Bean 其实是一种 Java 的对象,在 Spring 中,Bean 除了普通定义的 Java 代码以外, 还需要一些其他的元信息,例如:定义的 XML,或者 @Component 注解。所以 Spring 中的对象和普通 Java 对象是不一样的,为了体现 Spring 的高贵性,就将 Spring 中 的对象称为 Bean。Bean 的 XML 配置非常恶心且繁琐,不需要记的,用的时候去官网上 查就行。

关于 Bean 的使用需要掌握以下几个要点

  • Bean Class, Name, Alias, Constructor arguments, Properties
  • Bean Scope: singlenton, prototype, request, session, application, websocket
  • Reference: ref / refid
  • Collections <list/>, <set/>, <map/>
  • Autowiring: no, byName, byType, constructor
  • *Aware Interfaces: ApplicationContextAware, ResourceLoaderAware, BeanFactoryAware
  • Bean Definition Inheritance: <parent/>
  • Lazy initialization mode
  • Initialization method
  • Initialization/Destruction method
  • Annotation: @Required, @Autowired

6 资源 Resource

Spring 秉承着一切皆资源,将可以被访问的东西都囊括在 Resource 接口中。并提供了 以下一些实现,用户可以直接通过 URL 来获取: classpath:config.xml, file:///data/config.xml, http://myservice/logo.png, /path/to/file.txt

  • UrlResource
  • ClassPathResource
  • FileSystemResource
  • ServletContextResource
  • InputStreamResource
  • ByteArrayResource

7 面向切面编程 AOP

AOP 框架创建的对象,代理就是目标对象的加强。Spring 中的 AOP 代理可以使 JDK 动态代理, 也可以是 CGLIB 代理,前者基于接口,后者基于子类。

  • Aspect 通常是一个类,里面可以定义切入点和通知
  • Join Point 程序执行过程中明确的点,一般是方法的调用
  • Advice 在特定的切入点上执行的增强处理
  • Pointcut 带有通知的连接点,在程序中主要体现为书写切入点表达式
package com.xyz.someapp;

import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Pointcut;

@Aspect
public class SystemArchitecture {

    /
     * A join point is in the web layer if the method is defined
     * in a type in the com.xyz.someapp.web package or any sub-package
     * under that.
     /
    @Pointcut("within(com.xyz.someapp.web..)")
    public void inWebLayer() {}

    /
     * A join point is in the service layer if the method is defined
     * in a type in the com.xyz.someapp.service package or any sub-package
     * under that.
     /
    @Pointcut("within(com.xyz.someapp.service..)")
    public void inServiceLayer() {}

    /
     * A join point is in the data access layer if the method is defined
     * in a type in the com.xyz.someapp.dao package or any sub-package
     * under that.
     /
    @Pointcut("within(com.xyz.someapp.dao..)")
    public void inDataAccessLayer() {}

    /
     * A business service is the execution of any method defined on a service
     * interface. This definition assumes that interfaces are placed in the
     * "service" package, and that implementation types are in sub-packages.
     *
     * If you group service interfaces by functional area (for example,
     * in packages com.xyz.someapp.abc.service and com.xyz.someapp.def.service) then
     * the pointcut expression "execution(* com.xyz.someapp..service..(..))"
     * could be used instead.
     *
     * Alternatively, you can write the expression using the 'bean'
     * PCD, like so "bean(Service)". (This assumes that you have
     * named your Spring service beans in a consistent fashion.)
     */
    @Pointcut("execution( com.xyz.someapp..service..(..))")
    public void businessService() {}

    /*
     * A data access operation is the execution of any method defined on a
     * dao interface. This definition assumes that interfaces are placed in the
     * "dao" package, and that implementation types are in sub-packages.
     */
    @Pointcut("execution( com.xyz.someapp.dao..(..))")
    public void dataAccessOperation() {}

}

8 事务 @Transactional

8.1 @Transactional

事务不仅仅是 Spring 特有的特性,在数据库并发中普遍存在,具体事务的特性我在 Oracle 中事务一节中有详细地讨论,这里着重记录 Spring 中 @Transactional 标注的使 用

在 Spring 中,事务使用操作的相关讨论出现在 Data Access 一节中, @Transactional 开启一个事务,例如下面的 FooService 开启了事务管理,虽然 Spring 也支持 XML 方式配置事务,但是比较麻烦,尽量避免使用

@Transactional
public class DefaultFooService implements FooService {

  Foo getFoo(String fooName) { /* ... */ }
  Foo getFoo(String fooName, String barName) { /* ... */ }
  void insertFoo(Foo foo) { /* ... */ }
  void updateFoo(Foo foo) { /* ... */ }

}

事务显然有很多属性可以配置,例如:是否只读,是否传播,隔离级别,超时的时长, 以及为何中异常回滚,因为属性多,所以需移步官网文档 Transactional Settings

@Transactional(readOnly = true)
public class DefaultFooService implements FooService {
  public Foo getFoo(String fooName) { /* ... */ }

  // these settings have precedence for this method
  @Transactional(readOnly = false, propagation = Propagation.REQUIRES_NEW)
  public void updateFoo(Foo foo) { /* ... */ }
}

在 Spring 中,事务包含一个传播方式,大体上有如下几种:

  • PROPAGATION_REQUIRED 所有方法串在一个事务中执行
  • PROPAGATION_REQUIRES_NEW 每次进入子方法时,新开一个事务
  • PROPAGATION_NESTED 所有方法串在一个事务中,但是进入子方法时记录保存点 (Save Point)

Spring 也支持多事务管理员角色的注解,使用方法参考下面例子

public class TransactionalService {

  @Transactional("order")
  public void setSomething(String name) { /* ... */ }

  @Transactional("account")
  public void doSomething() { /* ... */ }
}

也可以自己定制的方式定义属于业务逻辑的事务管理注解,参考如下的例子

@Target({ElementType.METHOD, ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Transactional("order")
public @interface OrderTx {
}

@Target({ElementType.METHOD, ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Transactional("account")
public @interface AccountTx {
}
public class TransactionalService {
  @OrderTx
  public void setSomething(String name) { /* ... */ }
  @AccountTx
  public void doSomething() { /* ... */ }
}

8.2 TransactionTemplate

Spring 实现了 TransactionTemplate 模版类,也可以很方便的操纵事务相关的业务

public class SimpleService implements Service {

  // single TransactionTemplate shared amongst all methods in this instance
  private final TransactionTemplate transactionTemplate;

  // use constructor-injection to supply the PlatformTransactionManager
  public SimpleService(PlatformTransactionManager transactionManager) {
    this.transactionTemplate = new TransactionTemplate(transactionManager);
  }

  public Object someServiceMethod() {
    return transactionTemplate.execute(new TransactionCallback() {
        // the code in this method executes in a transactional context
        public Object doInTransaction(TransactionStatus status) {
          updateOperation1();
          return resultOfUpdateOperation2();
        }
      });
  }
}

9 测试

Spring 框架的一个特色就是提供测试 Spring Testing 的功能,对于日常工作测试,主 要了解 @Transactional, @Rollback 以及 @Commit 这样三个注解功能。Spring 测试时默认开启事务,如果测试中修改或删除了数据,在测试后要求修改回来。

10 Spring MVC

10.1 简介

Spring MVC 是 Spring Web MVC 通用叫法,由于 spring-webmvc 包的缘故,之前才被 称为 Spring Web MVC。 他提供了开发 MVC 页面的框架,它期初是建立在 Servlet API 基 础上。 我一般讲 Spring MVC 项目组织成如下结构:

spring-mvc-module.png

10.2 核心组件

Spring MVC 是对通用 Servlet 的一个代理,其实现的 DispatcherServlet 提供对请 求预处理的共享算法。WebApplicationContext 是 ApplicationContext 的子类,他除 了继承 Spring 的 IoC 功能以外,提供了许多对 Web 操作的功能。

10.3 Servlet 配置

Spring MVC 提供了配置 Servlet 的方法。实现 WebApplicationinitializer 接口的 类都可以在 Web 应用程序启动时被加载。具体操作如下:

import org.springframework.web.WebApplicationInitializer;
public class MyWebApplicationInitializer implements WebApplicationInitializer {
  @Override
  public void onStartup(ServletContext container) {
    XmlWebApplicationContext appContext = new XmlWebApplicationContext();
    appContext.setConfigLocation("/WEB-INF/spring/dispatcher-config.xml");

    ServletRegistration.Dynamic registration =
      container.addServlet("dispatcher", new DispatcherServlet(appContext));
    registration.setLoadOnStartup(1);
    registration.addMapping("/");
  }
}

10.4 控制器

Spring MVC 默认使用 @Controller 来表示控制器,如果后台开发的大部分返回 Json 的话,我们可以使用 @RestController 来注解,他相当于 @Controller@ResponseBody 这两个的组合。

@Controller
public class HelloController {

  @GetMapping("/hello")
  public String handle(Model model) {
    model.addAttribute("message", "Hello World!");
    return "index";
  }
}

11 HTTP 请求映射

通常,可以通过二级路由的方法管理控制器类,详见如下例子:

@RestController
@RequestMapping("/persons")
class PersonController {

  @GetMapping("/{id}")
  public Person getPerson(@PathVariable Long id) { /* ... */ }

  @PostMapping
  @ResponseStatus(HttpStatus.CREATED)
  public void add(@RequestBody Person person) { /* ... */ }

}

路径参数中包含如下三种通配符:

  • ? 匹配一个字符
  • * 匹配零个或多个路径中的字符
  • ** 匹配零个或多个路径
@GetMapping("/owners/{ownerId}/pets/{petId}")
public Pet findPet(@PathVariable Long ownerId, @PathVariable Long petId) { /* ... */ }

@GetMapping("/pets/{petId}")
public Pet findPet(@PathVariable("petId") id) { /* ... */ }

@GetMapping("/{name:[a-z-]+}-{version:\\d\\.\\d\\.\\d}{ext:\\.[a-z]+}")
public void handle(@PathVariable String version, @PathVariable String ext) { /* ... */ }

@GetMapping("/pets/{petId}")
// same as
@RequestMapping("/pets/{petId}", method=HttpMethod.GET)

匹配 Content-Type 字段的值,将控制器局限到对应的类型的文件处理

@PostMapping(path = "/pets", consumes = "application/json")
public void addPet(@RequestBody Pet pet) { /* ... */ }

@PostMapping(path = "/pets", consumes = "!text/plain")
public void addPet(@RequestBody Pet pet) { /* ... */ }

返回多媒体类型的流,参考 Accept 请求头

@GetMapping(path = "/pets/{petId}", produces = "application/json")
@ResponseBody
public Pet getPet(@PathVariable String petId) { /* ... */ }

根据请求的添加来部分匹配

@GetMapping(path = "/pets/{petId}", params = "myParam=myValue")
public void findPet(@PathVariable String petId) { /* ... */ }

@GetMapping(path = "/pets", headers = "myHeader=myValue")
public void findPet(@PathVariable String petId) { /* ... */}

还可以通过以下的方式来动态注册请求 URL 到处理方法

@Configuration
public class MyConfig {

  @Autowired
  public void setHandlerMapping(RequestMappingHandlerMapping mapping, UserHandler handler)
    throws NoSuchMethodException {

    RequestMappingInfo info = RequestMappingInfo
      .paths("/user/{id}").methods(RequestMethod.GET).build();

    Method method = UserHandler.class.getMethod("getUser", Long.class);

    mapping.registerMapping(info, handler, method);
  }
}

12 响应函数参数和返回值

Method Handler 包括方法参数和返回值注解。其中请求的方法输入参数包括: 常用注解 @RequestParam, @RequestHeader, @RequestBody 。还包括一些类 javax.servlet.ServletRequest, javax.servlet.ServletResponse, HttpMethod

返回值也支持好多种注解已经类,列表等: @ResponseBody, HttpEntity<B>

矩阵变量

// GET /pets/42;q=11;r=22
@GetMapping("/pets/{petId}")
public void findPet(@PathVariable String petId, @MatrixVariable int q) {
  // petId == 42
  // q == 11
}

// GET /owners/42;q=11/pets/21;q=22
@GetMapping("/owners/{ownerId}/pets/{petId}")
public void findPet(@MatrixVariable(name="q", pathVar="ownerId") int q1,
               @MatrixVariable(name="q", pathVar="petId") int q2) {
  // q1 == 11
  // q2 == 22
}

// GET /pets/42
@GetMapping("/pets/{petId}")
public void findPet(@MatrixVariable(required=false, defaultValue="1") int q) {
  // q == 1
}

// GET /owners/42;q=11;r=12/pets/21;q=22;s=23
@GetMapping("/owners/{ownerId}/pets/{petId}")
public void findPet(@MatrixVariable MultiValueMap<String, String> matrixVars,
               @MatrixVariable(pathVar="petId") MultiValueMap<String, String> petMatrixVars) {
  // matrixVars: ["q" : [11,22], "r" : 12, "s" : 23]
  // petMatrixVars: ["q" : 22, "s" : 23]
}

请求参数

@Controller
@RequestMapping("/pets")
public class EditPetForm {
  @GetMapping
  public String setupForm(@RequestParam("petId") int petId, Model model) {
    Pet pet = this.clinic.loadPet(petId);
    model.addAttribute("pet", pet);
    return "petForm";
  }
}

请求头参数

@GetMapping("/demo")
public void handle(@RequestHeader("Accept-Encoding") String encoding,
              @RequestHeader("Keep-Alive") long keepAlive) {
}

Cookie 值

@GetMapping("/demo")
public void handle(@CookieValue("JSESSIONID") String cookie) {
}

上传文件

@Controller
public class FileUploadController {

  @PostMapping("/form")
  public String handleFormUpload(@RequestParam("name") String name,
                            @RequestParam("file") MultipartFile file) {
    if (!file.isEmpty()) {
      byte[] bytes = file.getBytes();
      // store the bytes somewhere
      return "redirect:uploadSuccess";
    }
    return "redirect:uploadFailure";
  }
}

异常处理

@Controller
public class SimpleController {
  @ExceptionHandler
  public ResponseEntity<String> handle01(IOException ex) { /* ... */ }

  @ExceptionHandler({FileSystemException.class, RemoteException.class})
  public ResponseEntity<String> handle02(IOException ex) { /* ... */ }

  @ExceptionHandler({FileSystemException.class, RemoteException.class})
  public ResponseEntity<String> handle(Exception ex) { /* ... */ }
}

全局控制器异常处理,在 RESTFul API 设计中需要优雅的异常处理方式,Spring 建议 可以通过 @ExceptionHandlerResponseEntity 来将返回消息方法返回体中,并 在 @ControllerAdvice 中集中处理。在定义异常实体类时,可以继承 ResponseEntityExceptionHandler 类。

// Target all Controllers annotated with @RestController
@ControllerAdvice(annotations = RestController.class)
public class ExampleAdvice1 {}

// Target all Controllers within specific packages
@ControllerAdvice("org.example.controllers")
public class ExampleAdvice2 {}

// Target all Controllers assignable to specific classes
@ControllerAdvice(assignableTypes = {ControllerInterface.class, AbstractController.class})
public class ExampleAdvice3 {}

13 跨越 CORS @CrossOrigin

CORS 在 MDN 中有详细的介绍,下面只展示 Spring 中如何配置 CORS。允许单个 URL 和处 理函数跨越可以直接使用 @CrossOrigin 来解决

@RestController
@RequestMapping("/account")
public class AccountController {

  @CrossOrigin
  @GetMapping("/{id}")
  public Account retrieve(@PathVariable Long id) { /* ... */ }

  @DeleteMapping("/{id}")
  public void remove(@PathVariable Long id) { /* ... */ }
}

允许类跨域将注解写类头上

@CrossOrigin(origins = "https://domain2.com", maxAge = 3600)
@RestController
@RequestMapping("/account")
public class AccountController {

  @GetMapping("/{id}")
  public Account retrieve(@PathVariable Long id) { /* ... */ }

  @DeleteMapping("/{id}")
  public void remove(@PathVariable Long id) { /* ... */ }
}

或者混合注解方法

@CrossOrigin(maxAge = 3600)
@RestController
@RequestMapping("/account")
public class AccountController {

  @CrossOrigin("https://domain2.com")
  @GetMapping("/{id}")
  public Account retrieve(@PathVariable Long id) { /* ... */ }

  @DeleteMapping("/{id}")
  public void remove(@PathVariable Long id) { /* ... */ }
}

允许全栈跨域需要实现配置类

@Configuration
@EnableWebMvc
public class WebConfig implements WebMvcConfigurer {

  @Override
  public void addCorsMappings(CorsRegistry registry) {

    registry.addMapping("/api/**")
      .allowedOrigins("https://domain2.com")
      .allowedMethods("PUT", "DELETE")
      .allowedHeaders("header1", "header2", "header3")
      .exposedHeaders("header1", "header2")
      .allowCredentials(true).maxAge(3600);

    // Add more mappings...
  }
}

14 MVC 配置 @EnableWebMvc

14.1 类型转换配置

@Configuration
@EnableWebMvc
public class WebConfig implements WebMvcConfigurer {

  @Override
  public void addFormatters(FormatterRegistry registry) {
    DateTimeFormatterRegistrar registrar = new DateTimeFormatterRegistrar();
    registrar.setUseIsoFormat(true);
    registrar.registerFormatters(registry);
  }
}

14.2 静态资源配置

@Configuration
@EnableWebMvc
public class WebConfig implements WebMvcConfigurer {

  @Override
  public void addResourceHandlers(ResourceHandlerRegistry registry) {
    registry.addResourceHandler("/resources/**")
      .addResourceLocations("/public", "classpath:/static/")
      .setCachePeriod(31556926);
  }
}

15 异步请求

15.1 DeferredResult

@GetMapping("/quotes")
@ResponseBody
public DeferredResult<String> quotes() {
  DeferredResult<String> deferredResult = new DeferredResult<String>();
  // Save the deferredResult somewhere..
  return deferredResult;
}

// From some other thread...
deferredResult.setResult(result);

15.2 Callable

@PostMapping
public Callable<String> processUpload(final MultipartFile file) {

  return new Callable<String>() {
    public String call() throws Exception {
      // ...
      return "someView";
    }
  };
}

16 远端过程调用

Spring 集成了许多常用的远端过程调用的方法,在选取的时候需要根据各自的优缺点来 权衡

  • RMI 一般比较快,但是只适合 Java 之间的调用
  • HTTP invoker
  • Hessian 这是工业界调用比较有名的一种远端过程调用的协议
  • JMS (Java Message Service) 具体使用方法参考官方文档 doc
  • JMX (Java Management Extensions) 是一个为应用程序植入管理功能的框架。用户可 以在任何 Java 应用程序中使用这些代理和服务实现管理

17 客户端请求 RestTemplate

Spring 提供了两种 Rest 客户端来进行客户端请求, RestTemplate 和 WebClient。在 日常工作中使用 RestTemplate 比较多,具体可查阅集成一节中的详细叙述 endpoints

  • RestTemplate 是 Spring 自带的调用客户端,支持自带的模板类和同步的 HTTP 调用
  • WebClient 处理异步调用

17.1 初始化

直接使用 Apache HttpComponents 来作为 Spring 的代理来初始化一个 RestTemplate 对象

RestTemplate template = new RestTemplate(new HttpComponentsClientHttpRequestFactory());

新建 RestTemplate 对象,并设置超时为 3 秒

HttpComponentsClientHttpRequestFactory factory = new HttpComponentsClientHttpRequestFactory();
factory.setConnectTimeout(3000);
RestTemplate template = new RestTemplate(factory);

17.2 UriComponents 构造 URI 对象

UriComponentsBuilder 可以新建一个 UriComponents 的实例,示例程序如下

URI uri = UriComponentsBuilder
  .fromUriString("https://example.com/hotels/{hotel}")
  .queryParam("q", "{q}")
  .encode()
  .buildAndExpand("Westin", "123")
  .toUri();

URI uri = UriComponentsBuilder
  .fromUriString("https://example.com/hotels/{hotel}")
  .queryParam("q", "{q}")
  .build("Westin", "123");

17.3 URI 编码

Spring 的 UriComponentsBuilder 自带 encode 方法,方便对 URL 编码

URI uri = UriComponentsBuilder.fromPath("/hotel list/{city}")
  .queryParam("q", "{q}")
  .encode()
  .buildAndExpand("New York", "foo+bar")
  .toUri();
// Result is "/hotel%20list/New%20York?q=foo%2Bbar"

// queryParam 可以处理 query string
URI uri = UriComponentsBuilder.fromPath("/hotel list/{city}")
  .queryParam("q", "{q}")
  .build("New York", "foo+bar");

URI uri = UriComponentsBuilder.fromPath("/hotel list/{city}?q={q}")
  .build("New York", "foo+bar");

// 传递 Map<String, Object> params 对象的 query string
UriComponentsBuilder builder = UriComponentsBuilder.fromHttpUrl(uriTemplate);
params.forEach((k, v) -> { builder.queryParam(k, "{" + k + "}"); });
builder.encode().buildAndExpand(params).toUri(); // 获取到 URI

17.4 URI

restTemplate 支持多种请求模式,例如 GET 请求可以使用下面的方式来出来发送 HTTP 请求来获取数据

// 直接传递参数
String result = restTemplate.getForObject(
    "https://example.com/hotels/{hotel}/bookings/{booking}", String.class, "42", "21");
// 把参数封装在 Map 对象中
Map<String, String> vars = Collections.singletonMap("hotel", "42");
String result = restTemplate.getForObject(
    "https://example.com/hotels/{hotel}/rooms/{hotel}", String.class, vars);
// 自动编码空格
restTemplate.getForObject("https://example.com/hotel list", String.class);
// Results in request to "https://example.com/hotel%20list"

17.5 请求头

使用 exchange() 可以在请求中添加请求头,参考如下示例

String uriTemplate = "https://example.com/hotels/{hotel}";
URI uri = UriComponentsBuilder.fromUriString(uriTemplate).build(42);

RequestEntity<Void> requestEntity = RequestEntity.get(uri)
    .header(("MyRequestHeader", "MyValue")
    .build();

ResponseEntity<String> response = template.exchange(requestEntity, String.class);

String responseHeader = response.getHeaders().getFirst("MyResponseHeader");
String body = response.getBody();

17.6 请求体

直接传递第一个参数作为请求体,在 POST 请求中,输入对象直接被序列化成请求体

URI location = template.postForLocation("https://example.com/people", person);

在 GET 请求中,请求返回的消息体会被反序列化成对应对象

Person person = restTemplate.getForObject("https://example.com/people/{id}", Person.class, 42);

17.7 Jackson JSON

有时候需要修改一下默认请求的序列化方式,例如我们序列化用户是不希望将用户的密 码暴露出来,可以通过下面的方式实现

MappingJacksonValue value = new MappingJacksonValue(new User("eric", "7!jd#h23"));
value.setSerializationView(User.WithoutPasswordView.class);
// 构造请求实体
RequestEntity<MappingJacksonValue> requestEntity =
    RequestEntity.post(new URI("https://example.com/user")).body(value);
// 发送请求,获取回复
ResponseEntity<String> response = template.exchange(requestEntity, String.class);

17.8 Multipart

Spring 还提供了发送多部分请求的方法,使用 MultiValueMap 对象可以构造请求参数

MultiValueMap<String, Object> parts = new LinkedMultiValueMap<>();
// 构造请求的多部分的请求参数
parts.add("fieldPart", "fieldValue");
parts.add("filePart", new FileSystemResource("...logo.png"));
parts.add("jsonPart", new Person("Jason"));
// 构造请求头和添加另外的参数
HttpHeaders headers = new HttpHeaders();
headers.setContentType(MediaType.APPLICATION_XML);
parts.add("xmlPart", new HttpEntity<>(myBean, headers));

一旦多部分的请求参数构造好了,就可以使用如下方法来发送请求

MultiValueMap<String, Object> parts = ...;
template.postForObject("https://example.com/upload", parts, Void.class);

18 发送邮件 Email

在发送邮件前,先准备业务中涉及到邮件的接口,例如定义好订单接口

public interface OrderManager {
  void placeOrder(Order order);
}

通过 Spring 提供的方法发送简单 Email 示例如下

import org.springframework.mail.MailException;
import org.springframework.mail.MailSender;
import org.springframework.mail.SimpleMailMessage;

public class SimpleOrderManager implements OrderManager {

  private MailSender mailSender;
  private SimpleMailMessage templateMessage;

  public void setMailSender(MailSender mailSender) {
    this.mailSender = mailSender;
  }

  public void setTemplateMessage(SimpleMailMessage templateMessage) {
    this.templateMessage = templateMessage;
  }

  public void placeOrder(Order order) {
    // 1. 处理业务逻辑 ...
    // 2. 调用协调模块,处理发送的顺序 ...
    // 3. 复制模板,使用线程安全的模板进入发送队列
    SimpleMailMessage msg = new SimpleMailMessage(this.templateMessage);
    msg.setTo(order.getCustomer().getEmailAddress());
    msg.setText("Dear " + order.getCustomer().getFirstName()
                + order.getCustomer().getLastName()
                + ", thank you for placing order. Your order number is "
                + order.getOrderNumber());
    try{
      this.mailSender.send(msg);
    } catch (MailException ex) {
      // simply log it and go on...
      System.err.println(ex.getMessage());
    }
  }

}

19 执行任务和任务调度

传统的 Spring 中的任务调度通过实现 TaskExecutorTaskScheduler 接口来控 制,为了避免使用 XML 配置。我还可以在使用注解的方式来定义异步任务。要使用任务 需要在配置类中开启异步任务和任务调度,具体如下

@Configuration
@EnableAsync
@EnableScheduling
public class AppConfig {
}

可以使用 @Scheduled 注解来定义周期任务,如下

@Scheduled(fixedDelay=5000)
public void doSomething() {
  // something that should execute periodically
}

@Scheduled(initialDelay=1000, fixedRate=5000)
public void doSomething() {
  // something that should execute periodically
}

@Scheduled(cron="*/5 * * * * MON-FRI")
public void doSomething() {
  // something that should execute on weekdays only
}

可以使用 @Async 注解来定义异步任务

@Async
void doSomething() {
  // this will be executed asynchronously
}

@Async
void doSomething(String s) {
  // this will be executed asynchronously
}

@Async
Future<String> returnSomething(int i) {
  // this will be executed asynchronously
}

@Async 注解的异步任务有时候会发送异常,对于异常处理方式如下

public class MyAsyncUncaughtExceptionHandler implements AsyncUncaughtExceptionHandler {
  @Override
  public void handleUncaughtException(Throwable ex, Method method, Object... params) {
    // 处理异常逻辑
  }
}

20 缓存 Cache

缓存 Cache 使用如下的注解俩定义

  • @Cacheable 标注定义缓存的方法
  • @CacheEvict: 标注删除缓存
  • @CachePut: 标注更新缓存
  • @Caching: 重新组合和定义缓存,一般适用于定义组合式的缓存策略
  • @CacheConfig: 标注在类上,定义类级别共享的缓存

20.1 开启缓存

通过配置类的方式开启缓存

@Configuration
@EnableCaching
public class AppConfig {
}

20.2 @Cacheable

@Cacheable 是用来声明方法是可缓存的。将结果存储到缓存中以便后续使用相同参 数调用时不需执行实际的方法。直接从缓存中取值。最简单的格式需要制定缓存名称。

定义缓存的方法如下

@Cacheable("books")
public Book findBook(ISBN isbn) { /* ... */ }

@Cacheable({"books", "isbns"})
public Book findBook(ISBN isbn) { /* ... */ }

// SpEL
@Cacheable(cacheNames="books", key="#isbn")
public Book findBook(ISBN isbn, boolean checkWarehouse, boolean includeUsed) { /* ... */}
@Cacheable(cacheNames="books", key="#isbn.rawNumber")
public Book findBook(ISBN isbn, boolean checkWarehouse, boolean includeUsed) { /* ... */}
@Cacheable(cacheNames="books", key="T(someType).hash(#isbn)")
public Book findBook(ISBN isbn, boolean checkWarehouse, boolean includeUsed) { /* ... */}

定义同步缓存

@Cacheable(cacheNames="foos", sync=true)
public Foo executeExpensiveOperation(String id) { /* ... */ }

条件缓存

@Cacheable(cacheNames="book", condition="#name.length() < 32")
public Book findBook(String name) { /* ... */ }

@Cacheable(cacheNames="book", condition="#name.length() < 32", unless="#result.hardback")
public Book findBook(String name) { /* ... */ }

@Cacheable(cacheNames="book", condition="#name.length() < 32", unless="#result?.hardback")
public Optional<Book> findBook(String name) { /* ... */ }

20.3 其它注解

@CachePut 也可以声明一个方法支持缓存功能。与 @Cacheable 不同的是使用 @CachePut 标注的方法在执行前不会去检查缓存中是否存在之前执行过的结果,而是 每次都会执行该方法,并将执行结果以键值对的形式存入指定的缓存中

@CachePut(cacheNames="book", key="#isbn")
public Book updateBook(ISBN isbn, BookDescriptor descriptor) { /* ... */}

@CacheEvict 是用来标注在需要清除缓存元素的方法或类上的。当标记在一个类上时 表示其中所有的方法的执行都会触发缓存的清除操作

// allEntries 表示删除所有项目
@CacheEvict(cacheNames="books", allEntries=true)
public void loadBooks(InputStream batch) { /* ... */}

@Caching 注解可以让我们在一个方法或者类上同时指定多个 Spring Cache 相关的 注解。

@Caching(evict = { @CacheEvict("primary"), @CacheEvict(cacheNames="secondary", key="#p0") })
public Book importBooks(String deposit, Date date) { /* ... */}

有时候一个类中可能会有多个缓存操作,而这些缓存操作可能是重复的。这个时候可以 使用 @CacheConfig

@CacheConfig("books")
public class BookRepositoryImpl implements BookRepository {
  @Cacheable
  public Book findBook(ISBN isbn) { /* ... */ }
}

21 参考链接

Last Updated 2021-07-21 Wed 22:09. Created by Jinghui Hu at 2020-03-15 Sun 22:27.