一、Servlet介绍
Servlet简单的说就是一个Java程序,目的和Javabean差不多,为了使得JSP页面中代码简洁、清晰;
JavaBean不需要配置,只需要放在WEB-INF/classes中即可;
Servlet也是放在 WEB-INF/classes/中,并在web.xml中配置如下形式:
<servlet>
<servlet-name></servlet-name>
<servlet-class></servlet-class>
</servlet>
<servlet-mapping>
<servlet-name></servlet-name>
<url-pattern></url-pattern>
</servlet-mapping>
如果需要设置配置信息,则需要形式如下:
<servlet>
<servlet-name></servlet-name>
<servlet-class></servlet-class>
<init-param>
<param-name></param-name>
<param-value></param-value>
</init-param>
</servlet>
<servlet-mapping>
<servlet-name></servlet-name>
<url-pattern></url-pattern>
</servlet-mapping>
注意:在url-pattern中,主目录为:"/",而不是"\"!
Servlet可以处理客户端传来的请求,即request,并且可以返回给客户端回应,即response,这个操作通过
(1)public void service(ServletRequest req,ServletResponse resp)throws ServletExeption,IOException{}
(2)public void doGet(HttpServletRequest req,HttpServletResponse resp)throws ServletExeption,IOException{}
(3)public void doPost(HttpServletRequest req,HttpServletResponse resp)throws ServletExeption,IOException{}
完成;
这里需要注意的是service的参数只是ServletRequest,而其他两个函数的参数是HttpServletRequest;
一般如果我们要自定义Servlet,则需要继承HttpServlet类,并覆盖相应的方法即可;
二、Servlet的结构
Servlet生命周期为:加载-->初始化--->服务--->销毁--->卸载;
加载:web容器加载Servlet,即创建一个Servlet实例;
初始化:调用servlet的init方法,为了完成一些预备动作;
当请求到来时,
服务:创建一个request、response对象,并创建一个线程,调用类service方法;
销毁:调用destroy()方法;当一个Servlet对象长时间不使用或web容器(tomcat)关闭时调用;
卸载:即退出;
继承HttpServlet后,可以覆写以下方法:
1.public void init(ServletConfig config)throws ServletException{} //初始化Servlet,(1)当需要使用Servlet时调用;(2)如果在web.xml中配置,则可以web容器启动时自动加载;配置如下:
<serlvet>
<serlvet-name></servlet-name>
<servlet-class></servlet-class>
<load-on-startup>1</load-on-startup>
</servlet>
2.public void init(ServletConfig config)throws ServletExeption{} //初始化Servlet,可以得到配置信息
3.public void doGet(HttpServletRequest req,HttpServletResponse resp)throws ServletExeption,IOException{} 当get方式传递,则调用此方法
4.public void doPost(HttpServletRequest req,HttpServletResponse resp)throws ServletExeption,IOException{}当post方法传递,则调用此方法
5.public void service(HttpServletRequest req,HttpServletResponse resp)throws ServletExeption,IOException{}
6.public void destroy(){} //销毁时调用
注意:
1. 当1,2同时出现时,2有较高优先级;
2. 当3或4和5同时出现时,5具有较高优先级;
3 .PrintWriter writer = resp.getWriter();可以获得输出流;
注意:writer输出时需要输出HTML结构;
三、Servlet常见问题
1.在init方法中,通过config.getInitParameter()方法取得配置信息;
2.req.getSession():取得Session对象;
3.super.getServletContext();取得application对象;因为GenericServlet中有getServletContext方法;
4.resp.getWriter()返回一个PrintWriter用以输出文本数据、resp.getOutputStream()输出二进制数据,并且两者不能同时调用;
5.在init(ServletConfig config)方法中需要调用super.init(config);
6. 在service()方法中getServletConfig()返回 ServletConfig;
四、Servlet跳转
1.客户端跳转:resp.sendRedirect("1.jsp") ; //类似于内置对象中的跳转;
2.服务器跳转:req.getRequestDispatcher("/hello.jsp").forward(req,resp);能够跳转到hello.jsp中;
注意:客户端跳转和服务器端跳转的区别;
注意:这里的getRequestDispatcher中的网页一定要加“ / ”
五、MVC设计模式
在之前我们讲过JSP+JAVABEAN的DAO开发模式,这个适用于小型开发;
MVC最早是由SmallTalk提出的;
Controller: Servlet 负责接收客户请求并转发给Model;
Model :JavaBean 负责真正处理业务逻辑;
View:JSP 负责输出结果;
EJB(Enterprise JavaBean);
MVC(Model View Control)是一种以Servlet为核心的开发模式,流程如下:
步骤如下:
1.客户端发送请求给Servlet;
2.Servlet接收请求后处理,并可以调用JavaBean(即进行数据库操作,并返回结果);
3.Servlet返回结果给JSP显示;(通过设置request属性后调用RequestDispatcher方法跳转,并在JSP页面中接收request属性);
因此JSP只是用于显示,而JavaBean只和Servlet通信;
注意:在MVC中,使用requestDispatcher的机会很多,我们都是通过这个类进行服务器跳转的;
六、MVC实例
功能:登录功能
1.JavaBean部分
User.java
package org.vo;
public class User{
private String id;
private String name;
private String password;
public String getId(){
return id;
}
public void setId(String id){
this.id = id;
}
public String getName(){
return name;
}
public void setName(String name){
this.name = name;
}
public void setPassword(String password){
this.password = password;
}
public String getPassword(){
return password;
}
}
IUserDAO.java
package org.dao;
import org.vo.*;
public interface IUserDAO{
public boolean findLogin(User user)throws Exception;
}
package org.dao.impl;
import java.sql.*;
import org.vo.*;
import org.dao.*;
public class UserDAOImpl implements IUserDAO{
private Connection con;
public UserDAOImpl(Connection con){
this.con = con;
}
public boolean findLogin(User user)throws Exception{
boolean flag = false;
String sql = "SELECT name FROM user WHERE id=? AND password=?";
PreparedStatement stat = con.prepareStatement(sql);
stat.setString(1,user.getId());
stat.setString(2,user.getPassword());
ResultSet rs = stat.executeQuery();
if(rs.next()){
user.setName(rs.getString(1));
flag = true;
}
return flag;
}
}
UserDAOProxy.java
package org.dao.proxy;
import org.dao.*;
import org.vo.*;
import org.dao.impl.*;
import org.dbc.*;
public class UserDAOProxy implements IUserDAO{
private DatabaseConnection dbc;
private IUserDAO idao;
public UserDAOProxy(){
dbc = new DatabaseConnection();
idao = new UserDAOImpl(dbc.getConnection());
}
public boolean findLogin(User user)throws Exception{
if(user==null){
return false;
}
boolean flag = idao.findLogin(user);
dbc.close();
return flag;
}
}
DatabaseConnection.java
package org.dbc;
import java.sql.*;
public class DatabaseConnection{
public static final String DRIVER = "com.mysql.jdbc.Driver";
public static final String URL = "jdbc:mysql://localhost:3306/mldn";
public static final String USER = "root";
public static final String PASS = "123456";
private Connection con;
public DatabaseConnection(){
try{
Class.forName(DRIVER);
con = DriverManager.getConnection(URL,USER,PASS);
}
catch(Exception e){}
}
public Connection getConnection(){
return con;
}
public void close(){
try{
if(con!=null){
con.close();
}
}
catch(Exception e){}
}
}
DAOFactory.java
package org.factory;
import org.dao.*;
import org.dao.proxy.*;
public class DAOFactory{
public static IUserDAO getInstance(){
return new UserDAOProxy();
}
}
以上代码是JavaBean部分;MVC的特点是用Servlet调用JavaBean,而不是JSP调用JavaBean;
2.Servlet部分
以下是Servlet部分:Servlet是接收客户端请求,并调用JavaBean进行数据库操作;并把结果通过设置request属性传给JSP进行显示;(贯穿核心)
LoginServlet.java
package org.servlet;
import java.util.*;
import javax.servlet.*;
import javax.servlet.http.*;
import org.factory.*;
import org.vo.*;
import java.io.*;
public class LoginServlet extends HttpServlet{
public void doGet(HttpServletRequest req,HttpServletResponse resp)throws ServletException,IOException{
String path = "login.jsp";
String id = req.getParameter("id");
String pass = req.getParameter("pass");
List<String> info = new ArrayList<String>();
if(id==null||"".equals(id)){
info.add("id不能为空");
}
if(pass==null||"".equals(pass)){
info.add("密码不能为空");
}
if(info.size()==0){
User user = new User();
user.setId(id);
user.setPassword(pass);
try{
if(DAOFactory.getInstance().findLogin(user)==true){
info.add("欢迎光临");
}
else{
info.add("错误的用户名和密码");
}
}
catch(Exception e){}
}
req.setAttribute("info",info);
req.getRequestDispatcher(path).forward(req,resp);
}
public void doPost(HttpServletRequest req,HttpServletResponse resp)throws ServletException,IOException{
this.doGet(req,resp);
}
}
Login.jsp
<%@page contentType="text/html" pageEncoding="GBK" import="java.util.*"%>
<html>
<head>
<title>A</title>
</head>
<script language="Javascript">
function validate(f){
if(!(/^\w{5,15}$/.test(f.id.value))){
alert("id长度不对");
f.id.focus();
return false;
}
if(!(/^\w{5,15}$/.test(f.pass.value))){
alert("密码长度不对");
f.pass.focus();
return false;
}
return true;
}
</script>
<body>
<h2>用户登录程序</h2>
<%
request.setCharacterEncoding("GBK");
List<String> info = (List<String>)request.getAttribute("info");
if(info!=null){
Iterator<String>iter = info.iterator();
while(iter.hasNext()){
%>
<h4><%=iter.next()%></h4>
<%
}
}
%>
<form action="LoginServlet" method="post" onSubmit="return validate(this)">
用户ID:<input type="text" name="id"/><br />
密码:<input type="password" name="pass"/><br />
<input type="submit" value="提交"/>
</form>
</body>
</html>
数据库脚本:
CREATE TABLE user(
id varchar(30) PRIMARY KEY,
name varchar(30) ,
password varchar(30)
);
INSERT INTO user VALUES('XIAZDONG','xiazdong','12345');
因此最后再强调一下Mode1和Mode2的区别:
Mode1是通过JSP调用JavaBean;Mode2是通过Servlet调用JavaBean;
Mode1在JSP中仍然有处理的部分,而Mode2中JSP只负责显示;
补充:Servlet调用过程顺序图
ServletContext介绍
<context-param>
<param-name></param-name>
<param-value></param-value>
</context-param>
配置;
context.getRequestDispatcher("/1.html").forward(request,response);
InputStream in = context.getResourceAsStream("*.properties");
Properties props = new Properties();
props.load(in);
补充:web开发中地址书写问题
地址书写时常用到“/”开头,而“/”有两个宗旨:
1.面向浏览器,则"/"表示http://localhost:8080/
2.面向服务器,则"/"表示当前web应用;
面向浏览器的意思是:是否会让浏览器地址栏改变;其余的都是面向服务器;
2.response.sendRedirect("地址B");
补充:在提供客户发出请求之前的过程
1.从web.xml中读出Servlet初始化参数和上下文初始化参数;
2.创建一个ServletConfig对象和ServletContext对象;
3.将Servlet初始化参数的引用赋给ServletConfig对象,把上下文初始化参数赋值给ServletContext对象;
4.创建ServletContextListener监听器实例;
5.调用contextInitialized()方法;
6.创建这个Servlet实例;
7.调用init方法;
注意:
(1)在Servlet构造函数中还没有ServletConfig对象,虽然能够调用getServletConfig()方法;
(2)ServletConfig对象在Servlet实例创建之前就已经创建;
(3)web.xml的初始化参数只会读一次,如果需要更新,则需要重新部署;
上下文初始化参数:整个web应用都能够访问的初始化参数;
<context-param>
<param-name>name</param-name>
<param-value>value</param-value>
</context-param>
getServletContext().getInitParameter("name")即可;
每个Servlet有一个ServletConfig,每个web应用有一个ServletContext;
如果web应用时分布式的,则每个JVM都有一个ServletContext;
注意:getAttribute()返回的是Object;
Person p = getServletContext().getAttribute("person");是错误的!!!!!!!!!!!!!!!!!!!!!
Person p = (Person)getServletContext().getAttribute("person");是正确的;