采用Java Web所实现的MVC结构图如下,其中控制器部分采用Servlet来实现,模型部分采用JavaBean来实现,而大部分的视图采用Jsp页面来实现。
思想基础
JSP+JavaBean两层结构工作原理应该是比较熟悉的,也比较好理解。
但是有一点必须要清楚就是用户通过浏览器来发送网页的请求,此请求到达服务器后在服务器端查找对应的网页,如果是首次请求(第二次就不用解释执行了),对于JSP来说要生成Servlet,然后通过Servlet引擎来执行 Servlet,把调用JavaBean的结果嵌入到页面中返回给用户的浏览器。
JSP+JavaBean+Servlet三层结构的实质是多了一个Controller:Servlet来分发客户端浏览器的请求。如果把起控制器作用的Servlet的作用理解为对客户端的请求进行预处理对理解Servlet将有很大的帮助。通过web.xml配置文件可以找到用户请求和特定的 Servlet的对应关系,每个Servlet都有一个特定的Servlet对象与之对应,所以说处理用户请求的就是一个继承自HttpServlet的 Servlet对象。
﹤!-- JSPC servlet mappings start --﹥ ﹤servlet﹥ ﹤servlet-name﹥ms1﹤/servlet-name﹥ ﹤servlet-class﹥news.FirstAction﹤/servlet-class﹥ ﹤/servlet﹥ ﹤servlet﹥ ﹤servlet-name﹥ms2﹤/servlet-name﹥ ﹤servlet-class﹥news.DetailAction﹤/servlet-class﹥ ﹤/servlet﹥ ﹤!-- JSPC servlet mappings end --﹥ ﹤servlet-mapping﹥ ﹤servlet-name﹥ms1﹤/servlet-name﹥ ﹤url-pattern﹥/newsmain﹤/url-pattern﹥ ﹤/servlet-mapping﹥ ﹤servlet-mapping﹥ ﹤servlet-name﹥ms2﹤/servlet-name﹥ ﹤url-pattern﹥/newsDetail﹤/url-pattern﹥ ﹤/servlet-mapping﹥
如上面所示的摘自web.xml的一段配置servlet,第一部分主要用来配置 Servlet与具体的Servlet对象关联,第二部分主要用来配置请求由哪个Servlet处理,Servlet名字的关联,处理请求就与具体 Servlet处理对象关联起来,比如说,客户端浏览器发来/newsmain的请求,它由ms1 servlet进行处理,通过ms1就能找到相对应的serlet对象news.FirstAction,即 /newsmain-﹥ms1-﹥news.FirstAction,这也就是配置文件的意义所在。到现在懂得了用户/newsmain请求会被news.FirstAction类的对象进行处理,所以说,要看懂程序就要看懂FirstAction的作用是什么就行了。比如说下面是 FirstAction的一个实现。
public final class FirstAction extends HttpServlet { protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { DB db = new DB(); HttpSession session = req.getSession(); try { session.setAttribute(Constants.NEWS_LIST_KEY, News .SearchNewsTitle(db)); } catch (Exception e) { e.printStackTrace(); } db.close(); String target = "/P43_News/newsMain.jsp"; resp.sendRedirect(target); } }
通过这个实现可以看到,当服务器收到客户端请求执行 News.SearchNewsTitle(db)的操作,然后把返回值通过session.setAttribute放到session里,然后通过 resp.sendRedirect(target)间接转移到newsMain.jsp,这样在newsMain.jsp里通过 session.getAttribute函数就可以得到在存储在session里的对应值。
回过头来就容易看出JSP+JavaBean工作原理和JSP+JavaBean+Servlet工作原理的不同了,两层结构必须把预处理放在JSP中进行,比如说 News.SearchNewsTitle(db),三层结构先把预处理在Servlet里进行了,然后相当于把这个处理结果通过Session返回给 JSP,让JSP更关注于界面的显示。
登陆注册模块需求
1 注册
1.1 用户的注册表单(用户名,密码,邮箱,昵称,验证码)
1.2 提交注册:要做(用户名,密码,邮箱,昵称,验证码)的校验。
1.2.1 用户名,密码,邮箱,昵称是在客户端浏览器完成的,通过JS来实现。
1.2.2 验证码的是要在服务器端的程序完成的。
2.如果注册表单的校验通过, 那么就进行业务逻辑的判断。
2.1 如果用户已经存在, 告诉用户错误信息。
2.2 如果邮箱已经存在, 告诉用户错误信息。
2.3 如果都不存在 .则进行第3步。
3. 将用户的信息 保存到数据库中
4. 注册 成功, 跳转到 登录 页面
5. 登陆
5.1 将用户的登陆信息发送到后台进行验证
5.2 如果验证成功,则跳转到首页
5.3 如果跳转失败,则跳转到登陆页面,并提示错误信息。
项目目录结构
项目的源码分成四个包文件,分别用来存取模型,视图,控制器和工具类,具体文件如下:
对于视图,我们定义三个JSP页面,如下所示:
定义视图
login.jsp页面
<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%><!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://.mysql.jdbc.Driver"); bInited = true; System.out.println("Success loading Mysql Driver!"); } public Connection getConnection() throws ClassNotFoundException, SQLException { if (!bInited) { initJDBC(); } // 连接URL为 jdbc:mysql//服务器地址/数据库名 // 后面的2个参数分别是登陆用户名和密码 Connection conn = DriverManager.getConnection( "jdbc:mysql://localhost:3306/数据库", "用户名", "密码"); return conn; } public boolean loginSuccess(String userName, String password) { boolean returnValue = false; String sql = "SELECT * FROM user where username=? and password=?"; Connection conn = null; PreparedStatement ps=null; int i=0; try { conn = getConnection(); ps=conn.prepareStatement(sql); ps.setString(1, userName); ps.setString(2, password); ResultSet rs=ps.executeQuery(); if(rs.next()){ returnValue=true; } } catch (ClassNotFoundException e) { e.printStackTrace(); } catch (SQLException e) { e.printStackTrace(); } return returnValue; } public boolean updateUser(User user) { boolean flag=false; int i=0; Connection conn=null; PreparedStatement ps=null; String sql= "insert into user (username,password,nickname,email) values(?,?,?,?)"; try { conn = getConnection(); ps=conn.prepareStatement(sql); ps.setString(1, user.getUsername()); //对占位符设置值,占位符顺序从1开始,第一个参数是占位符的位置,第二个参数是占位符的值。 ps.setString(2, user.getPassword()); ps.setString(3, user.getNickname()); ps.setString(4, user.getEmail()); i=ps.executeUpdate(); if(i>0){ flag=true; } } catch (ClassNotFoundException e) { e.printStackTrace(); }catch (SQLException e) { e.printStackTrace(); } return flag; } public boolean serchUserName(String userName){ boolean returnValue = false; String sql = "SELECT * FROM user where username=?"; Connection conn = null; PreparedStatement ps=null; try { conn = getConnection(); ps=conn.prepareStatement(sql); ps.setString(1, userName); ResultSet rs=ps.executeQuery(); if(rs.next()){ returnValue=true; } } catch (ClassNotFoundException e) { e.printStackTrace(); } catch (SQLException e) { e.printStackTrace(); } return returnValue; } public boolean serchEmail(String email){ boolean returnValue = false; String sql = "SELECT * FROM user where email=?"; Connection conn = null; PreparedStatement ps=null; int i=0; try { conn = getConnection(); ps=conn.prepareStatement(sql); ps.setString(1, email); ResultSet rs=ps.executeQuery(); if(rs.next()){ returnValue=true; } } catch (ClassNotFoundException e) { e.printStackTrace(); } catch (SQLException e) { e.printStackTrace(); } return returnValue; }}