java - 如何解析 URL 并使用 Spring MVC 'reflectively' 运行方法?

标签 java spring spring-mvc

我有一个 Spring Boot 应用程序,它以通常的方式使用 Spring MVC,带有一堆 @RequestMapping 方法、Freemarker 定义等。这一切都与 WebMvcConfigurerAdapter 类捆绑在一起。

我想提供一项服务,用户提交有效 URL 列表,web 应用程序将计算出将调用哪个 Controller ,传入参数,并返回每个 URL 的组合结果——所有这些都合而为一请求。

这将使用户不​​必进行数百次 HTTP 调用,但仍允许他们在需要时发出一次性请求。理想情况下,我只需要注入(inject)一个自动配置的 Spring bean,这样我就不必重复 Spring 内部进行的 URL 解析、调整和处理,并且 Controller 的其他 Controller 列表永远不会与 < em>真正的 Controller 列表。

我希望写这样的东西(简化为只处理一个 URL,这毫无意义但更容易理解):

@Autowired BeanThatSolvesAllMyProblems allMappings;

@PostMapping(path = "/encode", consumes = MediaType.TEXT_PLAIN_VALUE)
@ResponseBody
public String encode(@RequestBody String inputPath) {
    if (allMappings.hasMappingForPath(inputPath)) {
        return allMappings.getMapping(inputPath).execute();
    } else {
        return "URL didn't match, sorry";
    }
}

相反,我不得不定义 Spring bean 我不知道它们做了什么,并且一直在重复一些 Spring 打算为我做的事情,我担心这不会起作用 与用户自己调用电话时的情况相同:

// these two are @Beans, with just their default constructor called.
@Autowired RequestMappingHandlerMapping handlers;
@Autowired RequestMappingHandlerAdapter adapter;

@PostMapping(path = "/encode", consumes = MediaType.TEXT_PLAIN_VALUE)
@ResponseBody
public String encode(@RequestBody String inputText) {
    final HttpServletRequest mockRequest = new MockHttpServletRequest(null, inputText);
    final StringBuilder result = new StringBuilder();

    this.handlers.getHandlerMethods().forEach((requestMappingInfo, handlerMethod) -> {
        if (requestMappingInfo.getPatternsCondition().getMatchingCondition(mockRequest) != null) {
            try {
                final MockHttpServletResponse mockResponse = new MockHttpServletResponse();
                result.append("Result: ").append(adapter.handle(mockRequest, mockResponse, handlerMethod));
                result.append(", ").append(mockResponse.getContentAsString());
                result.append("\n");
            } catch (Exception e) {
                logger.error(e.getMessage(), e);
            }
        }
    });

    return result.toString();
}

我以为我在这条路上做得很好,但它因 Missing URI template variable 错误而失败,而且我不仅不知道如何将请求参数放入(另一件事哪个 Spring 可以自行处理),但我什至不确定这是执行此操作的正确方法。那么我如何从 webapp 本身内部“反射性地”模拟 Spring MVC 请求?

最佳答案

JSON API 规范通过允许每个请求发送多个操作来解决这个问题。甚至有一个相当成熟的实现支持这个功能,称为 Elide .但我想这可能不能完全满足您的要求。

无论如何,这是您可以做的。

你必须考虑到 DispatcherServlet持有 handlerMappings用于检测适当的列表 request处理程序和 handlerAdaptors .两个列表的选择策略都是可配置的(参见 DispatcherServlet#initHandlerMappings#initHandlerAdapters )。

您应该想出一种您更喜欢检索 handlerMappings 列表的方法/initHandlerAdapters并与DispatcherServlet保持同步.

之后您可以实现自己的 HandlerMapping/HandlerAdaptor (或在您的示例中呈现一个 Controller 方法)将处理 request/encode小路。

顺便说一句,HandlerMapping正如 javadoc 所说的那样

Interface to be implemented by objects that define a mapping between requests and handler objects

或者简单地说如果我们取DefaultAnnotationHandlerMapping这将映射我们的 HttpServletRequests@Controller@RequestMapping 注释的方法.有这个映射 HandlerAdapter准备传入请求以使用 Controller 方法,f.ex。提取请求 params , body并使用它们调用 Controller 的方法。

有了这个,你可以提取URL来自主要request , 创建 stub 列表 HttpRequests保存进一步处理所需的信息并通过它们循环调用:

HandlerExecutionChain getHandler(HttpServletRequest request) throws Exception {
    for (HandlerMapping hm : this.handlerMappings) {
        if (logger.isTraceEnabled()) {
            logger.trace(
                    "Testing handler map [" + hm + "] in DispatcherServlet with name '" + getServletName() + "'");
        }
        HandlerExecutionChain handler = hm.getHandler(request);
        if (handler != null) {
            return handler;
        }
    }
    return null;
}

有一个handlerMapping你叫

    HandlerAdapter getHandlerAdapter(Object handler) throws ServletException {
    for (HandlerAdapter ha : this.handlerAdapters) {
        if (logger.isTraceEnabled()) {
            logger.trace("Testing handler adapter [" + ha + "]");
        }
        if (ha.supports(handler)) {
            return ha;
        }
    }

然后你终于可以打电话了

ha.handle(processedRequest, response, mappedHandler.getHandler());

这反过来会使用参数执行 Controller 的方法。

但是考虑到所有这些,我不建议采用这种方法,而是考虑使用 JSON API规范或任何其他。

关于java - 如何解析 URL 并使用 Spring MVC 'reflectively' 运行方法?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39212895/

相关文章:

java - 使用java将OCTL字符转换为ISO-8859-15(html)

java - apache cxf 中使用 queryParam 的方法选择

java - 何时使用 Bean Validation 与自定义验证框架?

java - 如何使用 JDBC 引用/转义标识符,例如列名?

JDK中XML相关api

Grails 中的 Spring Oauth2 提供程序 - 依赖项

Spring Boot 使用配置文件启用/禁用嵌入式 tomcat

mongodb - 如何将嵌套对象引用到其他集合 Mongodb

java - org.h2.jdbc.JdbcSQL异常 : Column "Salman" not found;

java - Spring Reactive MVC 与 @EnableAsync