1

我们正在构建某种自动自我描述 REST 服务(实时文档生成)。为此,我们有一个控制器方法,它查找所有控制器 bean 并获取请求映射信息以将它们显示在一个友好的 html 页面中。

为此,我们使用 MetadataReader(通过 CachingMetadataReaderFactory 创建)来获取类的元数据。当我们获取公共方法的 MethodMetaData 时,我们发现 RequestMapping 注释以及生产和值参数,但方法字段始终是一个空数组,尽管我们在源代码中配置了它并且映射工作。所以信息应该在某个地方。这相当令人费解!:-)

编辑:方法字段为空=> RequestMapping 注释有一个方法字段,它是一个 RequestMethod 对象数组。如果您尝试从 MethodMetaData 实例中读取它,则它是一个空数组。示例: metadata.getAnnotationAttributes(RequestMapping.class.getName()).get("method")

我尝试在spring框架源码中查找原因,但目前还没有找到原因……

有任何想法吗?

仅供参考:我们使用的是 Spring 3.1

4

2 回答 2

1

出于自己的好奇心,我创建了一个小示例项目,并对 Spring 提供的 MetadataReader 进行了一些操作。对于演示,我创建了一个非常简单的控制器,如下所示:

@Controller
public class SomeAnnotatedController {

   @RequestMapping(method = {RequestMethod.GET}, value = "/someUrl")
   public void someMethod() {
       // do something later
   }

}

我无法使用 Spring MetadataReader 从注释中提取正确的信息。

@Test
public void shouldReturnMethodArrayWithSpringMetadataReader() throws Exception {
    MetadataReader metadataReader = new CachingMetadataReaderFactory().getMetadataReader(SomeAnnotatedController.class.getName());
    Set<MethodMetadata> annotatedMethods = metadataReader.getAnnotationMetadata().getAnnotatedMethods(RequestMapping.class.getName());
    assertEquals(1, annotatedMethods.size());
    MethodMetadata methodMetadata = annotatedMethods.iterator().next();
    assertEquals("someMethod", methodMetadata.getMethodName());
    Map<String, Object> annotationAttributes = methodMetadata.getAnnotationAttributes(RequestMapping.class.getName());
    assertTrue(annotationAttributes.containsKey("method"));
    RequestMethod[] methodAttribute = (RequestMethod[]) annotationAttributes.get("method");
    assertEquals(1, methodAttribute.length);
}

在最后一行运行此测试失败,并告诉您这是一个空数组...

java.lang.AssertionError: 
Expected :1
Actual   :0

用原生 Java 做同样的事情感觉更容易一些,并返回正确的信息。

@Test
public void shouldReturnMethodArrayWithPlainJava() throws Exception {
    Method method = SomeAnnotatedController.class.getDeclaredMethod("someMethod");
    RequestMapping annotation = method.getAnnotation(RequestMapping.class);
    assertEquals(1, annotation.method().length);
    assertEquals(RequestMethod.GET, annotation.method()[0]);
}

所以我很遗憾地告诉您,我没有找到问题的解决方案,但也许示例项目或基于纯 java 的文档替代方案可能会有所帮助。

于 2012-04-24T20:30:27.047 回答
1

这不是您所要求的直接答案,而是一种非常好的方式来对 REST API 进行自我记录。在他的 github 位置使用 Rossen Stoyanchev 描述的端点文档控制器:https ://github.com/rstoyanchev/spring-mvc-31-demo.git

总而言之,您的控制器看起来像这样:

@Controller
public class EndpointDocController {

    private final RequestMappingHandlerMapping handlerMapping;

    @Autowired
    public EndpointDocController(RequestMappingHandlerMapping handlerMapping) {
        this.handlerMapping = handlerMapping;
    }

    @RequestMapping(value="/endpointdoc", method=RequestMethod.GET)
    public void show(Model model) {
        model.addAttribute("handlerMethods", this.handlerMapping.getHandlerMethods());
    }

}

并且您的 jsp 会以这种方式引用方法、生产、消费、方法签名的属性,假设 hm 是一个处理程序方法:

Patterns:${hm.key.patternsCondition.patterns}
Method: ${hm.key.methodsCondition.methods}
Method signature: ${hm.value}
Consumes: ${hm.key.consumesCondition.expressions}
Produces: ${hm.key.producesCondition.expressions}
于 2012-04-25T01:39:31.363 回答