SpringWebFlux默认配置或者自定义配置,并且实现统一结果集处理(提供测试案例)

 以下示例仅供参考,提供实现思路,并不能直接使用,可能会出现意向不到BUG,因为我只是单存的测试某个效果

package luck.spring.boot.webflux;

import lombok.extern.slf4j.Slf4j;
import org.reactivestreams.Publisher;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.core.ResolvableType;
import org.springframework.http.MediaType;
import org.springframework.http.codec.EncoderHttpMessageWriter;
import org.springframework.http.codec.HttpMessageReader;
import org.springframework.http.codec.HttpMessageWriter;
import org.springframework.http.codec.ServerCodecConfigurer;
import org.springframework.http.codec.json.Jackson2JsonEncoder;
import org.springframework.http.server.reactive.ServerHttpRequest;
import org.springframework.http.server.reactive.ServerHttpResponse;
import org.springframework.util.MimeType;
import org.springframework.web.cors.CorsConfiguration;
import org.springframework.web.cors.reactive.CorsWebFilter;
import org.springframework.web.cors.reactive.UrlBasedCorsConfigurationSource;
import org.springframework.web.reactive.config.CorsRegistry;
import org.springframework.web.reactive.config.DelegatingWebFluxConfiguration;
import org.springframework.web.reactive.config.WebFluxConfigurer;
import reactor.core.publisher.Flux;
import reactor.core.publisher.Mono;

import java.util.List;
import java.util.Map;

// 如果需要高度自定WebFlux的配置,可以继承DelegatingWebFluxConfiguration进行扩展
// 因为@EnableWebFlux导入的是默认的DelegatingWebFluxConfiguration
// @EnableWebFlux
@Configuration
@Slf4j
public class WebFluxConfig extends DelegatingWebFluxConfiguration {
    // 设置请求跨域的过滤器,这个跨域处理事通过过滤器处理的
    // 和addCorsMappings
    // @Bean
    public CorsWebFilter corsFilter() {
        CorsConfiguration config = new CorsConfiguration();
        config.setAllowCredentials(true);
        config.addAllowedOrigin("*");
        config.addAllowedHeader("*");
        config.addAllowedMethod("*");
        UrlBasedCorsConfigurationSource source = new UrlBasedCorsConfigurationSource();
        source.registerCorsConfiguration("/**", config);
        return new CorsWebFilter(source);
    }

    @Bean
    public WebFluxConfigurer webFluxConfigurer() {
        return new WebFluxConfigurer() {
            @Override
            public void configureHttpMessageCodecs(ServerCodecConfigurer configurer) {
                // 开启编码解码器的详细日志信息
                configurer.defaultCodecs().enableLoggingRequestDetails(true);
                List<HttpMessageWriter<?>> writers = configurer.getWriters();
                List<HttpMessageReader<?>> readers = configurer.getReaders();
                // 自定义的编码解码器,就是对请求响应体的处理器
                EncoderHttpMessageWriter<Object> writer = new EncoderHttpMessageWriter<>(new Jackson2JsonEncoder() {
                    @Override
                    public boolean canEncode(ResolvableType elementType, MimeType mimeType) {
                        return CharSequence.class.isAssignableFrom(elementType.resolve()) || super.canEncode(elementType, mimeType);
                    }
                }) {
                    @Override
                    public Mono<Void> write(Publisher<?> inputStream, ResolvableType actualType, ResolvableType elementType, MediaType mediaType, ServerHttpRequest request, ServerHttpResponse response, Map<String, Object> hints) {
                        if (CharSequence.class.isAssignableFrom(elementType.resolve())) {
                            mediaType = MediaType.APPLICATION_JSON;
                        }
                        if (inputStream instanceof Flux flux) {
                            inputStream = flux.flatMap(value -> {
                                if (!(value instanceof Result)) {
                                    return Flux.just(Result.OK(value));
                                }
                                return Flux.just(value);
                            });
                        }
                        if (inputStream instanceof Mono mono) {
                            inputStream = mono.flatMap(value -> {
                                if (!(value instanceof Result)) {
                                    return Mono.just(Result.OK(value));
                                }
                                return Mono.just(value);
                            });
                        }
                        return super.write(inputStream, actualType, elementType, mediaType, request, response, hints);
                    }
                };
                log.error("writer is {}", writers.hashCode());
                configurer.customCodecs().register(writer);
            }

            // 处理跨域,这种方式是在获取handler的时候处理的
            @Override
            public void addCorsMappings(CorsRegistry registry) {
                // registry.addMapping("/**")
                //         .allowedOrigins("*")
                //         .allowedMethods("*")
                //         .allowedHeaders("*")
                //         .exposedHeaders("*")
                //         .maxAge(3600);
            }
        };
    }
}

  • 3
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Spring Webflux中,可以通过实现`org.springframework.web.reactive.result.view.View`接口来创建自定义的返回值处理器,进而实现自动封装结果。 具体步骤如下: 1. 创建一个类实现`View`接口,并实现其中的方法。在`render`方法中,可以将控制器方法返回的数据封装成需要的格式,比如JSON格式,并将其写入响应体中。 ```java import org.springframework.core.ResolvableType; import org.springframework.http.MediaType; import org.springframework.http.MediaTypeUtils; import org.springframework.http.ReactiveHttpOutputMessage; import org.springframework.http.server.reactive.ServerHttpRequest; import org.springframework.http.server.reactive.ServerHttpResponse; import org.springframework.http.server.reactive.ServerHttpResponseDecorator; import org.springframework.web.reactive.result.view.AbstractView; import org.springframework.web.reactive.result.view.View; import reactor.core.publisher.Mono; import java.util.HashMap; import java.util.Map; public class ResultView implements View { @Override public String getContentType() { return MediaType.APPLICATION_JSON_VALUE; } @Override public Mono<Void> render(Map<String, ?> model, MediaType contentType, ServerHttpRequest request, ServerHttpResponse response) { Object result = model.get("result"); if (result == null) { result = new HashMap<>(); } response.getHeaders().setContentType(MediaType.APPLICATION_JSON); return response.writeWith(Mono.just(response.bufferFactory().wrap(result.toString().getBytes()))); } } ``` 在这个例子中,我们创建了一个名为`ResultView`的类来实现`View`接口。在`render`方法中,我们从`model`中取出控制器方法返回的数据,将其封装成JSON格式,并写入响应体中。 2. 在Spring配置文件中配置该返回值处理器。可以通过`WebFluxConfigurer`来配置,如下所示: ```java import org.springframework.context.annotation.Configuration; import org.springframework.web.reactive.config.ViewResolverRegistry; import org.springframework.web.reactive.config.WebFluxConfigurer; @Configuration public class WebConfig implements WebFluxConfigurer { @Override public void configureViewResolvers(ViewResolverRegistry registry) { registry.viewResolver(new ResultViewResolver()); } } ``` 在这个例子中,我们创建了一个名为`ResultViewResolver`的类,将其注册到Spring Webflux中。 3. 在Controller中使用该返回值处理器。可以通过在控制器方法上添加`@ModelAttribute`注解来将返回值封装成一个模型对象,然后将该模型对象传递给视图解析器。 ```java import org.springframework.web.bind.annotation.GetMapping; import org.springframework.web.bind.annotation.ModelAttribute; import org.springframework.web.bind.annotation.RestController; import reactor.core.publisher.Mono; @RestController public class MyController { @GetMapping("/my/path") public Mono<Map<String, Object>> handleRequest() { Map<String, Object> result = new HashMap<>(); // 处理请求逻辑 return Mono.just(result); } @ModelAttribute("result") public Mono<Map<String, Object>> getResult() { return handleRequest(); } } ``` 在这个例子中,我们在控制器方法上添加了`@ModelAttribute`注解,将返回值封装成一个模型对象,并将其传递给视图解析器。在`getResult`方法中,我们调用了`handleRequest`方法来获取控制器方法的返回值。该返回值将在视图解析器中被封装成JSON格式,并写入响应体中。 通过这种方式,我们可以实现自动封装结果,减少代码重复,提高开发效率。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值