如果您使用 JAX-RS 2.0,则可以使用 @BeanParam,它允许您将任意 @XxxParam 注释属性和 @Context 对象注入任意 bean 类。例如
public class Bean {
@QueryParam("blah")
String blah;
}
@GET
public Response get(@BeanParam Bean bean) {}
如果你想要不可变,你甚至可以注入到构造函数中。例如
public static class Pagination {
private final List<String> sort;
private final Optional<String> from;
public Pagination(@QueryParam("sort") List<String> sort,
@QueryParam("from") Optional<String> from) {
this.sort = sort;
this.from = from;
}
public List<String> getSort() { return sort; }
public Optional<String> getFrom() { return from; }
}
如果您注意到Optional 正在被注入。通常这是不可能的,但我为它创建了一个ParamConverter。您可以在this answer 中阅读有关它的更多信息。它基本上允许我们注入任意对象,从参数的 String 值创建。
@Provider
public static class OptionalParamProvider implements ParamConverterProvider {
@Override
public <T> ParamConverter<T> getConverter(Class<T> rawType,
Type genericType,
Annotation[] annotations) {
if (Optional.class != rawType) {
return null;
}
return (ParamConverter<T>)new ParamConverter<Optional>() {
@Override
public Optional fromString(String value) {
return Optional.ofNullable(value);
}
@Override
public String toString(Optional value) {
return value.toString();
}
};
}
}
OptionalParamProvider 的好处是它允许您在需要注入@FormParam、@QueryParam、@PathParm 和所有其他@XxxParams 的任何地方使用Optional(multitpart 除外) .
我不知道您使用的是什么 JAX-RS 实现,但以上内容应该适用于所有实现。下面是一个 Jersey 测试用例,使用 Jersey Test Framework。您可以像运行任何其他 JUnit 测试一样运行该类。
import java.lang.annotation.Annotation;
import java.lang.reflect.Type;
import java.util.List;
import java.util.Optional;
import javax.ws.rs.BeanParam;
import javax.ws.rs.GET;
import javax.ws.rs.Path;
import javax.ws.rs.QueryParam;
import javax.ws.rs.core.Response;
import javax.ws.rs.ext.ParamConverter;
import javax.ws.rs.ext.ParamConverterProvider;
import javax.ws.rs.ext.Provider;
import org.glassfish.jersey.server.ResourceConfig;
import org.glassfish.jersey.test.JerseyTest;
import static org.hamcrest.CoreMatchers.containsString;
import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertThat;
import org.junit.Test;
public class BeanParamTest extends JerseyTest {
@Provider
public static class OptionalParamProvider implements ParamConverterProvider {
@Override
public <T> ParamConverter<T> getConverter(Class<T> rawType,
Type genericType,
Annotation[] annotations) {
if (Optional.class != rawType) {
return null;
}
return (ParamConverter<T>)new ParamConverter<Optional>() {
@Override
public Optional fromString(String value) {
return Optional.ofNullable(value);
}
@Override
public String toString(Optional value) {
return value.toString();
}
};
}
}
public static class Pagination {
private final List<String> sort;
private final Optional<String> from;
public Pagination(@QueryParam("sort") List<String> sort,
@QueryParam("from") Optional<String> from) {
this.sort = sort;
this.from = from;
}
public List<String> getSort() { return sort; }
public Optional<String> getFrom() { return from; }
}
@Path("bean")
public static class PaginationResource {
@GET
public String get(@BeanParam Pagination pagination) {
StringBuilder sb = new StringBuilder();
sb.append(pagination.getSort().toString());
if (pagination.getFrom().isPresent()) {
sb.append(pagination.getFrom().get());
}
return sb.toString();
}
}
@Override
public ResourceConfig configure() {
return new ResourceConfig(PaginationResource.class)
.register(OptionalParamProvider.class);
}
@Test
public void should_return_all_sort_and_from() {
Response response = target("bean")
.queryParam("sort", "foo")
.queryParam("sort", "bar")
.queryParam("from", "baz")
.request().get();
assertEquals(200, response.getStatus());
String message = response.readEntity(String.class);
assertThat(message, containsString("foo"));
assertThat(message, containsString("bar"));
assertThat(message, containsString("baz"));
System.out.println(message);
response.close();
}
}
这是运行测试所需的唯一 Maven 依赖项
<dependency>
<groupId>org.glassfish.jersey.test-framework.providers</groupId>
<artifactId>jersey-test-framework-provider-grizzly2</artifactId>
<version>2.19</version>
<scope>test</scope>
</dependency>