java第三十天(ServletContext、HTTPServletReaues、HTTPServletResponse、中文乱码问题)

Servlet配置方式

1. 全路径匹配

以 / 开始 /a /aa/bb

localhost:8080/项目名称/aa/bb

2. 路径匹配 , 前半段匹配

以 / 开始 , 但是以 * 结束 /a/* /*

  • 其实是一个通配符,匹配任意文字

localhost:8080/项目名称/aa/bb

3. 以扩展名匹配

写法: 没有/ 以 * 开始 *.扩展名 *.aa *.bb

ServletContext

Servlet 上下文

每个web工程都只有一个ServletContext对象。 说白了也就是不管在哪个servlet里面,获取到的这个类的对象都是同一个。

//1. 获取对象
	ServletContext context = getServletContext();

作用

  1. 获取全局配置参数
  2. 获取web工程中的资源
  3. 存取数据,servlet间共享数据 域对象

可以获取全局配置参数

获取全局参数

可以获取Web应用中的资源

1. 获取资源在tomcat里面的绝对路径

	先得到路径,然后自己new InpuStream

		context.getRealPath("") //这里得到的是项目在tomcat里面的根目录。

		D:\tomcat\apache-tomcat-7.0.52\apache-tomcat-7.0.52\wtpwebapps\Demo03\
	
	 	String path = context.getRealPath("file/config.properties");

		D:\tomcat\apache-tomcat-7.0.52\apache-tomcat-7.0.52\wtpwebapps\Demo03\file\config.properties


2. getResourceAsStream 获取资源 流对象

	直接给相对的路径,然后获取流对象。

通过classloader去获取web工程下的资源

使用ServletContext存取数据。

  1. 定义一个登陆的html页面, 定义一个form表单

  1. 定义一个Servlet,名为LoginServlet
if (load==true) {
	int count=0;
	Object totalcount = getServletContext().getAttribute("count");
	if (totalcount!=null) {
		count=(int) totalcount;
	}
	getServletContext().setAttribute("count", count+1);
	System.out.println("登陆成功的次数是"+count+"次");
	//设置状态码重新定位
	response.setStatus(302);
	response.setHeader("Location", "load.html");//此处为相对路径
}
if (load==false) {
	response.getWriter().write("登陆失败...");
}
  1. 针对成功或者失败,进行判断,然后跳转到不一样的网页

细节:

	<!-- 	
	A路径: Servlet的路径
		http://localhost:8080/Demo4/login
	
	B路径: 当前这个html的路径:
		http://localhost:8080/Demo4/login.html -->
	
	
	<form action="Login" method="get">
		账号:<input type="text" name="username"/><br>
		密码:<input type="text" name="password"/><br>
		<input type="submit" value="登录"/>
	</form>

ServletContext 生命周期

服务器启动的时候,会为托管的每一个web应用程序,创建一个ServletContext对象

从服务器移除托管,或者是关闭服务器。

  • ServletContext 的作用范围

只要在这个项目里面,都可以取。 只要同一个项目。 A项目 存, 在B项目取,是取不到的? ServletContext对象不同。

HttpServletRequest

这个对象封装了客户端提交过来的一切数据。

1. 可以获取客户端请求头信息

	//得到一个枚举集合  
	Enumeration<String> headerNames = request.getHeaderNames();
	while (headerNames.hasMoreElements()) {
		String name = (String) headerNames.nextElement();
		String value = request.getHeader(name);
		System.out.println(name+"="+value);
		
	}

2. 获取客户端提交过来的数据

	String name = request.getParameter("name");
	String address = request.getParameter("address");
	System.out.println("name="+name);
	System.out.println("address="+address);

	-------------------------------------------------

	//name=zhangsan&name=lisi&name=wangwu 一个key可以对应多个值。

	Map<String, String[]> map = request.getParameterMap();
	
	Set<String> keySet = map.keySet();
	Iterator<String> iterator = keySet.iterator();
	while (iterator.hasNext()) {
		String key = (String) iterator.next();
		System.out.println("key="+key + "--的值总数有:"+map.get(key).length);
		String value = map.get(key)[0];
		String value1 = map.get(key)[1];
		String value2 = map.get(key)[2];
		
		System.out.println(key+" ======= "+ value + "=" + value1 + "="+ value2);
	}

3. 获取中文数据

客户端提交数据给服务器端,如果数据中带有中文的话,有可能会出现乱码情况,那么可以参照以下方法解决。

* 如果是GET方式

1. 代码转码
		String username = request.getParameter("username");
		String password = request.getParameter("password");
		
		System.out.println("userName="+username+"==password="+password);
		
		//get请求过来的数据,在url地址栏上就已经经过编码了,所以我们取到的就是乱码,
		//tomcat收到了这批数据,getParameter 默认使用ISO-8859-1去解码
		
		//先让文字回到ISO-8859-1对应的字节数组 , 然后再按utf-8组拼字符串
		username = new String(username.getBytes("ISO-8859-1") , "UTF-8");
		System.out.println("userName="+username+"==password="+password);
	
		直接在tomcat里面做配置,以后get请求过来的数据永远都是用UTF-8编码。 


2. 可以在tomcat里面做设置处理 conf/server.xml 加上URIEncoding="utf-8"

	   <Connector connectionTimeout="20000" port="8080" protocol="HTTP/1.1" redirectPort="8443" URIEncoding="UTF-8"/>

* 如果是POST方式

	这个说的是设置请求体里面的文字编码。  get方式,用这行,有用吗? ---> 没用
	request.setCharacterEncoding("UTF-8");

	这行设置一定要写在getParameter之前。

HttpServletResponse

负责返回数据给客户端。

* 输出数据到页面上

	//以字符流的方式写数据	
	//response.getWriter().write("<h1>hello response...</h1>");
	
	//以字节流的方式写数据 
	response.getOutputStream().write("hello response2222...".getBytes());

响应的数据中有中文,那么有可能出现中文乱码

* 以字符流输出

response.getWriter()

	//1. 指定输出到客户端的时候,这些文字使用UTF-8编码
	response.setCharacterEncoding("UTF-8");
	
	//2. 直接规定浏览器看这份数据的时候,使用什么编码来看。
	response.setHeader("Content-Type", "text/html; charset=UTF-8");
	
	response.getWriter().write("我爱黑马训练营...");

* 以字节流输出

response.getOutputStream()

	//1. 指定浏览器看这份数据使用的码表
	response.setHeader("Content-Type", "text/html;charset=UTF-8");
	
	//2. 指定输出的中文用的码表
	response.getOutputStream().write("我爱深圳黑马训练营..".getBytes("UTF-8"));


	--------------------------------------------

不管是字节流还是字符流,直接使用一行代码就可以了。

response.setContentType("text/html;charset=UTF-8");

然后在写数据即可。

演练下载资源。

1. 直接以超链接的方式下载,不写任何代码。 也能够下载东西下来。

	让tomcat的默认servlet去提供下载:<br>
	<a href="download/aa.jpg">aa.jpg</a><br>
	<a href="download/bb.txt">bb.txt</a><br>
	<a href="download/cc.rar">cc.rar</a><br>

原因是tomcat里面有一个默认的Servlet -- DefaultServlet 。这个DefaultServlet 专门用于处理放在tomcat服务器上的静态资源。

2. 手写代码提供下载

html中代码

<h3>手写代码提供下载</h3>
<a href="Load?filename=aa.txt">aa.txt</a><br/>
<a href="Load?filename=bb.jpg">bb.jpg</a><br/>
<a href="Load?filename=cc.zip">cc.zip</a><br/>

创建一个servlet命名为Load,代码

import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;

import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

/**
 * Servlet implementation class Load
 */
public class Load extends HttpServlet {
	
	@SuppressWarnings("resource")
	protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
		//获得要下载的文件名
		String filename = request.getParameter("filename");
		//获得这个文件在tomcat中的绝对路径
		String realPath = getServletContext().getRealPath("download/"+filename);
		//让浏览器收到这个资源的时候,以加载的方式提醒用户而不是直接展示
		response.setHeader("Content-Disposition", "attachment;filename="+filename);
		
		response.setContentType("text/html;charset=UTF-8");
		
		InputStream fileInputStream = new FileInputStream(realPath);
		OutputStream fileOutputStream=response.getOutputStream();
		byte[] bt=new byte[1024];
		int len=0;
		while ((len=fileInputStream.read(bt)) > -1) {
			fileOutputStream.write(bt,0,len);			
		}
		fileInputStream.close();
		fileOutputStream.close();
		
	}

	/**
	 * @see HttpServlet#doPost(HttpServletRequest request, HttpServletResponse response)
	 */
	protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
		// TODO Auto-generated method stub
		doGet(request, response);
	}

}

总结

  1. Servlet注册方式

  2. ServletContext【重点】

     作用:
    
     	1. 获取全局参数
    
     	2. 获取工程里面的资源。
    
     	3. 资源共享。  ServletContext 域对象
    
     有几个 一个 
    
     什么时候创建 ? 什么时候销毁
    
     服务器启动的时候给每一个应用都创建一个ServletContext对象, 服务器关闭的时候销毁
    

    简单登录

  3. HttpServletRequest【重点】

     1. 获取请求头
    
     2. 获取提交过来的数据
    
  4. HttpServletResponse【重点】

     负责输出数据到客户端,其实就是对之前的请求作出响应
    
  5. 中文乱码问题。【重点】

  6. 下载

作业:

1. 完成注册 

2. 完成登录


V1.1 最好配合上数据库,完成注册和登录的功能。

作业步骤分析:

先创建一个数据库

创建表格存储用户信息,可以规定字符格式之类的,配置JDBC驱动,拷贝JDBC驱动的jar包放在WebContent下的我web-inf下的lib里面右键build path

分析:

第一步需要一个注册界面
<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title>Insert title here</title>
</head>
<body>
<!-- 这里action="Login",对应的是servlet配置中的url-pattern -->
	<form action="Login" method="get">
	姓名:<input type="text" name="username">
	密码:<input type="password" name="password">
	<input type="submit" value="注册">
	</form>
</body>
</html>
第二创建一个servlet命名为Login,当注册内容不为空的时候将数据存储到数据库的表格中,如果存储成功跳转登陆界面,失败的话直接输出注册失败;
import java.io.IOException;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import s20190712.Dao.impl.UserImpl;


/**
 * Servlet implementation class Login
 */
public class Login extends HttpServlet {
	
	protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
		//获得客户端的信息
		String username= request.getParameter("username");
		String password= request.getParameter("password");
		
		username=new String(username.getBytes("ISO-8859-1"),"UTF-8");
		password=new String(password.getBytes("ISO-8859-1"),"UTF-8");
		
		System.out.println("用户名:"+username+"密码:"+password);
		
		response.setContentType("text/html;charset=UTF-8");
		//校验
	    if (!username.isEmpty()||!password.isEmpty()) {	
	    	UserImpl user=new UserImpl();
	    	int insert = user.insert(username, password);	
	    	if(insert>0) {
	    		//设置状态码重新定位
		    	response.setStatus(302);
		    	response.setHeader("Location", "config.html");//此处为相对路径	
	    	}    		
			
		}else {
			System.out.println("注册失败");
			response.getWriter().write("注册失败...");
		}
	}

	/**
	 * @see HttpServlet#doPost(HttpServletRequest request, HttpServletResponse response)
	 */
	protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
		// TODO Auto-generated method stub
		doGet(request, response);
	}

}
第三需要一个登陆界面;
<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title>Insert title here</title>
</head>
<body>
	<h1>您已经成功注册!</h1>
	<form action="Config" method="post">
		姓名:<input type="text" name="username">
		密码:<input type="password" name="password">
		<input type="submit" value="登陆">
	</form>
</body>
</html>
第四创建一个servlet命名为Config,登陆信息不为空的时候,将信息在数据库中按条件查询,可以匹配上跳转登陆成功,失败直接跳转登陆失败;
import java.io.IOException;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import s20190712.Dao.impl.UserImpl;


/**
 * Servlet implementation class Config
 */
public class Config extends HttpServlet {
	
	protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
			
		//获得客户端的信息
		        request.setCharacterEncoding("utf-8");
				String username= request.getParameter("username");
				String password= request.getParameter("password");
				
				response.setContentType("text/html;charset=UTF-8");
				//校验
			    if (!username.isEmpty()||!password.isEmpty()) {	
			    	UserImpl user=new UserImpl();
			    	boolean load = user.load(username, password);
			    	if (load==true) {
				    	int count=0;
						Object totalcount = getServletContext().getAttribute("count");
						
						if (totalcount!=null) {
							count=(int) totalcount;
						}
						getServletContext().setAttribute("count", count+1);
						System.out.println("登陆成功的次数是"+count+"次");
				    	//设置状态码重新定位
				    	response.setStatus(302);
				    	response.setHeader("Location", "load.html");//此处为相对路径		
					}
					if (load==false) {
						response.getWriter().write("登陆失败...");
					}
					
				}else {

					response.getWriter().write("请返回页面输入登陆信息");
				}
	}

	/**
	 * @see HttpServlet#doPost(HttpServletRequest request, HttpServletResponse response)
	 */
	protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
		// TODO Auto-generated method stub
		doGet(request, response);
	}

}

第五需要一个登陆成功界面,显示下载资源链接;
<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title>Insert title here</title>
</head>
<body>
<h3>这里有资源可以下载tomcat默认的servlet去提供下载</h3>
<a href="download/aa.txt">aa.txt</a><br/>
<a href="download/bb.jpg">bb.jpg</a><br/>
<a href="download/cc.zip">cc.zip</a><br/>

<h3>手写代码提供下载</h3>
<a href="Load?filename=aa.txt">aa.txt</a><br/>
<a href="Load?filename=bb.jpg">bb.jpg</a><br/>
<a href="Load?filename=cc.zip">cc.zip</a><br/>
</body>
</html>
第六创建一个servlet命名为Load,获取tomcat中提供下载的资源以字节流的方式读取并且读出,注意中文乱码的问题
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;

import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

/**
 * Servlet implementation class Load
 */
public class Load extends HttpServlet {
	
	@SuppressWarnings("resource")
	protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
		//获得要下载的文件名
		String filename = request.getParameter("filename");
		filename=new String(filename.getBytes("ISO-8859-1"),"UTF-8");//注意这里如果文件名是中文需要对其进行编码处理
		//获得这个文件在tomcat中的绝对路径
		String realPath = getServletContext().getRealPath("download/"+filename);
		//让浏览器收到这个资源的时候,以加载的方式提醒用户而不是直接展示
		response.setHeader("Content-Disposition", "attachment;filename="+filename);
		
		response.setContentType("text/html;charset=UTF-8");
		
		InputStream fileInputStream = new FileInputStream(realPath);
		OutputStream fileOutputStream=response.getOutputStream();
		byte[] bt=new byte[1024];
		int len=0;
		while ((len=fileInputStream.read(bt)) > -1) {
			fileOutputStream.write(bt,0,len);			
		}
		fileInputStream.close();
		fileOutputStream.close();
		
	}

	/**
	 * @see HttpServlet#doPost(HttpServletRequest request, HttpServletResponse response)
	 */
	protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
		// TODO Auto-generated method stub
		doGet(request, response);
	}

}

第七完善数据库连接和注册插入数据、登陆查询数据的方法

在src下创建file,JDBC.properties
url=jdbc:mysql://localhost:3306/user
username=root
password=321
driverClass=com.mysql.jdbc.Driver
工具类

注意如果出现报错JDBC驱动找不到的情况,两种方法,第一种在Java\jre1.8.0_151\lib\ext中配置JDBC驱动,就不必在工程目录里面配置了;第二种,注册一下驱动,DriverManager或者Class.forName()都可以

import java.io.InputStream;
import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Statement;
import java.util.Properties;

import com.mysql.jdbc.Driver;


public class JDBCUtil {
	
	static String driverClass =null;
	static String url =null;
	static String username =null;
	static String password =null;
	
	static {
		try {
			Properties prop=new Properties();
			InputStream is=JDBCUtil.class.getClassLoader().getResourceAsStream("JDBC.properties");
			prop.load(is);
			driverClass = prop.getProperty("driverClass");
			url = prop.getProperty("url");
			username = prop.getProperty("username");
			password = prop.getProperty("password");
		} catch (Exception e) {
			e.printStackTrace();
		}
	}
		   public static Connection getConn() {
			   Connection conn=null;
				try {
					//Class.forName(driverClass);
					conn = DriverManager.getConnection(url,username,password);
				} catch (Exception e) {
					e.printStackTrace();
				}
				return conn;
			}
    	public static void release(Connection conn,Statement st,ResultSet rs) {
			closeConn(conn);
			closeRs(rs);
			closeSt(st);
		}	
    	public static void release(Connection conn,Statement st) {
			closeConn(conn);
			closeSt(st);
		}	
	private static void closeConn(Connection conn) {	
			try {
				if (conn!=null) {
					conn.close();
				}
				
			} catch (SQLException e) {
				
				e.printStackTrace();
			}finally{
				conn=null;
			}	
		
	}
	
	private static void closeSt(Statement st) {	
		try {
			if (st!=null) {
				st.close();
			}
			
		} catch (SQLException e) {
			
			e.printStackTrace();
		}finally{
			st=null;
		}	
	
	}
	
	private static void closeRs(ResultSet rs) {	
		try {
			if (rs!=null) {
				rs.close();
			}
			
		} catch (SQLException e) {
			
			e.printStackTrace();
		}finally{
			rs=null;
		}	
	
	}
}
接口和实现类不在一个包中,有返回值
public interface UserDao {
	int insert(String username, String password);
	
	boolean load(String username, String password);
}

import java.sql.Connection;
import java.sql.PreparedStatement;
import java.sql.ResultSet;

import com.mysql.jdbc.Statement;

import s20190712.jdbc.dao.UserDao;
import s20190712.jdbc.util.JDBCUtil;

public class UserImpl implements UserDao{

	@Override
	public int insert(String username, String password) {
			Connection conn=null;
			PreparedStatement ps=null;
			int rs =0;
			try {
				//1.创建连接对象
				conn=JDBCUtil.getConn();
				
				//2.创建preparestatement对象,预处理sql语句
				String sql="insert into m_user values(null,?,?)";
				ps = conn.prepareStatement(sql);//注意导包import java.sql.PreparedStatement;
				//3.给占位符赋值
				ps.setString(1,username);
				ps.setString(2,password);
				//4.执行sql语句
				rs = ps.executeUpdate();
				if(rs>0){
					System.out.println("更新成功");
				}else{
					System.out.println("更新失败");
				}
			
			} catch (Exception e) {
				e.printStackTrace();
			}finally {
				//5.释放资源
				JDBCUtil.release(conn,ps);	
				return rs;
			}					
		}
	@Override
	public boolean load(String username, String password) {
		boolean b=false;
		Connection conn=null;
		Statement st=null;
		ResultSet rs=null;
		try {
			//1.创建连接对象
			conn=JDBCUtil.getConn();
			
			//2.创建preparestatement对象,预处理sql语句
			String sql="select * from m_user where username = ? and password = ?";
			PreparedStatement ps = conn.prepareStatement(sql);//注意导包import java.sql.PreparedStatement;
			//3.给占位符赋值
			ps.setString(1,username);
			ps.setString(2,password);
			//4.执行sql语句
			rs = ps.executeQuery();
			if(rs.next()){
				b=true;
			}else{
				b=false;
			}
		
		} catch (Exception e) {
			e.printStackTrace();
		}finally {
			//5.释放资源
			JDBCUtil.release(conn,st,rs);
			return b;
		}		
	}
}

转载于:https://my.oschina.net/u/4110331/blog/3074145

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值