首先,您可以将拦截器绑定到特定路径,这样就解决了一半的问题。
第二个问题是获取对已请求的 Person 资源的引用。您可以在这里查看使用 ThreadLocal:
public class PersonContext{
private static ThreadLocal<Person> context = new ThreadLocal<Person>();
public static Person getPerson(Person person){
return context.get();
}
public static void setPerson(Person person){
context.set(person);
}
public static void cleanUp(){
context.remove();
}
}
标准的 JPA 实体监听器可用于绑定对加载的 Person 的引用:
public class PersonListener{
@PostLoad
public void setContext(Person person){
PersonContext.set(person);
}
}
Entity监听器可以注册到Entity上:
@Entity
@EntityListeners(PersonListener.class)
public class Person(){
}
在拦截器中,您可以获得对绑定到当前线程上下文的 Person 的引用,并相应地设置标头:
public class PersonInterceptor extends HandlerInterceptorAdapter {
@Override
public void postHandle(HttpServletRequest request, HttpServletResponse response,
Object handler, ModelAndView modelAndView) throws Exception {
if(request.getMethod().equals("GET")){
Person person = PersonContext.getPerson();
response.addHeader("age", person.getAge());
}
PersonContext.cleanUp();
}
}
最后用 Spring Data Rest 注册你的拦截器:
@Bean
public MappedInterceptor myMappedInterceptor() {
return new MappedInterceptor(new String[]{"person/**"}, new PersonInterceptor());
}
问题:
- 加载后侦听器将在 Person 的每次加载时执行,即使在其他路径上也是如此,因此清理工作需要更好。在所有路径上注册第二个拦截器来处理清理??
另一种方法是在拦截器中重新查询数据库。所以上面可以用拦截器代替:
public class PersonInterceptor extends HandlerInterceptorAdapter {
@Autowired
private PersonRepository repository;
@Override
public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler,
ModelAndView modelAndView) throws Exception {
super.postHandle(request, response, handler, modelAndView);
if(request.getMethod().equals("GET")){
Long personID = // Extract id by examining request.getServletPath();
Person person = repository.findOne(personID);
response.addHeader("age", person.getAge());
}
}
}
这会产生额外查询的开销,但可以通过配置 OpenEntityManagerInViewFilter 来避免这种情况,这意味着它是从一级缓存中获取的。