• Java之自定义MVC(2)


    目录

    一、中央控制器动态加载存储子控制器

    二、参数传递封装优化

    三、返回值页面跳转优化

    四、框架配置文件可变


    一、中央控制器动态加载存储子控制器

    1. <?xml version="1.0" encoding="UTF-8"?>
    2. <config>
    3. <action path="/book" type="com.cxy.servlet.BookAction">
    4. <forward name="failed" path="/login.jsp" redirect="false" />
    5. <forward name="success" path="/main.jsp" redirect="true" />
    6. </action>
    7. </config>

     DispatcherServlet.java

    1. package com.cxy.framework;
    2. import java.io.IOException;
    3. import java.util.HashMap;
    4. import java.util.Map;
    5. import javax.servlet.ServletException;
    6. import javax.servlet.annotation.WebServlet;
    7. import javax.servlet.http.HttpServlet;
    8. import javax.servlet.http.HttpServletRequest;
    9. import javax.servlet.http.HttpServletResponse;
    10. import com.cxy.model.ActionModel;
    11. import com.cxy.model.ConfigModel;
    12. import com.cxy.model.ConfigModelFactory;
    13. import com.cxy.servlet.BookAction;
    14. /**
    15. * 中央控制器
    16. * 主要职能:接收浏览器请求,找到对应的处理人
    17. * 不处理任何业务逻辑,只接收请求
    18. * @author zjjt
    19. */
    20. @WebServlet("*.action")
    21. public class DispatcherServlet extends HttpServlet{
    22. private Map<String, Action> actions = new HashMap<String, Action>();
    23. /**
    24. * 通过建模我们知道,最终configModel对象包含config.xml中的所有子控制器信息
    25. * 同时为了解决中央控制器能够保存子控制器的信息,那么我们只需要引入configModel对象即可
    26. */
    27. private ConfigModel configmodel;
    28. //程序启动时值加载一次
    29. @Override
    30. public void init() throws ServletException {
    31. // actions.put("/book", new BookAction());
    32. // actions.put("/order", new BookAction());
    33. //-----------------------------优化后----------------------------------
    34. try {
    35. configmodel = ConfigModelFactory.build();
    36. } catch (Exception e) {
    37. // TODO Auto-generated catch block
    38. e.printStackTrace();
    39. }
    40. }
    41. @Override
    42. protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
    43. doPost(req, resp);
    44. }
    45. @Override
    46. protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
    47. //http://location:8080/mvc/book.action?me....
    48. String uri = req.getRequestURI();//路径
    49. // 要拿到/book,就是最后一个/到最会一个.的位置
    50. uri = uri.substring(uri.lastIndexOf("/"),uri.lastIndexOf("."));
    51. //Action action = actions.get(uri);
    52. //-------------------------------优化后--------------------------------------------------
    53. //相比于上一种从map集合获取子控制器的信息,当前需要获取config.xml中的全程径名,然后反射实例化
    54. ActionModel actionModel = configmodel.pop(uri);
    55. if(actionModel==null) {
    56. throw new RuntimeException("action 配置错误");
    57. }
    58. String type = actionModel.getType();
    59. try {
    60. //type是action子控制器的全路径名
    61. Action action =(Action) Class.forName(type).newInstance();
    62. action.execute(req, resp);
    63. } catch (Exception e) {
    64. // TODO: handle exception
    65. e.printStackTrace();
    66. }
    67. }
    68. }

    二、参数传递封装优化

    创建所需要传递值的对象Book.java

    1. package com.cxy.entity;
    2. public class Book {
    3. private int bid;
    4. private String bname;
    5. private float price;
    6. public int getBid() {
    7. return bid;
    8. }
    9. public void setBid(int bid) {
    10. this.bid = bid;
    11. }
    12. public String getBname() {
    13. return bname;
    14. }
    15. public void setBname(String bname) {
    16. this.bname = bname;
    17. }
    18. public float getPrice() {
    19. return price;
    20. }
    21. public void setPrice(float price) {
    22. this.price = price;
    23. }
    24. public Book() {
    25. // TODO Auto-generated constructor stub
    26. }
    27. public Book(int bid, String bname, float price) {
    28. super();
    29. this.bid = bid;
    30. this.bname = bname;
    31. this.price = price;
    32. }
    33. @Override
    34. public String toString() {
    35. return "Book [bid=" + bid + ", bname=" + bname + ", price=" + price + "]";
    36. }
    37. }

    创建模型驱动接口ModelDriven.java

    1. package com.cxy.framework;
    2. /**模型驱动接口,接收前台jsp传递的参数,并且封装带实体类中
    3. * @author zjjt
    4. *
    5. * @param <T>
    6. */
    7. public interface ModelDriven<T> {
    8. //拿到将要封装的类实例 Model
    9. T getModel();
    10. }

     实现Moldedriven接口,并实现接口的方法BookAction.java

    1. package com.cxy.servlet;
    2. import javax.servlet.http.HttpServletRequest;
    3. import javax.servlet.http.HttpServletResponse;
    4. import com.cxy.entity.Book;
    5. import com.cxy.framework.ActionSupport;
    6. import com.cxy.framework.ModelDriven;
    7. public class BookAction extends ActionSupport implements ModelDriven<Book>{
    8. //这里面还没有值,这样一来我们需要实现Moldedriven接口,并实现接口的方法
    9. private Book book =new Book();
    10. private void lod(HttpServletRequest req, HttpServletResponse resp) {
    11. System.out.println("在同一个servlet中调用 回显");
    12. }
    13. private void select(HttpServletRequest req, HttpServletResponse resp) {
    14. System.out.println("在同一个servlet中调用 增加");
    15. }
    16. private void update(HttpServletRequest req, HttpServletResponse resp) {
    17. System.out.println("在同一个servlet中调用 修改");
    18. }
    19. private void del(HttpServletRequest req, HttpServletResponse resp) {
    20. System.out.println("在同一个servlet中调用 删除");
    21. }
    22. private void add(HttpServletRequest req, HttpServletResponse resp) {
    23. //System.out.println("在同一个servlet中调用 查看 ");
    24. //String bid = req.getParameter("bid");
    25. // String bname = req.getParameter("bname");
    26. // String price = req.getParameter("price");
    27. // Book b = new Book();
    28. // b.setBid(Integer.parseInt(bid));
    29. // b.setBname(bname);
    30. // b.setPrice(Float.parseFloat(price));
    31. //存在问题:当值过多的之后容易出错
    32. // System.out.println("在同一个servlet中调用 add 方法 ");
    33. // return "failed";
    34. }
    35. @Override
    36. public Book getModel() {
    37. // TODO Auto-generated method stub
    38. return book;
    39. }
    40. }

     向DispatcherServlet.java里将前端的参数封装进实体类

    1. @Override
    2. protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
    3. //http://location:8080/mvc/book.action?me....
    4. String uri = req.getRequestURI();//路径
    5. // 要拿到/book,就是最后一个/到最会一个.的位置
    6. uri = uri.substring(uri.lastIndexOf("/"),
    7. uri.lastIndexOf("."));
    8. //Action action = actions.get(uri);
    9. //相比于上一种从map集合获取子控制器的信息,当前需要获取config.xml中的全程径名,然后反射实例化
    10. ActionModel actionModel = configmodel.pop(uri);
    11. if(actionModel==null) {
    12. throw new RuntimeException("action 配置错误");
    13. }
    14. String type = actionModel.getType();
    15. try {
    16. //type是action子控制器的全路径名
    17. Action action =(Action) Class.forName(type).newInstance();
    18. if(action instanceof ModelDriven) {
    19. //多态的使用将bookaction转换为ModelDriven
    20. ModelDriven md = (ModelDriven) action;
    21. //model指的是bookaction中的book实例
    22. Object model = md.getModel();
    23. //要给model中的属性赋值,要接受前端jsp传递过来的参数
    24. // PropertyUtils.getProperty(bean, name);//某一对象获取到值
    25. //将前端所有参数值封装进实体类
    26. BeanUtils.populate(model, req.getParameterMap());
    27. }
    28. action.execute(req, resp);
    29. } catch (Exception e) {
    30. // TODO: handle exception
    31. e.printStackTrace();
    32. }
    33. }
    34. }

    三、返回值页面跳转优化

    配置文件config.xml

    1. <?xml version="1.0" encoding="UTF-8"?>
    2. <config>
    3. <action path="/book" type="com.cxy.servlet.BookAction">
    4. <forward name="success" path="/demo2.jsp" redirect="false" /><!-- 转发界面 -->
    5. <forward name="failed" path="/demo3.jsp" redirect="true" /><!-- 重定向界面 -->
    6. </action>
    7. </config>

    参考xml建模里的configModel ,ConfigModelFactory,ForwardModel

    1. package com.cxy.framework;
    2. import javax.servlet.http.HttpServletRequest;
    3. import javax.servlet.http.HttpServletResponse;
    4. /**
    5. *
    6. *子控制器
    7. *对应请求的处理人
    8. * @author zjjt
    9. *
    10. */
    11. public interface Action {
    12. //封装继承多态
    13. String execute(HttpServletRequest req, HttpServletResponse resp);
    14. }
    1. package com.cxy.framework;
    2. import java.lang.reflect.Method;
    3. import javax.servlet.http.HttpServletRequest;
    4. import javax.servlet.http.HttpServletResponse;
    5. public class ActionSupport implements Action{
    6. @Override
    7. public String execute(HttpServletRequest req, HttpServletResponse resp) {
    8. String methodName = req.getParameter("methodName");
    9. //methodName可能是add/del/...
    10. //前台传递什么方法,就调用当前类对应的方法
    11. try {
    12. Method m = this.getClass()//相当于BookServlet
    13. .getDeclaredMethod(methodName, HttpServletRequest.class,HttpServletResponse.class);
    14. m.setAccessible(true);
    15. //调用当前类实例的methodName方法 传什么就掉什么方法
    16. return (String)m.invoke(this, req,resp);
    17. } catch (Exception e) {
    18. // TODO Auto-generated catch block
    19. e.printStackTrace();
    20. }
    21. return null;
    22. }
    23. }

    向DispatcherServlet.java进行配置

    1. @Override
    2. protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
    3. //http://location:8080/mvc/book.action?me....
    4. String uri = req.getRequestURI();//路径
    5. // 要拿到/book,就是最后一个/到最会一个.的位置
    6. uri = uri.substring(uri.lastIndexOf("/"),
    7. uri.lastIndexOf("."));
    8. //Action action = actions.get(uri);
    9. //相比于上一种从map集合获取子控制器的信息,当前需要获取config.xml中的全程径名,然后反射实例化
    10. ActionModel actionModel = configmodel.pop(uri);
    11. if(actionModel==null) {
    12. throw new RuntimeException("action 配置错误");
    13. }
    14. String type = actionModel.getType();
    15. try {
    16. //type是action子控制器的全路径名
    17. Action action =(Action) Class.forName(type).newInstance();
    18. if(action instanceof ModelDriven) {
    19. //多态的使用将bookaction转换为ModelDriven
    20. ModelDriven md = (ModelDriven) action;
    21. //model指的是bookaction中的book实例
    22. Object model = md.getModel();
    23. //要给model中的属性赋值,要接受前端jsp传递过来的参数
    24. // PropertyUtils.getProperty(bean, name);//某一对象获取到值
    25. //将前端所有参数值封装进实体类
    26. BeanUtils.populate(model, req.getParameterMap());
    27. }
    28. //action.execute(req, resp);
    29. String result = action.execute(req, resp);
    30. //result里面拿的是配置文件里forwode里的值
    31. ForwardModel forwardModel = actionModel.pop(result);
    32. if(forwardModel == null) {
    33. throw new RuntimeException("forwode 配置错误");
    34. }
    35. String path = forwardModel.getPath();
    36. //拿到是否需要转发的配置
    37. boolean redirect = forwardModel.isRedirect();
    38. if(redirect) {
    39. resp.sendRedirect(req.getServletContext().getContextPath() + path);
    40. }
    41. else {
    42. req.getRequestDispatcher(path).forward(req, resp);
    43. }
    44. } catch (Exception e) {
    45. // TODO: handle exception
    46. e.printStackTrace();
    47. }
    48. }
    49. }

    四、框架配置文件可变

    XML配置

    1. <?xml version="1.0" encoding="UTF-8"?>
    2. <web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://xmlns.jcp.org/xml/ns/javaee" xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_3_1.xsd" id="WebApp_ID" version="3.1">
    3. <display-name>MVC</display-name>
    4. <servlet>
    5. <servlet-name>mvc</servlet-name>
    6. <servlet-class>com.cxy.framework.DispatcherServlet</servlet-class>
    7. <init-param>
    8. <param-name>configLocation</param-name>
    9. <param-value>/cxy.xml</param-value>
    10. </init-param>
    11. </servlet>
    12. <servlet-mapping>
    13. <servlet-name>mvc</servlet-name>
    14. <url-pattern>*.action</url-pattern>
    15. </servlet-mapping>
    16. </web-app>

    将获取的xml文件在中央控制器里DispatcherServlet.java进行判断

    1. package com.cxy.framework;
    2. import java.io.IOException;
    3. import java.util.HashMap;
    4. import java.util.Map;
    5. import javax.servlet.ServletException;
    6. import javax.servlet.annotation.WebServlet;
    7. import javax.servlet.http.HttpServlet;
    8. import javax.servlet.http.HttpServletRequest;
    9. import javax.servlet.http.HttpServletResponse;
    10. import org.apache.commons.beanutils.BeanUtils;
    11. import com.cxy.model.ActionModel;
    12. import com.cxy.model.ConfigModel;
    13. import com.cxy.model.ConfigModelFactory;
    14. import com.cxy.model.ForwardModel;
    15. import com.cxy.servlet.BookAction;
    16. /**
    17. * 中央控制器
    18. * 主要职能:接收浏览器请求,找到对应的处理人
    19. * 不处理任何业务逻辑,只接收请求
    20. * @author zjjt
    21. */
    22. @WebServlet("*.action")
    23. public class DispatcherServlet extends HttpServlet{
    24. private Map<String, Action> actions = new HashMap<String, Action>();
    25. private ConfigModel configmodel;
    26. //程序启动时值加载一次
    27. @Override
    28. public void init() throws ServletException {
    29. // actions.put("/book", new BookAction());
    30. // actions.put("/order", new BookAction());
    31. try {
    32. //配置
    33. String configLocation = this.getInitParameter("configLocation");
    34. //没有改变配置文件
    35. if(configLocation==null||"".equals(configLocation)) {
    36. configmodel = ConfigModelFactory.build();
    37. }
    38. else {
    39. configmodel = ConfigModelFactory.build(configLocation);
    40. }
    41. } catch (Exception e) {
    42. // TODO Auto-generated catch block
    43. e.printStackTrace();
    44. }
    45. }
    46. }

    总结:
    ①  在不改动中央控制器任何代码的情况下,依旧可以动态加载存储控制器(子控制器加在了xml配置文件里面去,这样就可以不改动中央控制器的情况下也可以加载);
    ②   confog模型对象替代了Map集合,因为模型对象里面包含了所有的配置文件的信息,好处在于,只需要修改配置文件里的信息即可,可以不改变中央控制器的代码;
    ③  减少了代码的封装步骤。

     

  • 相关阅读:
    【SpringBoot】实现引入登录时的验证码功能
    这些阻碍程序员升职加薪的行为,你中招了几个?
    [ZJOI2009]假期的宿舍
    Kafka 入门知识,看这一篇就够了(上)
    模拟实现一个Linux中的简单版shell
    【Git】(六)子模块跟随主仓库切换分支
    PyTorch学习:使用pytorch进行数据预处理
    NNZ 2591 是环状甘氨酸脯氨酸 (cGP) 小肽的合成类似物 | MedChemExpress (MCE)
    APA泊车名词解释
    vue-cli脚手架初始化项目+(node + webpack + 淘宝镜像)
  • 原文地址:https://blog.csdn.net/m0_62528678/article/details/125495674