MVC的架构模式,一直是JavaEE开发中所遵循的标准,如今很多框架都已经很好的实现了MVC,像大家所熟知的Struts,SpringMVC,JSF等,但是如果没有任何框架的支持,仅仅通过JavaWeb中servlet,jsp等基本知识的运用,可以如何实现MVC的开发模式呢?本文将通过一个实例来讲解Servlet所实现的MVC架构。

下载项目源码请点击这里:

由于本项目在实现的过程中有很多建立的细节,这些都在我之前的博文中有所提及,而在此文中就直接默认所有的问题都已经解决,然后运用了JDBC轻量型封装的DAO框架,对于Servlet中文乱码的解决采取了过滤器的处理,对于Mysql可能遇到的中文乱码问题都已经解决,如果有此方面的疑惑或者问题的话,可以参考一下三篇博文:

资源引入

下面就来构建整个ServletMVC的实战项目

项目概况

首先给出JavaWeb项目ServletMVC最终的结构,如图1所示:

JAVA项目实例含代码 java开发项目实例_List

图1

注:在这个过程中我将主要讲解基于Servlet的MVC架构的搭建过程,对于Dao的封装,mysql的sql语句,实体类的编写等我都不会提及,还请大家参照以上三个链接进行参照,最终的代码大家也可以在我分享的资源中进行下载,在这里我仅仅给出项目中相对重要的代码已经实现原理,有什么问题,我将会和大家共同解决。

下面给出在发布到Tomcat中后,通过http://localhost:8080/ServletMVC访问后可以实现的效果,如图2所示:

JAVA项目实例含代码 java开发项目实例_MVC_02

图2

在这里,主要的操作步骤如下:

用户名,生日,会员状态填写后,点击增加信息,成功后,跳到显示所有用户界面(如果没有填写,有相应的后台验证信息),该界面有直接访问所有用户界面的链接,可以直接跳转查看所有用户信息。

在所有用户显示界面,将有修改,删除,增加用户三个操作,如果点击修改,则跳转到用户修改界面,当然当前修改的信息将直接在修改界面中显示,如果点击删除,将从数据库中删除这条信息,如果点击增加用户,则回到步骤一。

在用户修改界面,将有确认修改信息和显示所有信息两个操作,如果确认修改,将对数据库中修改的用户进行信息更新,如果显示所有信息,将执行步骤二。

这个过程中有着action的标识的都是通过基于Servlet所实现的Controller来进行管理,也是MVC中的核心部分,下面将来讲解Servlet实现的MVC的具体原理和实现的步骤。

Servlet MVC实现

一般情况的MVC实现,都是通过多个继承HttpServlet的类充当Controller,因此需要多个Servlet的编写,在此文中,Controller只有一个类,而通过不同的Action类来执行相应的处理操作,因此在这个过程中,具体的实现原理图如图3所示:

JAVA项目实例含代码 java开发项目实例_java_03

图3

在这个过程中,主要的实现原理如下讲解:

比如图2中userAddAction这个过程,

ActionServlet是一个固定的处理方式的Controller,在方法中首先获取前端将要调用的action名字来确定指向,如userAddAction,前端就要通过userAdd.do?actionName=userAddAction的方式指向调用UserAddAction这个action类,而在ActionServlet中doPost(……)方法中,将获取actionName,即为userAddAction,而Servlet的生命周期中,init()方法是在doPost(……)方法之前执行的,而且只执行一次,这个时候因为已经将actionName对应userAddAction的实例new UserAddAction()添加到了application属性范围及web容器中(只有一个实例),所以就可以通过key->value的形式,获取要调用的action处理类UserAddAction,从而获取一个url,然后根据这个url跳转到相应的显示页面(show.jsp)。

下面给出这个过程中用到的资源文件:

ActionServlet.java(Controller)
package com.steven.controller;
import java.io.IOException;
import javax.servlet.ServletContext;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import com.steven.model.UserAddAction;
import com.steven.model.UserDeleteAction;
import com.steven.model.UserListAction;
import com.steven.model.UserModifyAction;
import com.steven.model.UserUpdateAction;
import com.steven.util.IModel;
/**
* 核心处理器类,用来接收客户端的所有请求, 根据请求调用模型实现业务逻辑的处理, 在模型处理完后根据处理结果再进行视图的转发
*
* @author Steven
*
*/
public class ActionServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp)
throws ServletException, IOException {
// TODO Auto-generated method stub
doPost(req, resp);
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp)
throws ServletException, IOException {
// 获取请求提交的参数,该参数用来标识执行的请求处理
String cmd = req.getParameter("actionName");
// 根据参数获取模型的实例
IModel model = (IModel) getServletContext().getAttribute(cmd);
// 通过模型对象进行业务逻辑处理,返回视图路径
String url = model.execute(req, resp);
// 根据视图路径进行页面转发
if (url != null) {
req.getRequestDispatcher(url).forward(req, resp);
} else {
// 如果路径出错,跳转到错误页面
req.getRequestDispatcher("WEB-INF/jsp/error.jsp")
.forward(req, resp);
}
}
@Override
public void init() throws ServletException {
// 获取application
ServletContext application = getServletContext();
// 将业务模型的实例写入application
application.setAttribute("userAddAction", new UserAddAction());
application.setAttribute("userListAction", new UserListAction());
application.setAttribute("userModifyAction", new UserModifyAction());
application.setAttribute("userDeleteAction", new UserDeleteAction());
application.setAttribute("userUpdateAction", new UserUpdateAction());
}
}
IModel(Model模型的接口)
IModel.java
package com.steven.util;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
/**
* 模型接口,所有业务逻辑处理器都应实现该接口
*
* @author Steven
*
*/
public interface IModel {
/**
* 模型接口
*
* @param request
* @return
*/
public String execute(HttpServletRequest request,HttpServletResponse resp);
}

实现的Model处理器(UserAddAction)

UserAddAction.java
package com.steven.model;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Calendar;
import java.util.List;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import com.steven.dao.impl.UserDao;
import com.steven.entity.User;
import com.steven.util.IModel;
/**
* 实现统一规定的模型
*
* @author Steven
*
*/
public class UserAddAction implements IModel {
// 获得数据库操作的DAO
private UserDao userDao;
// 获取日期操作类
private Calendar calendar = Calendar.getInstance();
public UserAddAction() {
userDao = new UserDao();
}
@Override
public String execute(HttpServletRequest request,
HttpServletResponse response) {
// 获取前台表单提交后的用户名
String userName = request.getParameter("userName");
// 生日
String birthday = request.getParameter("birthday");
// 是否是VIP
String isVip = request.getParameter("isVip");
// 进行转换VIP的boolean类型
boolean isVipFlag = isVip.toLowerCase().equals("yes") ? true : false;
// 年龄
int age = 0;
// 创建进行验证的标志信息
boolean checkFlag = true;
if ("".equals(userName) || userName == null) {
request.setAttribute("userNameError", "请填写用户名");
checkFlag = false;
} else {
request.setAttribute("userName", userName);
}
if (birthday != null && !"".equals(birthday)) {
// 得到年龄
age = calendar.get(Calendar.YEAR)
- Integer.parseInt(birthday.substring(0, 4)) + 1;
request.setAttribute("birthday", birthday);
} else {
checkFlag = false;
// 进行后台校验
request.setAttribute("birthdayError", "请选择出生日期");
}
// 如果有没有填的选项,则后台校验不成功,进行跳转
if (!checkFlag) {
return "index.jsp";
}
// 穿件要创建的用户
User user = null;
try {
// 生成用户信息
user = new User(userName, age, new SimpleDateFormat("yyyy-MM-dd")
.parse(birthday), isVipFlag);
} catch (ParseException e) {
e.printStackTrace();
}
// 将用户添加到数据库中
userDao.doCreate(user);
// 获取用户集合,进行显示
List userList = userDao.findAll();
// 设置到request属性范围中
request.setAttribute("userList", userList);
// 设置跳转后的页面
return "WEB-INF/jsp/show.jsp";
}
}

以及跳转后的视图show.jsp页面

show.jsp(View)
String path = request.getContextPath();
String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/";
%>
User Show 
 
cellspacing="0" border="1">

用户名

年龄

生日

是否是会员

操作


${user.userName }
${user.age }

不是

修改

删除

增加用户

暂时还没有任何数据,点击此处进行增添数据

因此这就是这个增加用户的功能所要实现的MVC封装架构,当然,对于ActionServlet需要配置web.xml

web.xml
String path = request.getContextPath();
String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/";
%>
User Show 
 
cellspacing="0" border="1">
用户名

年龄

生日

是否是会员

操作

${user.userName }
${user.age }

不是

修改

删除

增加用户

暂时还没有任何数据,点击此处进行增添数据

这就是Servlet MVC实现的核心,其他的功能比如显示所有用户,删除用户,修改用户,所对应的action分别编写,Controller只有ActionServlet这个类,而init()方法中要将需要的actionName增加进去。

从这里很容易看出MVC的主要优缺点

优点:

Controller将Model和View分离,降低耦合度

ActionServlet类的重用,以及IModel的统一

维护性高,以后添加相应的功能,只要照着这个模式编写即可,容易理解和编写

缺点:

这个过程都是根据具体项目实施采取的封装模式,除非使用一些现成的框架,固定住了这个模式,否则没有明确定义

开发前需要准备工作,对于一些轻型开发略显繁琐,复杂性高了,效率低了

所有的控制都有Controller来控制,将显得View和Controller紧密联系,重用性较低

这就是整个Servlet MVC实现的核心,具体的代码还请参照代码资源

在此恭祝大家新年快乐,学习愉快!