编码问题的解决
请求参数编码问题
解决post方式编码
request.setCharacterEncoding("UTF-8"); //告诉服务器客户端什么编码,只能处理post请求方式
解决get方式编码
String name = new String(name.getBytes(“iso-8859-1”),”UTF-8”);
响应数据编码问题
服务器发送响应数据到浏览器过程编码和解码的过程如下
由于服务器中的编方案和浏览器的解码方案不统一,所以产生了乱码。所以解决方法就是要统一编码和解码的方案。
代码一
//服务器中默认的编码是ISO-8859-1,它是不支持中文的,tomcat规定的。
//告诉服务器应该使用UTF-8解析文本
response.setCharacterEncoding("UTF-8");
//告诉客户端要使用什么编码
response.setHeader("content-type", "text/html;charset=UTF-8");
代码二
//告诉服务器应该使用UTF-8解析文本;告诉客户端要使用什么编码
response.setContentType("text/html;charset=UTF-8");
Http响应
文件下载
//通过路径得到一个输入流
String path = this.getServletContext().getRealPath("/WEB-INF/classes/美女.jpg");
FileInputStream fis = new FileInputStream(path);
//创建字节输出流
ServletOutputStream sos = response.getOutputStream();
//得到要下载的文件名
String filename = path.substring(path.lastIndexOf("\\")+1);
//设置文件名的编码
filename = URLEncoder.encode(filename,"UTF-8");
//告知客户端要下载文件
response.setHeader("content-disposition", "attachment;filename="+filename);
response.setHeader("content-type", "image/jpeg");
//执行输出操作
int len = 1;
byte[] b = new byte[2048];
while((len = fis.read(b))!=-1){
sos.write(b,0,len);
}
sos.close();
fis.close();
验证码
1.自己实现
//在内存中创建一个图像对象
int width = 110;//图像宽
int height = 25;
BufferedImage img = new BufferedImage(width,height,BufferedImage.TYPE_INT_RGB);
//创建一个画笔
Graphics g = img.getGraphics();
//给边框一个色
g.setColor(Color.RED);
g.drawRect(0, 0, width-1, height-1);
//给图片添加背景色
g.setColor(Color.PINK);//设置一个颜色
g.fillRect(1, 1, width-2, height-2);//填充颜色
//设置文本样式
g.setColor(Color.BLUE);
g.setFont(new Font("宋体",Font.BOLD|Font.ITALIC,15));
//给图片添加文本
Random rand = new Random();
int position = 20;
for(int i = 0; i < 4; i++){
g.drawString(rand.nextInt(10)+"",position,20);//给图片填充文本
position += 20;
}
//添加九条干扰线
for(int i = 0; i < 9; i++){
g.drawLine(rand.nextInt(width), rand.nextInt(height), rand.nextInt(width), rand.nextInt(height));
}
//将图片对象以流的方式输出到客户端
ImageIO.write(img, "jpg", response.getOutputStream());
2.使用工具类实现
//告诉客户端不使用缓存
response.setHeader("pragma", "no-cache");
response.setHeader("cache-control", "no-cache");
response.setIntHeader("expires", 0);
ValidateCode vc = new ValidateCode(110, 25, 4, 9);//(验证码宽度,高度,字符个数,干扰线个数)
vc.write(response.getOutputStream());
3.login.html代码
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>Insert title here</title>
<script type="text/javascript">
function changeCode(){
var img = document.getElementsByTagName("img")[0];
//img.src = "/day09_00_HttpServletResponse/ServletDemo4";
img.src = "/day09_00_HttpServletResponse/ServletDemo4?time ="+new Date().getTime()
}
</script>
</head>
<body>
<form action="#" method = "post">
用户名 :<input type = "text" name = "userName"/><br/>
密码: <input type = "password" name = "pwd"/><br/>
验证码:<input type = "text" name = "code"/>
<img src = "/day09_00_HttpServletResponse/ServletDemo4"/><a href = "javascript:changeCode()">看不清楚换一张</a>
</form>
</body>
</html>
定时刷新功能
//告诉服务器应该使用UTF-8解析文本;告诉客户端要使用什么编码`
response.setContentType("text/html;charset=UTF-8");
// response.setIntHeader("refresh", 1);//设置1秒钟刷新一次
// Random r = new Random();
// response.getWriter().write(r.nextInt()+"");
response.getWriter().write("注册成功!三秒钟转到主页");
response.setHeader("refresh", "3;url =/day09_00_HttpServletResponse/ServletDemo4");
2.Http请求
2.1 请求行
主要方法:
getMethod(); 获得请求方式
getRequestURL();返回客户端发出请求时的完整URL。
getRequestURI(); 返回请求行中的资源名部分。
getContextPath(); 当前应用的虚拟目录 /day09_01_request
getQueryString() ; 返回请求行中的参数部分。
//访问地址为:http://localhost:8080/day09_01_HttpServletRequest/ServletDemo1?username=tom
public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
// getMethod(); 获得请求方式
// ***getRequestURL();返回客户端发出请求时的完整URL。
// ***getRequestURI(); 返回请求行中的资源名部分。
// *****getContextPath(); 当前应用的虚拟目录 /day09_01_HttpServletRequest
// getQueryString() ; 返回请求行中的参数部分。
System.out.println(request.getMethod());// GET
System.out.println(request.getRequestURL());// http://localhost:8080/day09_01_HttpServletRequest/ServletDemo1
System.out.println(request.getRequestURI());// /day09_01_HttpServletRequest/ServletDemo1
System.out.println(request.getContextPath());// /day09_01_HttpServletRequest
System.out.println(request.getQueryString());// username=tom
}
2.2 请求头
2.2.1 请求头内容
Accept: text/html,image/* -- 浏览器接受的数据类型
Accept-Charset: ISO-8859-1 -- 浏览器接受的编码格式
Accept-Encoding: gzip,compress --浏览器接受的数据压缩格式
Accept-Language: en-us,zh- --浏览器接受的语言
Host: www.it315.org:80 --(必须的)当前请求访问的目标地址(主机:端口)
If-Modified-Since: Tue, 11 Jul 2000 18:23:51 GMT --浏览器最后的缓存时间
Referer: http://www.it315.org/index.jsp -- 当前请求来自于哪里
User-Agent: Mozilla/4.0 (compatible; MSIE 5.5; Windows NT 5.0) --浏览器类型
Cookie:name=eric -- 浏览器保存的cookie信息
Connection: close/Keep-Alive -- 浏览器跟服务器连接状态。close: 连接关闭 keep-alive:保存连接。
Date: Tue, 11 Jul 2000 18:23:51 GMT -- 请求发出的时间
2.2.2 主要方法
String getHeader(String name) //根据头名称得到头信息值
Enumeration getHeaderNames() //得到所有头信息name
Enumeration getHeaders(String name) //根据头名称得到相同名称头信息值
2.2.3 代码
(1)获取浏览器类型
//获得请求消息头的信息
//获取浏览器类型
String header = request.getHeader("User-Agent");
System.out.println(header);
if(header.toLowerCase().contains("msie")){
System.out.println("IE浏览器");
}else if(header.toLowerCase().contains("firefox")){
System.out.println("火狐浏览器");
}else if(header.toLowerCase().contains("chrome")){
System.out.println("谷歌浏览器");
}else{
System.out.println("其他浏览器");
}
}
(2)获取所有请求消息头的name
//获取所有请求消息头的name
Enumeration names = request.getHeaderNames();
while(names.hasMoreElements()){
String e = (String)names.nextElement();
System.out.println(e+":"+request.getHeader(e));
}
(3)获得相关name的消息头
//获得相关name的消息头
Enumeration headers = request.getHeaders("accept-language");
while(headers.hasMoreElements()){
String e = (String)headers.nextElement();
System.out.println(e+":"+request.getHeader(e));
System.out.println(e);
2.3 请求内容
2.3.1 与获取表单数据相关的方法
<input type="text" name="username" />
getParameter(name) 根据表单中name属性的名,获取value属性的值方法
getParameterValues(String name)专业为复选框取取提供的方法
getParameterNames() 得到表单提交的所有name的方法
getParameterMap 到表单提交的所有值的方法 //做框架用,非常实用
getInputStream 以字节流的方式得到所有表单数据
private void test1(HttpServletRequest request)
throws UnsupportedEncodingException {
//doGet(request, response);
//告诉服务器要使用什么编码,注:浏览器使用的是什么编码传过来的就是什么编码
request.setCharacterEncoding("UTF-8");
//根据表单中name属性获取value属性的值
String userName = request.getParameter("userName");
String pwd = request.getParameter("pwd");
String sex = request.getParameter("sex");
String []hobbys = request.getParameterValues("hobby");
String city = request.getParameter("city");
System.out.println(userName);
System.out.println(pwd);
System.out.println(sex);
for(int i = 0; hobbys != null && i < hobbys.length; i++){
System.out.println(hobbys[i]+"\t");
}
System.out.println();
System.out.println(city);
}
private void test2(HttpServletRequest request)
throws UnsupportedEncodingException {
//获取所有表单name
request.setCharacterEncoding("UTF-8");
Enumeration names = request.getParameterNames();
while(names.hasMoreElements()){
String name = (String)names.nextElement();//得到每一个name名
String [] values = request.getParameterValues(name);//根据name得到value值
for(String value:values){
System.out.println(name+"\t"+value);
}
}
}
private void test3(HttpServletRequest request) {
try {
User u = new User();
System.out.println("封装之前"+u);
//获取表单数据
Map<String,String[]> map = request.getParameterMap();
for(Map.Entry<String, String[]> m:map.entrySet()){
String name = m.getKey();
String[] value = m.getValue();
//创建一个属性描述器
PropertyDescriptor pd = new PropertyDescriptor(name, User.class);//根据name在User类中找get set方法
//得到setter属性
Method setter = pd.getWriteMethod();
if(value.length == 1){
setter.invoke(u, value[0]);//给一个值的变量赋值
}else{
setter.invoke(u, (Object)value);//相当于给复选框赋值
}
}
System.out.println("封装数据后:"+u);
} catch (Exception e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
//使用了框架
private void test4(HttpServletRequest request) {
try {
User u = new User();
System.out.println("封装之前"+u);
BeanUtils.populate(u, request.getParameterMap());
System.out.println("封装数据后:"+u);
} catch (IllegalAccessException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (InvocationTargetException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
2.3.2 获取非表单数据相关的方法
void setAttribute(String name, Object value);
Object getAttribute(String name);
void removeAttribute(String name);
3 转发和重定向
3.1 请求转发
请求转发:Servlet(源组件)先对客户请求做一些预处理操作,然后把请求转发给其他Web组件(目标组件)来完成包括生成响应结果在内的后续操作;
转发示意图:
转发 request 由服务器执行;客户端只发送一次请求;地址栏不变;转发能够传递数据;转发也是一个域对象;请求转发不能跳转到其他的应用;
servletContxt
转发代码:
request.getRequestDispatcher("/ServletDemo6").forward(request, response);
请求包含
请求包含:Servlet(源组件)把其他Web组件(目标组件)生成的响应结果包含到自身的响应结果中;
request.getRequestDispatcher("/ServletDemo6").include(request, response);
3.3 请求重定向
请求重定向过程如下:
重定向 response 由客户端执行;客户端发送两次请求;地址栏会发生变化;不能传递数据;可以跳转到其它应用
ServletDemo7代码:
public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
System.out.println("A:我要借钱!");
System.out.println("B:我不有,但是我可以告诉你谁有");
// //告诉客户端要重新定向的资源
// response.setStatus(302);
// //告诉浏览器要去访问哪个URL
// response.setHeader("location", "/day09_00_HttpServletResponse/ServletDemo8");
// response.sendRedirect("/day09_00_HttpServletResponse/ServletDemo8");
response.sendRedirect(request.getContextPath()+"/ServletDemo8");
System.out.println("A:我去了");
}
ServletDemo8代码:
public void doGet(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
System.out.println("我有钱,我借你!");
}
结果如下: