草庐IT

JavaWeb 多个Servlet整合优化

mrystar 2023-04-17 原文

JavaWeb 多个Servlet整合优化

由于一个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


页面响应

调用成功

有关JavaWeb 多个Servlet整合优化的更多相关文章

  1. ruby-on-rails - Rails 3 中的多个路由文件 - 2

    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上找到一个类似的问题

  2. ruby-on-rails - 在 Ruby 中循环遍历多个数组 - 2

    我有多个ActiveRecord子类Item的实例数组,我需要根据最早的事件循环打印。在这种情况下,我需要打印付款和维护日期,如下所示:ItemAmaintenancerequiredin5daysItemBpaymentrequiredin6daysItemApaymentrequiredin7daysItemBmaintenancerequiredin8days我目前有两个查询,用于查找maintenance和payment项目(非排他性查询),并输出如下内容:paymentrequiredin...maintenancerequiredin...有什么方法可以改善上述(丑陋的)代

  3. ruby-on-rails - Rails - 一个 View 中的多个模型 - 2

    我需要从一个View访问多个模型。以前,我的links_controller仅用于提供以不同方式排序的链接资源。现在我想包括一个部分(我假设)显示按分数排序的顶级用户(@users=User.all.sort_by(&:score))我知道我可以将此代码插入每个链接操作并从View访问它,但这似乎不是“ruby方式”,我将需要在不久的将来访问更多模型。这可能会变得很脏,是否有针对这种情况的任何技术?注意事项:我认为我的应用程序正朝着单一格式和动态页面内容的方向发展,本质上是一个典型的网络应用程序。我知道before_filter但考虑到我希望应用程序进入的方向,这似乎很麻烦。最终从任何

  4. ruby - 多个属性的 update_column 方法 - 2

    我有一个具有一些属性的模型: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

  5. ruby-on-rails - 在 ruby​​ .gemspec 文件中,如何指定依赖项的多个版本? - 2

    我正在尝试修改当前依赖于定义为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之间的所有版本,你可以这

  6. ruby - 使用多个数组创建计数 - 2

    我正在尝试按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']

  7. ruby-on-rails - before_filter 运行多个方法 - 2

    是否有可能:before_filter:authenticate_user!||:authenticate_admin! 最佳答案 before_filter:do_authenticationdefdo_authenticationauthenticate_user!||authenticate_admin!end 关于ruby-on-rails-before_filter运行多个方法,我们在StackOverflow上找到一个类似的问题: https://

  8. ruby-on-rails - Rails 3.1 中具有相同形式的多个模型? - 2

    我正在使用Rails3.1并在一个论坛上工作。我有一个名为Topic的模型,每个模型都有许多Post。当用户创建新主题时,他们也应该创建第一个Post。但是,我不确定如何以相同的形式执行此操作。这是我的代码:classTopic:destroyaccepts_nested_attributes_for:postsvalidates_presence_of:titleendclassPost...但这似乎不起作用。有什么想法吗?谢谢! 最佳答案 @Pablo的回答似乎有你需要的一切。但更具体地说...首先改变你View中的这一行对此#

  9. ruby-on-rails - 使用 ruby​​ 将多个实例变量转换为散列的更好方法? - 2

    我收到格式为的回复#我需要将其转换为哈希值(针对活跃商家)。目前我正在遍历变量并执行此操作: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作为该等式的第二部分,但这仍然是主要问题。

  10. ruby - Rails 关联 - 同一个类的多个 has_one 关系 - 2

    我的问题的一个例子是体育游戏。一场体育比赛有两支球队,一支主队和一支客队。我的事件记录模型如下: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列,在这种情况下

随机推荐