首页 诗词 字典 板报 句子 名言 友答 励志 学校 网站地图
当前位置: 首页 > 教程频道 > 软件管理 > 软件开发 >

JSP温习笔记——第11章 JSP 构架和MVC设计模式

2012-12-23 
JSP复习笔记——第11章 JSP 构架和MVC设计模式11.1 JSP Model I 体系结构11.2 JSP Model II 体系结构/MVC设

JSP复习笔记——第11章 JSP 构架和MVC设计模式
11.1 JSP Model I 体系结构  
11.2 JSP Model II 体系结构/MVC设计模式
11.3 使用MVC设计模式改写用户注册程序
    11.3.1 使用serlvet实现Controller层
    11.3.2 使用jsp实现表示层
    11.3.3 使用JavaBean实现模型层
11.3.4 使用JDBC和DAO模式实现数据库层
   11.4 本章小结


JSP + DAO设计模式

使用JSP+JavaBean开发速度快,有一个问题:JSP与JavaBean之间紧密耦合在一起,会对开发及维护造成麻烦。
使用JSP+JavaBean(模式1)开发适用于一次开发完成,而且团队成员较少是使用。
JSP + Servlet + JavaBean
对于模式一JSP与JavaBean之间紧密耦合在一起

分析:
JSP优点,开发前台界面方便,做UI开发容易
Servlet优点:是JAVA程序,安全性高,性能高
Servlet缺点:显示不方便

JavaBean优点:可重复调用,需要接受用户的请求参数,进行相应的处理

问题:
JSP跳转到Servlet可以通过表单或超链接
从Servlet跳转到JSP:使用response对象

Java代码
1.<h1>MVCDEMO</h1>  
2.<!--  
3.<h1><%=request.getAttribute("name")%></h1>  
4.-->  
5.<h1><%=session.getAttribute("name")%></h1>  
6.――――――――― 
<h1>MVCDEMO</h1>
<!--
<h1><%=request.getAttribute("name")%></h1>
-->
<h1><%=session.getAttribute("name")%></h1>
―――――――――


Java代码
1.package org.sky.darkness.servlet ;  
2. 
3.import java.io.* ;  
4.import javax.servlet.* ;  
5.import javax.servlet.http.* ;  
6. 
7.public class MVCServlet extends HttpServlet  
8.{  
9.    public void doGet(HttpServletRequest req,HttpServletResponse resp) throws IOException,ServletException  
10.    {  
11.        this.doPost(req,resp) ;  
12.    }  
13.    public void doPost(HttpServletRequest req,HttpServletResponse resp) throws IOException,ServletException  
14.    {  
15.        // 要传递一个值到mvcdemo.jsp中  
16.    // 要传递的内容只使用一次  
17.    // 一个页面跳转有用,request范围  
18.    // 既然request无法传递,那就扩大范围-session  
19.        // req.setAttribute("name","darkness") ;  
20.req.getSession().setAttribute("name","darkness") ;  
21.        resp.sendRedirect("mvcdemo.jsp");  
22.    }  
23.}; 
package org.sky.darkness.servlet ;

import java.io.* ;
import javax.servlet.* ;
import javax.servlet.http.* ;

public class MVCServlet extends HttpServlet
{
public void doGet(HttpServletRequest req,HttpServletResponse resp) throws IOException,ServletException
{
this.doPost(req,resp) ;
}
public void doPost(HttpServletRequest req,HttpServletResponse resp) throws IOException,ServletException
{
// 要传递一个值到mvcdemo.jsp中
    // 要传递的内容只使用一次
    // 一个页面跳转有用,request范围
    // 既然request无法传递,那就扩大范围-session
// req.setAttribute("name","darkness") ;
req.getSession().setAttribute("name","darkness") ;
resp.sendRedirect("mvcdemo.jsp");
}
};

JSP中两种跳转语句:
四种属性范围:
现在从Servlet中要传递的值只使用一次,如果把此值存放在session范围之中,则此内容只要用户一直与服务器保持连接,则此块内存空间要一直被占用,那么性能会很低。

解决方法:RequestDispatcher接口,是用于由Servlet到JSP进行服务器端跳转的接口

Java代码
1.req.setAttribute("name","darkness") ;  
2.// 与<jsp:forward/>功能相同  
3.req.getRequestDispatcher("mvcdemo.jsp").forward(req,resp); 
req.setAttribute("name","darkness") ;
// 与<jsp:forward/>功能相同
req.getRequestDispatcher("mvcdemo.jsp").forward(req,resp);


Java代码
1.-----------------mvc_login.jsp--------------------------------------  
2.<form action="mvcdemo" method="POST">  
3.输入姓名:<input type="text" name="uname">  
4.<input type="submit" value="提交">  
5.</form> 
-----------------mvc_login.jsp--------------------------------------
<form action="mvcdemo" method="POST">
输入姓名:<input type="text" name="uname">
<input type="submit" value="提交">
</form>


Java代码
1.-----------------MVCServlet.java--------------------------------------  
2.package org.sky.darkness.servlet ;  
3. 
4.import java.io.* ;  
5.import javax.servlet.* ;  
6.import javax.servlet.http.* ;  
7.import org.sky.darkness.bean.MVCCheck ;  
8. 
9.public class MVCServlet extends HttpServlet  
10.{  
11.    public void doGet(HttpServletRequest req,HttpServletResponse resp) throws IOException,ServletException  
12.    {  
13.        this.doPost(req,resp) ;  
14.    }  
15.    public void doPost(HttpServletRequest req,HttpServletResponse resp) throws IOException,ServletException  
16.    {  
17.        String name = req.getParameter("uname") ;  
18.        MVCCheck mc = new MVCCheck() ;  
19.        // 将请求内容设置到mc对象之中  
20.        mc.setName(name) ;  
21.        String path = null ;  
22.        if(mc.isValidate())  
23.        {  
24.            // 保存名字在request范围之中  
25.            req.setAttribute("name",mc.getName()) ;  
26.            path = "mvc_success.jsp" ;  
27.        }  
28.        else 
29.        {  
30.            path = "mvc_failure.jsp" ;  
31.        }  
32.        // 进行跳转  
33.        req.getRequestDispatcher(path).forward(req,resp) ;  
34.    }  
35.};  
36./* 
37.  <servlet> 
38.    <servlet-name>mvc</servlet-name> 
39.    <servlet-class>org.sky.darkness.servlet.MVCServlet</servlet-class> 
40.  </servlet> 
41.  <servlet-mapping> 
42.    <servlet-name>mvc</servlet-name> 
43.    <url-pattern>/mvcdemo</url-pattern> 
44.  </servlet-mapping> 
45.*/ 
-----------------MVCServlet.java--------------------------------------
package org.sky.darkness.servlet ;

import java.io.* ;
import javax.servlet.* ;
import javax.servlet.http.* ;
import org.sky.darkness.bean.MVCCheck ;

public class MVCServlet extends HttpServlet
{
public void doGet(HttpServletRequest req,HttpServletResponse resp) throws IOException,ServletException
{
this.doPost(req,resp) ;
}
public void doPost(HttpServletRequest req,HttpServletResponse resp) throws IOException,ServletException
{
String name = req.getParameter("uname") ;
MVCCheck mc = new MVCCheck() ;
// 将请求内容设置到mc对象之中
mc.setName(name) ;
String path = null ;
if(mc.isValidate())
{
// 保存名字在request范围之中
req.setAttribute("name",mc.getName()) ;
path = "mvc_success.jsp" ;
}
else
{
path = "mvc_failure.jsp" ;
}
// 进行跳转
req.getRequestDispatcher(path).forward(req,resp) ;
}
};
/*
  <servlet>
<servlet-name>mvc</servlet-name>
<servlet-class>org.sky.darkness.servlet.MVCServlet</servlet-class>
  </servlet>
  <servlet-mapping>
<servlet-name>mvc</servlet-name>
<url-pattern>/mvcdemo</url-pattern>
  </servlet-mapping>
*/


Java代码
1.------------------------------------- MVCCheck.java------------------------  
2.package org.sky.darkness.bean ;   
3. 
4.public class MVCCheck  
5.{  
6.    private String name ;  
7. 
8.    public void setName(String name)  
9.    {  
10.        this.name = name ;  
11.    }  
12.    public String getName()  
13. 
14.    {  
15.        return this.name ;  
16.    }  
17.    // 验证  
18.    public boolean isValidate()  
19.    {  
20.        if(this.name==null||"".equals(this.name))  
21.        {  
22.            return false ;  
23.        }  
24.        else 
25.        {  
26.            return true ;  
27.        }  
28.    }  
29.}; 
------------------------------------- MVCCheck.java------------------------
package org.sky.darkness.bean ;

public class MVCCheck
{
private String name ;

public void setName(String name)
{
this.name = name ;
}
public String getName()

{
return this.name ;
}
// 验证
public boolean isValidate()
{
if(this.name==null||"".equals(this.name))
{
return false ;
}
else
{
return true ;
}
}
};


Java代码
1.<%@page contentType="text/html;charset=gb2312"%>  
2.<h1>输入成功!!!</h1>  
3.<h2>欢迎:<%=request.getAttribute("name")%>光临!!!</h2>  
4.<%@page contentType="text/html;charset=gb2312"%>  
5.<h1>输入失败!!!</h1>  
6.<h2><a href="mvc_login.htm">重新登陆</a></h2> 
<%@page contentType="text/html;charset=gb2312"%>
<h1>输入成功!!!</h1>
<h2>欢迎:<%=request.getAttribute("name")%>光临!!!</h2>
<%@page contentType="text/html;charset=gb2312"%>
<h1>输入失败!!!</h1>
<h2><a href="mvc_login.htm">重新登陆</a></h2>



通过以上代码可以发现:
使用MVC开发程序,代码稍微复杂
JSP与JavaBean之间没有什么特别明显的直接关系
而Servlet根据JavaBean返回的内容进行跳转

Servlet中最好只有以下几种代码:
? 接收参数
? 调用JavaBean
? 进行跳转
? 有一些简单的逻辑判断


案例-使用mvc+DAO完成用户登陆

Java代码
1.DROP TABLE person ;  
2. 
3.CREATE TABLE person   
4.(  
5.    id varchar(20) not null primary key ,  
6.    name varchar(20) not null ,  
7.    password varchar(20)   
8.) ;  
9. 
10.INSERT INTO person (id,name,password) VALUES ('sky','darkness','wind') ;  
11.INSERT INTO person (id,name,password) VALUES ('cloud','hacker','creaker') ;  
12. 
13.-- 提交事务  
14.commit ; 
DROP TABLE person ;

CREATE TABLE person
(
id varchar(20) not null primary key ,
name varchar(20) not null ,
password varchar(20)
) ;

INSERT INTO person (id,name,password) VALUES ('sky','darkness','wind') ;
INSERT INTO person (id,name,password) VALUES ('cloud','hacker','creaker') ;

-- 提交事务
commit ;



Java代码
1.-----------------login.jsp--------------------------------  
2.<%@page contentType="text/html;charset=gb2312"%>  
3.<%@page import="java.util.*"%>  
4.<html>  
5.<head>  
6.    <title>登陆</title>  
7.</head>  
8.<body>  
9.<center>  
10.    <h1>登陆范例——MVC实现</h1>  
11.    <hr>  
12.    <br>  
13.    <br>  
14.    <!-- 加入更加详细的错误提示 -->  
15.    <%  
16.        if(request.getAttribute("errors")!=null)  
17.        {  
18.            // 有错误,要进行打印输出  
19.            List all = (List)request.getAttribute("errors") ;  
20.            Iterator iter = all.iterator() ;  
21.            while(iter.hasNext())  
22.            {  
23.    %>  
24.                <li><%=iter.next()%>  
25.    <%  
26.            }  
27.        }  
28.    %>  
29.      
30.    <form action="LoginServlet" method="post">  
31.    <table>  
32.        <tr>  
33.            <td colspan="2">用户登陆</td>  
34.        </tr>  
35.        <tr>  
36.            <td>用户名:</td>  
37.            <td><input type="text" name="id" value="${person.id}"></td>  
38.        </tr>  
39.        <tr>  
40.            <td>密&nbsp;&nbsp;码:</td>  
41.            <td><input type="password" name="password" value="${person.password}"></td>  
42.        </tr>  
43.        <tr>  
44.            <td colspan="2">  
45.            <input type="submit" value="登陆">  
46.            <input type="reset" value="重置">  
47.            </td>  
48.        </tr>  
49.    </table>  
50.    </form>  
51.</center>  
52.</body>  
53.</html>  
54.-------------------------LoginServlet.java---------------------------------------------  
55.// 建立MVC中的C,完成JSP+Servlet+JavaBean的开发模式  
56. 
57.package org.sky.darkness.servlet ;  
58. 
59.import java.io.* ;  
60.import java.util.* ;  
61.import javax.servlet.* ;  
62.import javax.servlet.http.* ;  
63.import org.sky.darkness.factory.* ;  
64.import org.sky.darkness.vo.* ;  
65. 
66.public class LoginServlet extends HttpServlet  
67.{  
68.    public void doGet(HttpServletRequest request,HttpServletResponse response) throws IOException,ServletException  
69.    {  
70.        this.doPost(request,response) ;  
71.    }  
72.    public void doPost(HttpServletRequest request,HttpServletResponse response) throws IOException,ServletException  
73.    {  
74.        // 声明一个集合类,用于保存错误信息  
75.        List errors = new ArrayList() ;  
76.        // 完成登陆验证,替换掉login_conf.jsp  
77.        String path = "login.jsp" ;  
78.        // 1、接收请求内容  
79.        String id = request.getParameter("id") ;  
80.        String password = request.getParameter("password") ;  
81.        // 2、进行数据合法性验证,包括是否为空,长度是否满足等  
82.        // 要将接收到的内容设置给PersonVO对象  
83.        PersonVo pv = new PersonVo() ;  
84.        pv.setId(id) ;  
85.        pv.setPassword(password) ;  
86.        pv.setErrors(errors) ;  
87.        // 3、如果合法,则进行数据库验证  
88.        if(pv.invalidate())  
89.        {  
90.            // 数据合法,可以进行数据库验证  
91.            if(DAOFactory.getPersonDAOInstance().isLogin(pv))  
92.            {  
93.                // 用户ID、密码合法  
94.                // 修改跳转路径  
95.                // 保存用户名到request范围之中  
96.                // request.setAttribute("name",pv.getName()) ;  
97.                path = "login_success.jsp" ;  
98.            }  
99.            else 
100.            {  
101.                // 用户ID、密码非法  
102.                errors.add("错误的用户ID及密码!") ;  
103.            }  
104.        }  
105.        // 将错误信息保存  
106.        request.setAttribute("errors",errors) ;  
107.        request.setAttribute("person",pv) ;  
108.        request.getRequestDispatcher(path).forward(request,response) ;  
109.    }  
110.};  
111.-------------------------login_success.jsp---------------------------  
112.<%@page contentType="text/html;charset=gb2312"%>  
113.<html>  
114.<head>  
115.    <title>登陆</title>  
116.</head>  
117.<body>  
118.<center>  
119.    <h1>登陆范例——MVC实现</h1>  
120.    <hr>  
121.    <br>  
122.    <br>  
123.    <h2>登陆成功</h2>  
124.    <h3>欢迎<font color="red" size="15">  
125.        ${person.name}  
126.    </font>光临!!!</h3>  
127.</center>  
128.</body>  
129.</html>  
130.-----------------------------PersonVo.java---------------------  
131.// 只包含setter和getter方法的类  
132. 
133.package org.sky.darkness.vo ;  
134. 
135.import java.util.* ;  
136.public class PersonVo  
137.{  
138.    // 表中所有字段  
139.    private String id ;  
140.    private String name ;  
141.    private String password ;  
142.    // 此属性用于保存全部错误信息  
143.    private List errors ;  
144. 
145.    public boolean invalidate()  
146.    {  
147.        boolean flag = true ;  
148.        // 验证ID  
149.        if(this.id==null||"".equals(this.id))  
150.        {  
151.            flag = false ;  
152.            errors.add("ID不能为空!") ;  
153.        }  
154.        else 
155.        {  
156.            // 进行长度验证:3~10位  
157.            if(this.id.length()<3||this.id.length()>10)  
158.            {  
159.                flag = false ;  
160.                errors.add("ID的长度应为3~10位!") ;  
161.            }  
162.        }  
163.        // 验证密码  
164.        if(this.password==null||"".equals(this.password))  
165.        {  
166.            flag = false ;  
167.            errors.add("密码不能为空!") ;  
168.        }  
169.        else 
170.        {  
171.            // 进行长度验证:3~10位  
172.            if(this.password.length()<3||this.password.length()>10)  
173.            {  
174.                flag = false ;  
175.                errors.add("密码的长度应为3~10位!") ;  
176.            }  
177.        }  
178.        return flag ;  
179.    }  
180. 
181.    public void setErrors(List errors)  
182.    {  
183.        this.errors = errors ;  
184.    }  
185.    public List getErrors()  
186.    {  
187.        return this.errors ;  
188.    }  
189. 
190.    // 生成getter和setter方法  
191.    public void setId(String id)  
192.    {  
193.        this.id = id ;  
194.    }  
195.    public void setName(String name)  
196.    {  
197.        this.name = name ;  
198.    }  
199.    public void setPassword(String password)  
200.    {  
201.        this.password = password ;  
202.    }  
203.    public String getId()  
204.    {  
205.        return this.id ;  
206.    }  
207.    public String getName()  
208.    {  
209.        return this.name ;  
210.    }  
211.    public String getPassword()  
212.    {  
213.        return this.password ;  
214.    }  
215.};  
216.----------------------------------PersonDAO.java----------------------  
217.// 本接口定义本项目中所操作person表的全部方法  
218. 
219.package org.sky.darkness.dao ;  
220. 
221.// 使用PersonVo类  
222.import org.sky.darkness.vo.* ;  
223. 
224.public interface PersonDAO  
225.{  
226.    // 需要一个登陆验证的方法  
227.    public boolean isLogin(PersonVo pv) ;  
228.}  
229.---------------------PersonDAOImpl.java---------------------------------  
230.// 具体实现DAO接口的类  
231. 
232.package org.sky.darkness.daoimpl ;  
233. 
234.// 需要连接数据库  
235.// 需要对VO的内容进行具体的验证  
236.import java.sql.* ;  
237.import org.sky.darkness.dao.* ;  
238.import org.sky.darkness.dbc.* ;  
239.import org.sky.darkness.vo.* ;  
240. 
241.public class PersonDAOImpl implements PersonDAO  
242.{  
243.    public boolean isLogin(PersonVo pv)  
244.    {  
245.        boolean flag = false ;  
246.        // 在此处成具体的数据库验证  
247. 
248.        // 声明一个数据库操作对象  
249.        PreparedStatement pstmt = null ;  
250.        // 声明一个结果集对象  
251.        ResultSet rs            = null ;  
252.        // 声明一个SQL变量,用于保存SQL语句  
253.        String sql              = null ;  
254.        // DataBaseConnection为具体的数据库连接及关闭操作类  
255.        DataBaseConnection dbc  = null ;  
256.        // 连接数据库  
257.        dbc = new DataBaseConnection() ;  
258. 
259.        // 编写SQL语句  
260.        sql = "SELECT name FROM person WHERE id=? and password=?" ;  
261.        try 
262.        {             
263.            // 实例化数据库操作对象  
264.            pstmt = dbc.getConnection().prepareStatement(sql) ;  
265. 
266.            // 设置pstmt的内容,是按ID和密码验证  
267.            pstmt.setString(1,pv.getId()) ;  
268.            pstmt.setString(2,pv.getPassword()) ;  
269. 
270.            // 查询记录  
271.            rs = pstmt.executeQuery() ;  
272.            // 判断是否有记录  
273.            if(rs.next())  
274.            {  
275.                // 如果有记录,则执行此段代码  
276.                // 用户是合法的,可以登陆  
277.                flag = true ;  
278.                pv.setName(rs.getString(1)) ;  
279.            }  
280.            // 依次关闭  
281.            rs.close() ;  
282.            pstmt.close() ;  
283.        }  
284.        catch(Exception e)  
285.        {  
286.            System.out.println(e) ;  
287.        }  
288.        finally 
289.        {  
290.            // 最后一定要保证数据库已被关闭  
291.            dbc.close() ;  
292.        }  
293.        return flag ;  
294.    }  
295.};  
296.-------------------------DAOFactory.java---------------------  
297.// 取得DAO实例的工厂类  
298. 
299.package org.sky.darkness.factory ;  
300. 
301.import org.sky.darkness.dao.* ;  
302.import org.sky.darkness.daoimpl.* ;  
303. 
304.public class DAOFactory  
305.{  
306.    public static PersonDAO getPersonDAOInstance()  
307.    {  
308.        return new PersonDAOImpl() ;  
309.    }  
310.};  
311.---------------------DataBaseConnection.java------------------  
312.// 本类只用于数据库连接及关闭操作  
313.package org.sky.darkness.dbc ;  
314. 
315.import java.sql.* ;  
316. 
317.public class DataBaseConnection  
318.{  
319.    // 属性  
320.    // 定义数据库操作的常量、对象  
321.    // 数据库驱动程序  
322.    private final String DBDRIVER   = "oracle.jdbc.driver.OracleDriver" ;  
323.    // 数据库连接地址  
324.    private final String DBURL      = "jdbc:oracle:thin:@localhost:1521:sky" ;  
325.    // 数据库用户名  
326.    private final String DBUSER     = "scott" ;  
327.    // 数据库连接密码  
328.    private final String DBPASSWORD = "tiger" ;  
329.    // 声明一个数据库连接对象  
330.    private Connection conn         = null ;  
331. 
332.    // 在构造方法之中连接数据库  
333.    public DataBaseConnection()  
334.    {  
335.        try 
336.        {  
337.            // 加载驱动程序  
338.            Class.forName(DBDRIVER) ;   
339.            // 连接数据库  
340.            conn = DriverManager.getConnection(DBURL,DBUSER,DBPASSWORD) ;     
341.        }  
342.        catch (Exception e)  
343.        {  
344.            System.out.println(e) ;  
345.        }  
346.    }  
347. 
348.    // 返回一个数据库连接  
349.    public Connection getConnection()  
350.    {  
351.        /// 返回连接对象  
352.        return this.conn ;  
353.    }  
354. 
355.    // 关闭数据库连接  
356.    public void close()  
357.    {  
358.        try 
359.        {  
360.            this.conn.close() ;           
361.        }  
362.        catch (Exception e)  
363.        {  
364.        }  
365.    }  
366.}; 

热点排行