SpringBoot2使用WebFlux函数式编程的方法

本文只是简单使用SpringBoot2使用WebFlux的函数式编程简单使用,后续会继续写关于Webflux相关的文章。

最近一直在研究WebFlux,后续会陆续出一些相关的文章。

首先看一下Srping官网上的一张图,对比一下SpringMvc和Spring WebFlux,如图:

在查看一下WebFlux的官方文档:https://docs.spring.io/spring/docs/current/spring-framework-reference/web-reactive.html,WebFlux提供了函数式编程,本文简单介绍一下WebFlux函数式编程简单使用。

新建项目

创建一个项目,pom文件中引入webflux依赖,完整pom文件如下:

<?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.dalaoyang</groupId>
  <artifactId>springboot2_webflux</artifactId>
  <version>0.0.1-SNAPSHOT</version>
  <packaging>jar</packaging>

  <name>springboot2_webflux</name>
  <description>springboot2_webflux</description>

  <parent>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-parent</artifactId>
    <version>2.0.3.RELEASE</version>
    <relativePath/> <!-- lookup parent from repository -->
  </parent>

  <properties>
    <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
    <project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding>
    <java.version>1.8</java.version>
  </properties>

  <dependencies>
    <dependency>
      <groupId>org.springframework.boot</groupId>
      <artifactId>spring-boot-starter</artifactId>
    </dependency>

    <dependency>
      <groupId>org.springframework.boot</groupId>
      <artifactId>spring-boot-starter-test</artifactId>
      <scope>test</scope>
    </dependency>

    <dependency>
      <groupId>org.springframework.boot</groupId>
      <artifactId>spring-boot-starter-webflux</artifactId>
    </dependency>
  </dependencies>

  <build>
    <plugins>
      <plugin>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-maven-plugin</artifactId>
      </plugin>
    </plugins>
  </build>

</project>

首先试试引入WebFlux依赖之后,SpringMvc方式是否还能使用,新建一个HelloController,完整代码如下,执行后发现,是可以正常执行访问的,这其实就是我们所说的注解式编程。

package com.dalaoyang.controller;

import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;

/**
 * @author dalaoyang
 * @project springboot_learn
 * @package com.dalaoyang.controller
 * @email yangyang@dalaoyang.cn
 * @date 2018/7/30
 */
@RestController
public class HelloController {

  @GetMapping("hello")
  public String Hello(){
    return "Hello this is SpringWebFlux";
  }
}

结果如图:

接下来使用函数式编程,首先查阅一下官方文档,如图:

我们需要创建一个HandlerFunction返回值为Mono,新建一个HiHandler,里面写一个方法Hi,完整代码如下:

package com.dalaoyang.handler;
import org.springframework.http.MediaType;
import org.springframework.stereotype.Component;
import org.springframework.web.reactive.function.BodyInserters;
import org.springframework.web.reactive.function.server.ServerRequest;
import org.springframework.web.reactive.function.server.ServerResponse;
import reactor.core.publisher.Mono;

/**
 * @author dalaoyang
 * @project springboot_learn
 * @package com.dalaoyang.handler
 * @email yangyang@dalaoyang.cn
 * @date 2018/7/30
 */
@Component
public class HiHandler {

  public Mono<ServerResponse> Hi(ServerRequest request) {
    return ServerResponse.ok().contentType(MediaType.APPLICATION_JSON)
        .body(BodyInserters.fromObject("Hi , this is SpringWebFlux"));
  }
}

其中ServerResponse是相应的封装对象,下面是它的源码,其中包含了响应状态,响应头等等,代码如下:

package org.springframework.web.reactive.function.server;

import java.net.URI;
import java.time.ZonedDateTime;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.function.BiFunction;
import java.util.function.Consumer;
import org.reactivestreams.Publisher;
import org.springframework.core.ParameterizedTypeReference;
import org.springframework.http.CacheControl;
import org.springframework.http.HttpHeaders;
import org.springframework.http.HttpMethod;
import org.springframework.http.HttpStatus;
import org.springframework.http.MediaType;
import org.springframework.http.ResponseCookie;
import org.springframework.http.codec.HttpMessageWriter;
import org.springframework.http.server.reactive.ServerHttpResponse;
import org.springframework.util.MultiValueMap;
import org.springframework.web.reactive.function.BodyInserter;
import org.springframework.web.reactive.result.view.ViewResolver;
import org.springframework.web.server.ServerWebExchange;
import reactor.core.publisher.Mono;

public interface ServerResponse {
  HttpStatus statusCode();

  HttpHeaders headers();

  MultiValueMap<String, ResponseCookie> cookies();

  Mono<Void> writeTo(ServerWebExchange var1, ServerResponse.Context var2);

  static ServerResponse.BodyBuilder from(ServerResponse other) {
    return new DefaultServerResponseBuilder(other);
  }

  static ServerResponse.BodyBuilder status(HttpStatus status) {
    return new DefaultServerResponseBuilder(status);
  }

  static ServerResponse.BodyBuilder status(int status) {
    return new DefaultServerResponseBuilder(status);
  }

  static ServerResponse.BodyBuilder ok() {
    return status(HttpStatus.OK);
  }

  static ServerResponse.BodyBuilder created(URI location) {
    ServerResponse.BodyBuilder builder = status(HttpStatus.CREATED);
    return (ServerResponse.BodyBuilder)builder.location(location);
  }

  static ServerResponse.BodyBuilder accepted() {
    return status(HttpStatus.ACCEPTED);
  }

  static ServerResponse.HeadersBuilder<?> noContent() {
    return status(HttpStatus.NO_CONTENT);
  }

  static ServerResponse.BodyBuilder seeOther(URI location) {
    ServerResponse.BodyBuilder builder = status(HttpStatus.SEE_OTHER);
    return (ServerResponse.BodyBuilder)builder.location(location);
  }

  static ServerResponse.BodyBuilder temporaryRedirect(URI location) {
    ServerResponse.BodyBuilder builder = status(HttpStatus.TEMPORARY_REDIRECT);
    return (ServerResponse.BodyBuilder)builder.location(location);
  }

  static ServerResponse.BodyBuilder permanentRedirect(URI location) {
    ServerResponse.BodyBuilder builder = status(HttpStatus.PERMANENT_REDIRECT);
    return (ServerResponse.BodyBuilder)builder.location(location);
  }

  static ServerResponse.BodyBuilder badRequest() {
    return status(HttpStatus.BAD_REQUEST);
  }

  static ServerResponse.HeadersBuilder<?> notFound() {
    return status(HttpStatus.NOT_FOUND);
  }

  static ServerResponse.BodyBuilder unprocessableEntity() {
    return status(HttpStatus.UNPROCESSABLE_ENTITY);
  }

  public interface Context {
    List<HttpMessageWriter<?>> messageWriters();

    List<ViewResolver> viewResolvers();
  }

  public interface BodyBuilder extends ServerResponse.HeadersBuilder<ServerResponse.BodyBuilder> {
    ServerResponse.BodyBuilder contentLength(long var1);

    ServerResponse.BodyBuilder contentType(MediaType var1);

    ServerResponse.BodyBuilder hint(String var1, Object var2);

    <T, P extends Publisher<T>> Mono<ServerResponse> body(P var1, Class<T> var2);

    <T, P extends Publisher<T>> Mono<ServerResponse> body(P var1, ParameterizedTypeReference<T> var2);

    Mono<ServerResponse> syncBody(Object var1);

    Mono<ServerResponse> body(BodyInserter<?, ? super ServerHttpResponse> var1);

    Mono<ServerResponse> render(String var1, Object... var2);

    Mono<ServerResponse> render(String var1, Map<String, ?> var2);
  }

  public interface HeadersBuilder<B extends ServerResponse.HeadersBuilder<B>> {
    B header(String var1, String... var2);

    B headers(Consumer<HttpHeaders> var1);

    B cookie(ResponseCookie var1);

    B cookies(Consumer<MultiValueMap<String, ResponseCookie>> var1);

    B allow(HttpMethod... var1);

    B allow(Set<HttpMethod> var1);

    B eTag(String var1);

    B lastModified(ZonedDateTime var1);

    B location(URI var1);

    B cacheControl(CacheControl var1);

    B varyBy(String... var1);

    Mono<ServerResponse> build();

    Mono<ServerResponse> build(Publisher<Void> var1);

    Mono<ServerResponse> build(BiFunction<ServerWebExchange, ServerResponse.Context, Mono<Void>> var1);
  }
}

在回过头了看上面官方文档的图片,还需要配置一个路由来类似@RequestMapping的功能,通过RouterFunctions.route(RequestPredicate, HandlerFunction)提供了一个路由器函数默认实现,新建一个HiRouter,代码如下:

package com.dalaoyang.router;

import com.dalaoyang.handler.HiHandler;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.http.MediaType;
import org.springframework.web.reactive.function.server.RequestPredicates;
import org.springframework.web.reactive.function.server.RouterFunction;
import org.springframework.web.reactive.function.server.RouterFunctions;
import org.springframework.web.reactive.function.server.ServerResponse;
/**
 * @author dalaoyang
 * @project springboot_learn
 * @package com.dalaoyang.router
 * @email yangyang@dalaoyang.cn
 * @date 2018/7/30
 */
@Configuration
public class HiRouter {
  @Bean
  public RouterFunction<ServerResponse> routeCity(HiHandler hiHandler) {
    return RouterFunctions
        .route(RequestPredicates.GET("/hi")
                .and(RequestPredicates.accept(MediaType.APPLICATION_JSON)),
            hiHandler::Hi);
  }
}

启动项目,通过控制台可以看到,两种方式的映射都被打印出来了,如图所示:

在浏览器访问,http://localhost:8080/hi,结果如图所示:

源码下载 :大老杨码云

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • 使用 Spring Boot 2.0 + WebFlux 实现 RESTful API功能

    概述 什么是 Spring WebFlux, 它是一种异步的, 非阻塞的, 支持背压(Back pressure)机制的Web 开发框架. 要深入了解 Spring WebFlux, 首先要了知道 Reactive Stream . 另一种编程姿势, 和命令式编程相对的姿势. WebFlux 支持两种编程风(姿)格(势) 使用 @Controller 这种基于注解的姿势, 与Sring MVC的姿势相同 基于Java 8 Lambda的函数式编程风格 注意: 上面只是两种编程的姿势, 和"普通话

  • SpringBoot2使用WebFlux函数式编程的方法

    本文只是简单使用SpringBoot2使用WebFlux的函数式编程简单使用,后续会继续写关于Webflux相关的文章. 最近一直在研究WebFlux,后续会陆续出一些相关的文章. 首先看一下Srping官网上的一张图,对比一下SpringMvc和Spring WebFlux,如图: 在查看一下WebFlux的官方文档:https://docs.spring.io/spring/docs/current/spring-framework-reference/web-reactive.html,W

  • 使用Python的toolz库开始函数式编程的方法

    在这个由两部分组成的系列文章的第二部分中,我们将继续探索如何将函数式编程方法中的好想法引入到 Python中,以实现两全其美. 在上一篇文章中,我们介绍了不可变数据结构 . 这些数据结构使得我们可以编写"纯"函数,或者说是没有副作用的函数,仅仅接受一些参数并返回结果,同时保持良好的性能. 在这篇文章中,我们使用 toolz 库来构建. 这个库具有操作此类函数的函数,并且它们在纯函数中表现得特别好. 在函数式编程世界中,它们通常被称为"高阶函数",因为它们将函数作为参

  • Java函数式编程(八):字符串及方法引用

    第三章 字符串,比较器和过滤器 JDK引入的一些方法对写出函数式风格的代码很有帮助.JDK库里的一些的类和接口我们已经用得非常熟悉了,比如说String,为了摆脱以前习惯的那种老的风格,我们得主动寻找机会来使用这些新的方法.同样,当我们需要用到只有一个方法的匿名内部类时,我们现在可以用lambda表达式来替换它了,不用再像原来那样写的那么繁琐了. 本章我们会使用lambda表达式和方法引用来遍历字符串,实现Comparator接口,查看目录中的文件,监视文件及目录的变更.上一章中介绍的一些方法还

  • 《JavaScript函数式编程》读后感

    本文章记录本人在学习 函数式 中理解到的一些东西,加深记忆和并且整理记录下来,方便之后的复习. 在近期看到了<JavaScript函数式编程>这本书预售的时候就定了下来.主要目的是个人目前还是不理解什么是函数式编程.在自己学习的过程中一直听到身边的人说面向过程编程和面向对象编程,而函数式就非常少.为了自己不要落后于其他同学的脚步,故想以写笔记的方式去分享和记录自己阅读中所汲取的知识. js 和函数式编程 书中用了一句简单的话来回答了什么是函数式编程: 函数式编程通过使用函数来将值转换为抽象单元

  • Python函数式编程

    主要内容 1.函数基本语法及特性 2.参数与局部变 3.返回值 4.递归 5.名函数 6.函数式编程介绍 7.阶函数 8.内置函数 函数基本语法及特性 定义 数学函数定义:一般的,在一个变化过程中,如果有两个变量x和y,并且对于x的每一 个确定的值,y都有唯一确定的值与其对应,那么我们就把x称为自变量,把y称为因变 量,y是x的函数.自变量x的取值范围叫做这个函数的定义域. 但编程中的「函数」概念,与数学中的函数是有很 同的 函数是逻辑结构化和过程化的一种编程方法 函数的优点 减少重复代码 使程

  • JavaScript的函数式编程基础指南

    引言 JavaScript是一种强大的,却被误解的编程语言.一些人喜欢说它是一个面向对象的编程语言,或者它是一个函数式编程语言.另外一些人喜欢说,它不是一个面向对象的编程语言,或者它不是一个函数式编程语言.还有人认为它兼具面向对象语言和函数式语言的特点,或者,认为它既不是面向对象的也不是函数式的,好吧,让我们先搁置那些争论. 让我们假设我们共有这样的一个使命:在JavaScript语言所允许的范围内,尽可能多的使用函数式编程的原则来编写程序. 首先,我们需要清理下脑子里那些关于函数式编程的错误观

  • 详解用函数式编程对JavaScript进行断舍离

    我和JavaScript 从1997年网景的Navigator 3浏览器开始就开始使用JavaScript.当时,JavaScript还只能做一些很简单的事情.我记得最酷的就是用JavaScript实现mouseover特性,在那个时候已经算得上是高科技了!当鼠标移过去之后,文本内容就神奇的改变了.因为当时都是pre-DHTML,你根本不需要隐藏或则显示DOM元素. 关于DHTML DHTML是Dynamic HTML的简称,就是动态的html(标准通用标记语言下的一个应用),是相对传统的静态的

  • Swift的函数式编程详解

    Swift 相比原先的 Objective-C 最重要的优点之一,就是对函数式编程提供了更好的支持. Swift 提供了更多的语法和一些新特性来增强函数式编程的能力,本文就在这方面进行一些讨论. Swift 概览 对编程语言有了一些经验的程序员,尤其是那些对多种不同类型的编程语言都有经验的开发者, 在学习新的语言的时候更加得心应手.原因在于编程语言本身也是有各种范式的, 把握住这些特点就可以比较容易的上手了. 在入手一门新的语言的时候,一般关注的内容有: 1.原生数据结构 2.运算符 3.分支控

  • JavaScript 函数式编程实践(来自IBM)第1/3页

    函数式编程简介 说到函数式编程,人们的第一印象往往是其学院派,晦涩难懂,大概只有那些蓬头散发,不修边幅,甚至有些神经质的大学教授们才会用的编程方式.这可能在历史上的某个阶段的确如此,但是近来函数式编程已经在实际应用中发挥着巨大作用了,而更有越来越多的语言不断的加入诸如 闭包,匿名函数等的支持,从某种程度上来讲,函数式编程正在逐步"同化"命令式编程. 函数式编程思想的源头可以追溯到 20 世纪 30 年代,数学家阿隆左 . 丘奇在进行一项关于问题的可计算性的研究,也就是后来的 lambd

  • javascript函数式编程程序员的工具集

    如果你仔细看了到目前为止出现过的示例代码,你会发现这里面的一些方法不太熟悉. 它们是map().filter()和reduce()函数,它们对任何语言的函数式编程都至关重要. 它们可以让你不必使用循环和语句,写出更简洁的代码. map().filter()和reduce()函数组成了函数式程序员工具集的核心部分,这个工具集包括一系列纯的. 高阶的函数,它们是函数式方法的主力.实际上,它们是纯函数和高阶函数的典型,它们以一个函数为输入, 返回一个输出结果,并且不产生副作用. 然而它们是浏览器中EC

随机推荐