由于一个Servlet只能接收一个地址的http请求,如果系统复杂度提高,就会有很多Servlet类。例如,对销售系统来说,可能会有OederInsertServlet, OrderUpdateServlet, OrderDeleteSerlvet, OrderQueryServlet等多个OrderServlet来处理订单这一种业务。看着就会很杂。如下图。同时如果请求前或请求后有一些处理的话,对应的方法就需要写很多次,维护难度也会提升


可以只用一个Servlet接收Order这一种业务所有的接口,通过在接口添加类似operate的标志参数对不同的操作进行区分。

protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
String operate = req.getParameter("operate");
if (null == operate){
doRequireOperate(req, resp);
}
SUtils.logInfo("调用前操作");
System.out.println("获取到的操作名为:" + operate);
switch (operate){
case "insert":{
System.out.println("do insert");
doInsertOperate(req, resp);
break;
}
case "update":{
System.out.println("do update");
doUpdateOperate(req, resp);
break;
}
case "delete":{
System.out.println("do delete");
doDeleteOperate(req, resp);
break;
}
case "query":{
System.out.println("do query");
doQueryOperate(req, resp);
break;
}
default:{
System.out.println("error");
}
SUtils.logInfo("调用后操作");
}
}
本地访问 http://localhost:8080/mvc/ver1?operate=insert
服务器日志输出如下

目前,对业务的操作是通过switch-case进行匹配处理的,如果操作数量增多则需要多增加case语句,代码就会显得臃肿。可以考虑通过反射调用具体的执行方法
下面展示的是核心代码
String operate = req.getParameter("operate");
System.out.println("获取到的操作名为:" + operate);
// 通过反射获取对应的方法
// 假定方法名均为do+${operate}+operate,驼峰,如operate为insert时,要调用的方法就是doInsertOperate
// Method[] methods = this.getClass().getMethods();
char[] chars = operate.toCharArray();
chars[0] -= 32;
String methodName = "do" + String.valueOf(chars) + "Operate";
// Optional<Method> targetMethod = Arrays.stream(methods).filter(m -> m.getName().equals(methodName)).findFirst();
try {
Method taragetMethod = this.getClass().getDeclaredMethod(methodName, HttpServletRequest.class, HttpServletResponse.class);
taragetMethod.setAccessible(true);
taragetMethod.invoke(this, req, resp);
} catch (NoSuchMethodException | IllegalAccessException | InvocationTargetException e) {
throw new RuntimeException(e);
}
当业务增多时,需要在每个业务里面单独写反射调用方法较为复杂,可以也通过反射来调用业务模块,再执行对应的方法,同时可以引入xml配置文件,来匹配对应的业务模块

@WebServlet("/mvc/ver3/*")
public class Ver3Servlet extends HttpServlet {
private Map<String, Object> beanControllerMap = new HashMap<>();
// 初始化,读取xml文件配置controller
@Override
public void init() throws ServletException {
// super.init();
System.out.println("aaa");
try (InputStream beanConfigs = this.getClass().getClassLoader().getResourceAsStream("MvcVer3Config.xml")) {
DocumentBuilderFactory documentBuilderFactory = DocumentBuilderFactory.newInstance();
DocumentBuilder documentBuilder = documentBuilderFactory.newDocumentBuilder();
Document beanDocument = documentBuilder.parse(beanConfigs);
Element documentElement = beanDocument.getDocumentElement();
NodeList nodeList = beanDocument.getDocumentElement().getElementsByTagName("bean");
for (int i = 0; i < nodeList.getLength(); i++) {
if (Node.ELEMENT_NODE == nodeList.item(i).getNodeType()){
Element item = (Element) nodeList.item(i);
String beanName = item.getAttribute("id");
String className = item.getAttribute("class");
Class<?> beanClass = Class.forName(className);
Object bean = beanClass.getDeclaredConstructors()[0].newInstance();
beanControllerMap.put(beanName, bean);
}
}
// documentElement.get
} catch (IOException | ParserConfigurationException | SAXException | ClassNotFoundException |
InvocationTargetException | InstantiationException | IllegalAccessException e) {
throw new RuntimeException(e);
}
System.out.println("aaa");
}
// 读取路径值,获取对应的controller对象,反射调用对应方法
@Override
protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
String servletPath = req.getPathInfo();
System.out.println(servletPath);
String controllerName = servletPath.split("/")[1];
String operateName = servletPath.split("/")[2];
operateName = "do" + getFirstUpper(operateName) + "Operate";
Object targetController = beanControllerMap.get(getFirstUpper(controllerName));
Method targetMethod;
try {
targetMethod = targetController.getClass().getDeclaredMethod(operateName, HttpServletRequest.class, HttpServletResponse.class);
} catch (NoSuchMethodException e) {
ErrorResponse(req, resp, e.getMessage());
e.printStackTrace();
return;
}
try {
targetMethod.invoke(targetController, req, resp);
} catch (IllegalAccessException | InvocationTargetException e) {
ErrorResponse(req, resp, e.getMessage());
e.printStackTrace();
return;
}
System.out.println("调用结束");
}
private void ErrorResponse(HttpServletRequest req, HttpServletResponse resp, String errorString) throws IOException {
req.setCharacterEncoding("utf-8");
resp.setCharacterEncoding("utf-8");
resp.setContentType("text/html;charset=utf-8");
PrintWriter out = resp.getWriter();
out.println("<!DOCTYPE html PUBLIC \"-//W3C//DTD HTML 4.01 Transitional//EN\" \"http://www.w3.org/TR/html4/loose.dtd\">\n");
out.println("<HTML lang=\"ch\">");
out.println(" <HEAD><TITLE>delete Servlet</TITLE></HEAD>");
out.println(" <BODY>");
out.print("所访问的模块不存在<br>");
out.print(errorString);
out.println(" </BODY>");
out.println("</HTML>");
out.flush();
out.close();
System.out.println("所访问的模块不存在");
}
private String getFirstUpper(String s){
char[] chars = s.toCharArray();
chars[0] -= 32;
return String.valueOf(chars);
}
}
解决多个业务配置问题,通过读取配置文件初始化Servlet,在收到请求时,取到对应的Servlet进行处理
<?xml version="1.0" encoding="utf-8" ?>
<beans>
<bean id="A" class="com.javawebdemo.mvc.controller.AController"/>
<bean id="B" class="com.javawebdemo.mvc.controller.BController"/>
<bean id="C" class="com.javawebdemo.mvc.controller.CController"/>
<bean id="D" class="com.javawebdemo.mvc.controller.DController"/>
</beans>
启动项目,访问对应路径
http://localhost:8080/mvc/ver3/a/query

页面响应

调用成功
Rails2.3可以选择随时使用RouteSet#add_configuration_file添加更多路由。是否可以在Rails3项目中做同样的事情? 最佳答案 在config/application.rb中:config.paths.config.routes在Rails3.2(也可能是Rails3.1)中,使用:config.paths["config/routes"] 关于ruby-on-rails-Rails3中的多个路由文件,我们在StackOverflow上找到一个类似的问题
我有多个ActiveRecord子类Item的实例数组,我需要根据最早的事件循环打印。在这种情况下,我需要打印付款和维护日期,如下所示:ItemAmaintenancerequiredin5daysItemBpaymentrequiredin6daysItemApaymentrequiredin7daysItemBmaintenancerequiredin8days我目前有两个查询,用于查找maintenance和payment项目(非排他性查询),并输出如下内容:paymentrequiredin...maintenancerequiredin...有什么方法可以改善上述(丑陋的)代
我需要从一个View访问多个模型。以前,我的links_controller仅用于提供以不同方式排序的链接资源。现在我想包括一个部分(我假设)显示按分数排序的顶级用户(@users=User.all.sort_by(&:score))我知道我可以将此代码插入每个链接操作并从View访问它,但这似乎不是“ruby方式”,我将需要在不久的将来访问更多模型。这可能会变得很脏,是否有针对这种情况的任何技术?注意事项:我认为我的应用程序正朝着单一格式和动态页面内容的方向发展,本质上是一个典型的网络应用程序。我知道before_filter但考虑到我希望应用程序进入的方向,这似乎很麻烦。最终从任何
我有一个具有一些属性的模型:attr1、attr2和attr3。我需要在不执行回调和验证的情况下更新此属性。我找到了update_column方法,但我想同时更新三个属性。我需要这样的东西:update_columns({attr1:val1,attr2:val2,attr3:val3})代替update_column(attr1,val1)update_column(attr2,val2)update_column(attr3,val3) 最佳答案 您可以使用update_columns(attr1:val1,attr2:val2
我正在尝试修改当前依赖于定义为activeresource的gem:s.add_dependency"activeresource","~>3.0"为了让gem与Rails4一起工作,我需要扩展依赖关系以与activeresource的版本3或4一起工作。我不想简单地添加以下内容,因为它可能会在以后引起问题:s.add_dependency"activeresource",">=3.0"有没有办法指定可接受版本的列表?~>3.0还是~>4.0? 最佳答案 根据thedocumentation,如果你想要3到4之间的所有版本,你可以这
我正在尝试按0-9和a-z的顺序创建数字和字母列表。我有一组值value_array=['0','1','2','3','4','5','6','7','8','9','a','b','光盘','e','f','g','h','i','j','k','l','m','n','o','p','q','r','s','','u','v','w','x','y','z']和一个组合列表的数组,按顺序,这些数字可以产生x个字符,比方说三个list_array=[]和一个当前字母和数字组合的数组(在将它插入列表数组之前我会把它变成一个字符串,]current_combo['0','0','0']
是否有可能:before_filter:authenticate_user!||:authenticate_admin! 最佳答案 before_filter:do_authenticationdefdo_authenticationauthenticate_user!||authenticate_admin!end 关于ruby-on-rails-before_filter运行多个方法,我们在StackOverflow上找到一个类似的问题: https://
我正在使用Rails3.1并在一个论坛上工作。我有一个名为Topic的模型,每个模型都有许多Post。当用户创建新主题时,他们也应该创建第一个Post。但是,我不确定如何以相同的形式执行此操作。这是我的代码:classTopic:destroyaccepts_nested_attributes_for:postsvalidates_presence_of:titleendclassPost...但这似乎不起作用。有什么想法吗?谢谢! 最佳答案 @Pablo的回答似乎有你需要的一切。但更具体地说...首先改变你View中的这一行对此#
我收到格式为的回复#我需要将其转换为哈希值(针对活跃商家)。目前我正在遍历变量并执行此操作:response.instance_variables.eachdo|r|my_hash.merge!(r.to_s.delete("@").intern=>response.instance_eval(r.to_s.delete("@")))end这有效,它将生成{:first="charlie",:last=>"kelly"},但它似乎有点hacky和不稳定。有更好的方法吗?编辑:我刚刚意识到我可以使用instance_variable_get作为该等式的第二部分,但这仍然是主要问题。
我的问题的一个例子是体育游戏。一场体育比赛有两支球队,一支主队和一支客队。我的事件记录模型如下:classTeam"Team"has_one:away_team,:class_name=>"Team"end我希望能够通过游戏访问一个团队,例如:Game.find(1).home_team但我收到一个单元化常量错误:Game::team。谁能告诉我我做错了什么?谢谢, 最佳答案 如果Gamehas_one:team那么Rails假设您的teams表有一个game_id列。不过,您想要的是games表有一个team_id列,在这种情况下