gpt4 book ai didi

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

转载 作者:塔克拉玛干 更新时间:2023-11-03 04:33:42 26 4
gpt4 key购买 nike

我有一个 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/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com