对于一个MVC框架来说,最重要的就是C了,特别是前端控制器。前端控制器首先要根据URL请求,来分发该请求应该由哪个controller中哪个方法来处理;然后controller处理完后,还要根据其返回值,最终定位要应该返回哪张视图给客户端,如图:
为些我们实现一个MVC框架主要就是实现这个前端控制器。我写了一个很简单的MVC框架,我将代码分享给大家。
由于工程是由maven搭建的,则pom.xml如下:
(其中jetty也配置好)
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd"> <modelVersion>4.0.0</modelVersion> <groupId>com.cloud.mvc</groupId> <artifactId>mvc-framework</artifactId> <packaging>war</packaging> <version>0.0.1-SNAPSHOT</version> <name>mvc-framework Maven Webapp</name> <url>http://maven.apache.org</url> <dependencies> <dependency> <groupId>junit</groupId> <artifactId>junit</artifactId> <version>3.8.1</version> <scope>test</scope> </dependency> <dependency> <groupId>javax.servlet</groupId> <artifactId>servlet-api</artifactId> <version>2.5</version> </dependency> </dependencies> <build> <finalName>mvc-framework</finalName> <plugins> <plugin> <groupId>org.mortbay.jetty</groupId> <artifactId>maven-jetty-plugin</artifactId> <version>6.1.10</version> <configuration> <contextPath>/test</contextPath> <connectors> <connector implementation="org.mortbay.jetty.nio.SelectChannelConnector"> <port>8080</port> </connector> </connectors> <scanIntervalSeconds>3</scanIntervalSeconds> </configuration> </plugin> </plugins> </build> </project>
如果要使用我这个框架,则工程目录下必需要有一个mvc.xml文件,而且文件名必需是mvc.xml。该文件的作用和struts2框架中的struts.xml的作用类似。
<?xml version="1.0" encoding="UTF-8"?> <mvc> <action name="user/select" class="com.cloud.mvc.controller.UserController" method="select"> <result name="success">/WEB-INF/view/index.jsp</result> <result name="faild">/index.jsp</result> </action> </mvc>
两个用于封装配置信息的类。
ActionConfig.java
package com.cloud.mvc.config; import java.util.HashMap; import java.util.Map; public class ActionConfig { private String name; private String clazz; private String method; private Map<String, ResultConfig> result; public String getName() { return name; } public void setName(String name) { this.name = name; } public String getClazz() { return clazz; } public void setClazz(String clazz) { this.clazz = clazz; } public String getMethod() { return method; } public void setMethod(String method) { this.method = method; } public Map<String, ResultConfig> getResult() { return result; } public void setResult(Map<String, ResultConfig> result) { this.result = result; } public ActionConfig() { } public ActionConfig(String name, String clazz, String method) { this.name = name; this.clazz = clazz; this.method = method; this.result = new HashMap<String, ResultConfig>(); } }
ResultConfig.java
package com.cloud.mvc.config; public class ResultConfig { private String name; private String content; public String getName() { return name; } public void setName(String name) { this.name = name; } public String getContent() { return content; } public void setContent(String content) { this.content = content; } public ResultConfig() { } public ResultConfig(String name, String content) { this.name = name; this.content = content; } }
用到的工具类。
XMLUtil.java
package com.cloud.mvc.util; import java.io.InputStream; import java.util.HashMap; import java.util.Map; import javax.xml.parsers.DocumentBuilder; import javax.xml.parsers.DocumentBuilderFactory; import org.w3c.dom.Document; import org.w3c.dom.Element; import org.w3c.dom.NamedNodeMap; import org.w3c.dom.Node; import org.w3c.dom.NodeList; import com.cloud.mvc.config.ActionConfig; import com.cloud.mvc.config.ResultConfig; public class XMLUtil { public static Map<String, ActionConfig> parseConfig() throws Exception { Map<String, ActionConfig> actionConfigs = new HashMap<String, ActionConfig>(); DocumentBuilderFactory dbf = DocumentBuilderFactory.newInstance(); DocumentBuilder db = dbf.newDocumentBuilder(); InputStream in = XMLUtil.class.getClassLoader().getResourceAsStream("mvc.xml"); Document document = db.parse(in); Element root = document.getDocumentElement(); NodeList actionNodes = root.getChildNodes(); for (int i = 0; i < actionNodes.getLength(); i++) { Node action = actionNodes.item(i); if (null != action && action.getNodeType() == Node.ELEMENT_NODE) { NamedNodeMap nodeMaps = action.getAttributes(); String actionName = nodeMaps.getNamedItem("name").getNodeValue(); String actionClass = nodeMaps.getNamedItem("class").getNodeValue(); String actionMethod = nodeMaps.getNamedItem("method").getNodeValue(); ActionConfig actionConfig = new ActionConfig(actionName, actionClass, actionMethod); NodeList resultNodes = action.getChildNodes(); for (int j = 0; j < resultNodes.getLength(); j++) { Node result = resultNodes.item(j); if (null != result && result.getNodeType() == Node.ELEMENT_NODE) { String resultName = result.getAttributes().getNamedItem("name").getNodeValue(); String resultContent = result.getTextContent(); ResultConfig resultConfig = new ResultConfig(resultName, resultContent); Map<String, ResultConfig> resultConfigs = actionConfig.getResult(); resultConfigs.put(resultName, resultConfig); } } actionConfigs.put(actionName, actionConfig); } } return actionConfigs; } }
最为核心的前端控制器。
DispatcherServlet.java
package com.cloud.mvc.servlet; import java.io.IOException; import java.lang.reflect.Method; import java.util.Map; import javax.servlet.ServletException; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import com.cloud.mvc.config.ActionConfig; import com.cloud.mvc.config.ResultConfig; import com.cloud.mvc.util.XMLUtil; public class DispatcherServlet extends HttpServlet { private static final long serialVersionUID = -1105148104350928967L; private Map<String, ActionConfig> actionConfigs; @Override public void init() throws ServletException { super.init(); try { actionConfigs = XMLUtil.parseConfig(); } catch (Exception e) { e.printStackTrace(); } } @Override protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { this.doPost(req, resp); } @Override protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { String contextPath = req.getContextPath(); String uri = req.getRequestURI(); String actionName = uri.substring(contextPath.length() + 1, uri.length()); ActionConfig actionConfig = actionConfigs.get(actionName); String clazz = actionConfig.getClazz(); String method = actionConfig.getMethod(); String result = null; try { Class<?> clazzObject = Class.forName(clazz); Method controllerMethod = clazzObject.getMethod(method, HttpServletRequest.class); result = (String) controllerMethod.invoke(clazzObject.newInstance(), req); } catch (Exception e) { e.printStackTrace(); } Map<String, ResultConfig> resultConfigs = actionConfig.getResult(); String returnPath = resultConfigs.get(result).getContent(); req.getRequestDispatcher(returnPath).forward(req,resp); } }
接下来就是将DispatcherServlet类配置到web.xml中。
web.xml
<?xml version="1.0" encoding="UTF-8"?> <web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://java.sun.com/xml/ns/javaee" xmlns:web="http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd" id="WebApp_ID" version="3.0"> <display-name>Archetype Created Web Application</display-name> <servlet> <servlet-name>mvc</servlet-name> <servlet-class>com.cloud.mvc.servlet.DispatcherServlet</servlet-class> <init-param> <param-name>contextConfigLocation</param-name> <param-value>classpath:mvc.xml</param-value> </init-param> <load-on-startup>1</load-on-startup> </servlet> <servlet-mapping> <servlet-name>mvc</servlet-name> <url-pattern>/</url-pattern> </servlet-mapping> </web-app>
在此只写一个controller类,用于测试。
UserController.java
package com.cloud.mvc.controller; import javax.servlet.http.HttpServletRequest; public class UserController { public String select(HttpServletRequest req) { return "faild"; } }
运行结果:
相关推荐
### 自己动手写PHP MVC框架 #### 一、引言 在PHP开发领域,MVC(Model-View-Controller)架构模式被广泛采用,它通过将业务逻辑、数据处理与界面展示分离,使得代码结构更加清晰,提高了软件的可维护性和可扩展性...
微信开发php+mvc框架
在自己手写PHP MVC框架的过程中,你可以深入理解这一模式的工作原理,提升编程技能。 **模型(Model)**: 模型是应用的核心,负责处理业务逻辑和数据管理。在PHP MVC框架中,模型类通常与数据库交互,执行增删改查...
4. 整合上述组件:将ORM集成到MVC框架中,使模型能够通过ORM与数据库交互;同时,利用IOC容器管理MVC组件的实例,实现它们之间的依赖关系。 5. 编写测试:为框架编写单元测试和集成测试,验证各个部分的功能是否...
【标题】:“自己动手写框架”是一门针对初学者的实践教程,旨在引导学习者从零开始构建自己的软件开发框架。这个过程不仅涵盖了基础的编程原理,还涉及到设计模式、软件架构以及模块化等多个关键领域。 【描述】:...
在本项目"java自动动手做框架mvc"中,我们将探讨如何构建一个简单的MVC框架,并了解其核心概念。 **Model(模型)**:模型是应用程序的核心,负责处理业务逻辑和数据。在Java中,这通常涉及到数据库交互、数据验证...
在本文中,我们将深入探讨自己动手编写SpringMVC框架项目源代码涉及的关键知识点。 1. MVC架构 MVC是一种设计模式,它将应用程序分为三个主要组件:模型(Model)、视图(View)和控制器(Controller)。模型负责...
在JavaWeb开发中,MVC(Model-View-Controller)模式是一种常见的设计模式,它将应用程序的业务逻辑、用户界面和数据访问分离,使得各部分的职责更加...动手实践吧,祝你在实现自己的JavaWeb MVC框架的旅程中收获满满!
### 自己动手写Struts:构建基于MVC的Web开发框架 #### 1. 框架的理解与自我开发的重要性 在IT行业中,“框架”是一个高频词汇,它指的是为了解决某一类问题而预先设计的一套结构化解决方案。对于Java程序员而言,...
接下来,我会通过具体的示例和练习,让读者亲自动手实践,深入理解MVC框架的实质和应用。最后,我会分享一些开发过程中的常见问题和解决方案,帮助读者更好地应对实际开发中的挑战。希望这份资源能对大家有所帮助,...
接下来,我会通过具体的示例和练习,让读者亲自动手实践,深入理解MVC框架的实质和应用。最后,我会分享一些开发过程中的常见问题和解决方案,帮助读者更好地应对实际开发中的挑战。希望这份资源能对大家有所帮助,...
ASP.NET MVC框架是微软开发的一款用于构建Web应用程序的开源框架,它基于模型-视图-控制器(MVC)设计模式,提供了高度可测试性和可维护性。本系列课程的第14部分主要聚焦于MVC中的过滤器机制,包括Authorization ...
描述中的“自己动手写框架”是指开发者将学习如何从零开始构建一个框架,理解框架的基本结构和工作原理。这通常包括路由处理、中间件的编写和组织、错误处理、中间件的顺序控制等。通过这种方式,开发者不仅能更好地...
MVC框架是一种广泛应用于Web开发的设计模式,它将应用程序的业务逻辑、用户界面和数据访问分离开来,从而提高了代码的可维护性和可扩展性。在本文中,我们将深入探讨MVC框架的基本概念、工作原理以及如何在实际项目...
《自己动手写一个Spring》这篇文章主要探讨了Spring框架的核心概念,并通过模拟其实现来帮助读者深入理解其工作原理。Spring是Java开发中最流行的框架之一,它以依赖注入(Dependency Injection,DI)和面向切面编程...
【PHP MVC框架详解】 在IT领域,PHP是一种广泛使用的服务器端脚本语言,尤其在Web开发中占有重要地位。PHP MVC(Model-View-Controller)框架是PHP开发中的一种常见架构模式,它将应用程序的不同部分分离开来,使得...
**正文** 本项目是一个基于MVC(Model-View-Controller)框架实现的ATM取款功能,主要目标是为了帮助初学者理解MVC...通过实际动手操作,你可以更好地理解这一模式在解决实际问题中的价值,并逐渐提升自己的编程技能。
一、文件结构 建立3个文件夹 controller文件夹存放控制器文件 view文件夹存放视图文件 model文件夹存放数据文件 建立1个index.php 作为唯一入口 二、控制器 我们在controller文件夹下建立一个democontroller.php文件...
动手实践Struts源代码的编写,不仅可以提升对Struts框架的理解,还能加深对MVC模式的认识,对于Java Web开发者来说,这是一个非常有价值的学习经历。通过实际操作,你将能够更好地应对项目开发中的挑战,提高问题...
2. **实践自定义MVC框架**:尝试自己动手实现一个简单的MVC框架,可以从处理HTTP请求、解析配置文件等方面入手。 3. **学习现有的MVC框架**:如Struts、Spring MVC等,这有助于加深对MVC的理解,并提供更多的实践...