1 JSP+JavaBean开发模式
1.1 JSP+JavaBean开发模式架构
1.2 JSP+JavaBean开发模式表写计算器
2 Servlet+JSP+JavaBean开发模式
2.1 Web开发中的请求-响应模型
2.2 标准MVC模型概述
2.3 MVC(Model-View-Controller)的概念
2.4 Web MVC概述
2.5 Servlet+JSP+JavaBean开发模式介绍
3 Servlet+JSP+JavaBean开发模式的缺点
3.1 Servlet作为控制器的缺点
3.2 JavaBean作为模型的缺点
在JSP+JavaBean架构中,JSP负责控制逻辑、表现逻辑、业务逻辑(JavaBean)的调用。
JSP+JavaBean模式适合开发业务逻辑不太复杂的web应用程序,这种模式下,JavaBean用于封装业务数据,JSP及负责处理用户请求,显示数据。
首先分析一下JSP和JavaBean各自的责任,JSP负责显示计算器(calculator)页面,工用户输入计算数据,并显示计算后的结果,JavaBean负责接收用户输入的计算数据并进行计算。
1、编写CalculatorBean,负责接收用户输的数据进行计算
CalculatorBean.java
package cn.calculator.domain;
import java.math.BigDecimal;
/**
* @author Administrator
*/
public class CalculatorBean {
private double firstNum;
private double secondNum;
private char operator = '+';
private double result;
public double getFirstNum() {
return firstNum;
}
public void setFirstNum(double firstNum) {
this.firstNum = firstNum;
}
public double getSecondNum() {
return secondNum;
}
public void setSecondNum(double secondNum) {
this.secondNum = secondNum;
}
public char getOperator() {
return operator;
}
public void setOperator(char operator) {
this.operator = operator;
}
public double getResult() {
return result;
}
public void setResult(double result) {
this.result = result;
}
/*
* 用于计算
*/
public void calculate() {
switch (this.operator) {
case '+':{
this.result = this.firstNum + this.secondNum;
break;
}
case '-':{
this.result = this.firstNum - this.secondNum;
break;
}
case '*':{
this.result = this.firstNum * this.secondNum;
break;
}
case '/':{
if(this.secondNum==0) {
throw new RuntimeException("被除数不能是0");
}
this.result = this.firstNum / this.secondNum;
this.result = new BigDecimal(this.result).setScale(2, BigDecimal.ROUND_HALF_UP).doubleValue();
break;
}
default:
throw new RuntimeException("对不起,传入的运算符非法!!");
}
}
}
2、编写calculator.jsp,负责显示计算器页面,供用户输入数据,并显示计算后的结果
calculator.jsp页面代码如下:
<%@ page language="java" contentType="text/html; charset=UTF-8"
pageEncoding="UTF-8"%>
<%--使用cn.calculator.domain.CalculatorBean --%>
<%--接收用户输入的数据 --%>
<%
//使用CalculatorBean进行计算
calculatorBean.calculate();
%>
使用【JSP+JavaBean开发模式】开发的简单计算器
计算结果是:
=
运行结果如下:
在平时的JavaWeb项目开发中,在不适用第三方MVC开发框架的情况下,通常会选择Servlet+JSP+JavaBean开发模式来开发JavaWeb项目,Servlet+JSP+JavaBean组合就是一种MVC开发模式,控制器(Controller)采用Servlet、模型(Model)采用JavaBean、视图(View)采用JSP。在讲解Servlet+JSP+JavaBean开发模式前,先简单了解一下MVC开发模式。
在Web世界里,具体步骤如下:
1、Web浏览器发起请求;
2、Web服务器接收请求,处理请求,最后产生相应(一般为HTML);
Web服务器处理完成后,返回给Web客户端(一般使我们的浏览器),客户端对接收的内容进行处理(Web浏览器将会对接收到的HTML内容进行渲染以展示给用户)
因此在Web世界里:都是Web客户端发起请求,Web服务器接收、处理并产生相应。
一般Web服务器是不能触动通知Web客户端更新内容的。虽然现在有些技术如服务推(Comet)、还有现在的HTML5 WebSocket可以实现Web服务器注定通知Web客户端 。
MVC模型:是一种架构型模式,本身不引入新功能,只是帮助我们将开发的结构组织的更加合理,使展示与模型分离、流程控制逻辑。业务逻辑与展示逻辑分离。
首先让我们了解一下MVC(Model-View-Controller)的概念:
Model(模型):数据模型,提供要展示的数据,因此包含数据和行为,可以认为是领域模型(domain)或JavaBean组件(包含数据和行为),不过现在一般都分离开:Value Object(数据)和服务层(行为)。也就是模型提供了模型数据查询和模型数据的状态更新等功能,包括数据和业务。
View(视图):负责进行模型的展示,一般就是我们见到的用户见面,客户想要看到的东西。
Controller(控制器):接收用户请求,委托给模型进行处理(状态改变),处理完毕把返回的模型数据返回给视图,由视图负责展示。也就是控制器做了调度员的工作。
从上图中,在标准的MVC中模型能主动推数据给视图进行更新(观察者设计模式,在模型上注册视图,当模型更新自动跟新视图),但是在Web开发中模型是无法主动推给视图的(无主动更新用户界面),因为在Web开发中是请求-响应模型。
那接下来我们看一下在Web里MVC是什么样的。
Web MVC中的M(模型)-V(视图)-C(控制器)概念和标准MVC概念一样,我们再看一下Web MVC标准架构,如下图所示:
在Web MVC模式下,模型无法主动推数据给视图,如果用户想要视图更新,需要再发送一次请求(即请求-响应模型)。
Servlet+JSP+JavaBean架构其实可以认为就是我们所说的Web MVC模型,只是控制器采用Servlet、模型采用JavaBean、视图采用JSP,如图:
具体示例代码:
1.模型(model)
2.视图(view)
3.控制器(controller)
从Servlet+JSP+JavaBean(Web MVC)架构可以看出,视图和模型分离了,控制逻辑和展示逻辑分离了。
Servlet+JSP+JavaBean(Web MVC)架构虽然实现了视图和模型分离以及控制逻辑和展示逻辑分离,但也有一些比较严重的缺点。
此处的控制器使用Servlet,使用Servlet作为控制器有以下几个缺点:
1、控制逻辑可能比较复杂,其实我们可以按照规约,如请求参数submitFlag=toLogin,我们其实可以直接调用toLogin方法,来简化控制逻辑;而且每个模块基本需要一个控制器,造成控制逻辑可能很复杂。现在流行的Web MVC框架(如Struts2)都支持"请求参数submitFlag=toAdd,就可以直接调用toAdd方法"这样的处理机制,在Struts2中类似这样的处理机制就称为"动态方法调用"
2、请求参数到模型的封装比较麻烦,如果能交给框架来做这件事情,我们可以从中得到解放。
请求参数到模型的封装代码:
// 1收集参数
String username = req.getParameter("username");
String password = req.getParameter("password");
// 2封装参数
UserBean user = new UserBean();
user.setUsername(username);
user.setPassword(password);
当有几十个甚至上百个参数需要封装到模型中时,这样写恐怕就痛苦万分了,要写几十次甚至上百次这样的代码,估计写到吐了,所以现在流行的Web MVC框架(如Struts2)都提供了非常方便的获取参数,封装参数到模型的机制,减少这些繁琐的工作。
3、选择下一个视图,严重依赖Servlet API,这样很难或基本不可能更换视图。
例如:使用Servlet API提供的request对象的getRequestDispatcher方法选择要展示给用户看的视图
private void toLogin(HttpServletRequest req, HttpServletResponse resp)
throws ServletException, IOException {
//使用Servlet API提供的request对象的getRequestDispatcher方法选择视图
// 此处和JSP视图技术紧密耦合,更换其他视图技术几乎不可能
request.getRequestDispatcher("/mvc/login.jsp").forward(request, response);
}
4、给视图传输要展示的模型数据,也需要使用Servlet API,更换视图技术也要一起更换,很麻烦。
例如:使用Servlet API提供的request对象给视图传输要展示的模型数据
//使用Servlet API提供的request对象给视图login.jsp传输要展示的模型数据(user)
request.setAttribute("user", user);
request.getRequestDispatcher("/mvc/login.jsp").forward(request, response)
此处模型使用JavaBean,JavaBean组件类既负责收集封装数据,又要进行业务逻辑处理,这样可能造成JavaBean组件类很庞大,所以一般现在项目都是采用三层架构,而不直接采用JavaBean。