我遇到了这个问题,最近我读到了关于REST arquitecture的内容,它很有道理,所以我想实现一个RESTful Web应用程序。
现在,我正在关注Front Controller pattern,这意味着所有的URL映射都转到controller.java
servlet,我按特定的URL映射,而不是使用/*
通配符,
控制器实现了四种HTTP方法POST
,GET
,PUT
,DELETE
,每种方法调用控制器service
方法,我根据{确定{1}}和HttpServletRequest
要执行的操作。
Controller.java
pathInfo
我在通过URI
@Override
protected void service(HttpServletRequest req, HttpServletResponse resp)
throws ServletException, IOException {
IAction action;
View view;
try {
action = ActionFactory.produceAction(req);
view = action.execute(req, resp);
switch (view.getDispatchMethod()) {
case REDIRECT:
resp.sendRedirect(resp.encodeURL(view.getResource()));
break;
case FORWARD:
req.getRequestDispatcher(view.getResource()).forward(req, resp);
break;
case INCLUDE:
req.getRequestDispatcher(view.getResource()).include(req,resp);
break;
default:
}
} catch (ActionFailedException uae) {
req.setAttribute("ActionName", "Action");
req.setAttribute("FailCause", uae.getMessage());
req.getRequestDispatcher(VIEW_FAIL.getResource()).forward(req, resp);
}
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp)
throws ServletException, IOException {
this.service(req, resp);
}
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
this.service(req, resp);
}
@Override
protected void doPut(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
this.service(req, resp);
}
@Override
protected void doDelete(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
this.service(req, resp);
}
加载特定订单时遇到了一个特定问题,它被映射到控制器servlet,执行了操作,并且我加载了操作返回{{{ 1}}类
/orders/*
然后根据返回视图的View.java
调度请求。
现在,这里是循环被触发的地方,我使用以下URL,
//ommited accessors and mutators for brevety.
public class View {
public enum DispatchMethod {
INCLUDE, FORWARD, REDIRECT
}
private DispatchMethod dispatchMethod;
private String resource;
public View(DispatchMethod dispatchMethod, String resource) {
this.dispatchMethod = dispatchMethod;
this.resource = resource;
}
}
getDispatchMethod()
被映射到myapp/orders/78965
执行适当的操作并且找到正确的顺序/orders/*
返回的视图为controller.java
,问题在于,使用三种可用的调度方法pathInfo()
,会在URL上重新触发请求,依此类推,依此类推,我从未到达{ {1}}呈现数据。
那么,你如何避免循环,因为我想保留显示订单号的URI我使用forward方法,我也想用servlet来做,我听说过UrlRewriteFilter可能在将来,但是现在,如果使用“Plain Vanilla”,我将如何使用Front Controller模式,是否有必要在new View(View.DispatchMethod.FORWARD,"order_details.jsp")
URI中添加一个额外的servlet?
真正感谢任何帮助或见解。
编辑1:
粘贴控制器的源代码,这是一个非常基本的源代码,我怀疑REDIRECT,FORWARD and INCLUDE
方法调用servlet的所有覆盖order_details.jsp
的方式是触发循环并且它可以通过分裂来解决它们。
答案 0 :(得分:2)
使用JAX-RS实现(如RESTEasy或Jersey)在Java中实现RESTful HTTP接口要容易得多。
使用前端控制器将请求分派给正确的资源是一种很好的方法,这正是这些JAX-RS框架所采用的方法。我担心你可能会通过编写一个定制的URL解析和调度机制来重新发明这个轮子,这可以在现成的时候进行。
JAX-RS是一种公开资源的轻量级方法。通过使用几个简单的注释,您可以暴露REST接口,而无需任何管道。例如:
public class Order {
@GET
@Path("/orders/{orderId}")
@Produces("text/html")
public void getOrder(@Context HttpServletResponse response,
@Context HttpServletRequest request,
@PathParam("orderId") String orderId) throws ServletException, IOException {
// ... create view and add to request here
request.getRequestDispatcher("orders.jsp").forward(request, response);
}
}
您可以看到将此类附加到URL路径(使用@Path
注释)是多么简单,以及使用@PathParam
从URL解析值的难易程度。由于您可以获得现成的所有管道/调度/解析,因此您可以专注于应用程序中特定于您的域的位(例如订单包含的内容)。