HTTP和Servlet
联系:
- HTTP是一个通信协议,而Servlet是服务器端程序,用于处理HTTP请求。
- Servlet通常用于处理HTTP请求,在服务器上生成动态内容,并生成HTTP响应。HTTP协议就是Servlet处理的基础。
区别:
- HTTP是一种协议,Servlet是一种编程模型。HTTP定义了客户端和服务器之间的通信规则,而Servlet是一种特定的编程模型,用于处理HTTP请求和生成HTTP响应。
- HTTP是跨平台的,而Servlet是Java语言特定的。HTTP协议可以在任何支持该协议的平台上使用,而Servlet是使用Java语言编写的,需要在Java平台上运行。
- HTTP更关注网络通信方面的内容,如请求方法、状态码、头部信息等。而Servlet更关注服务器端的业务逻辑,如处理请求参数、生成动态内容等。
HTTP
HTTP请求
Get请求(没有请求体)
Post请求
格式:
请求行
请求头1
请求头2
……
请求空行
请求体
请求行以一个方法符号开头,空格分开,后面跟着请求的UPI和协议的版本
HTTP响应
在接收和解释请求信息后,服务器返回一个HTTP响应信息,HTTP响应由三个部分组成:状态行,消息报头,响应正文
请求头
Referer:
- 该请求头指明请求从哪里来
响应头
Location:
- Location响应报头域用于重定向接受者到一个新的位置
- Location响应报头域,常用于更换域名的时候
Refresh:
- 自动跳转(单位是秒),可以在页面透过meta标签实现,也可以在后台实现
Servlet
Servlet的实现
//@WebServlet("/ser01")
//@WebServlet(name="Servlet",value = "/ser01")
//@WebServlet(name="Servlet",value = {"/ser01","ser001"})
//@WebServlet(name="Servlet",urlPatterns = "/ser01")
@WebServlet(name="Servlet",urlPatterns = {"/ser01","/ser001"})
public class Servlet01 extends HttpServlet {
@Override
protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//super.service(req, resp);
//打印内容
System.out.println("Hello");
resp.getWriter().write("Hello Servlet");
}
}
@WebServlet("/ser02")
public class Servlet02 extends GenericServlet {
@Override
public void service(ServletRequest servletRequest, ServletResponse servletResponse) throws ServletException, IOException {
System.out.println("继承GenericServlet类");
}
}
@WebServlet("/ser05")
public class Servlet05 extends HttpServlet {
/**
*就绪/服务方法(处理请求数据)
* 系统方法,服务器自动调用
* 当请求到达Servlet时,就会调用该方法
* 方法可以被调用多次
* @param req
* @param resp
*/
@Override
protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//super.service(req, resp);
//打印内容
System.out.println("Hello");
resp.getWriter().write("Hello Servlet");
}
/**
* 销毁方法
* 系统方法,服务器自动调用
* 当服务器关闭或应用程序停止时,调用该方法
* 只调用一次
*/
@Override
public void destroy() {
System.out.println("销毁");
super.destroy();
}
/**
* 初始化方法
* 系统方法,服务器自动调用
* 当请求到达Servlet容器时,Servlet同期会判断Servlet对象是否存在,如果不存在则创建并初始化
* 方法只会调用一次
* @throws ServletException
*/
@Override
public void init() throws ServletException {
System.out.println("初始化");
super.init();
}
}
接收请求
//获取客户端请求的完整URL(从http开始,到?前面结束
String url=request.getRequestURL().toString();
System.out.println("获取客户端请求的完整URL:"+url);
//获取客户端的部分URL(从站点名开始,到?前面结束
String uri=request.getRequestURI();
System.out.println("获取客户端的部分URL:"+uri);
//获取请求行中的参数部分
String queryString=request.getQueryString();
System.out.println("获取请求行中的参数部分:"+queryString);
//获取客户端的请求方式
String method=request.getMethod();
System.out.println("获取客户端的请求方式:"+method);
//获取HTTP版本号
String protocol=request.getProtocol();
System.out.println("获取HTTP版本号"+protocol);
//获取项目站点名
String webapp=request.getContextPath();//上下午路径
System.out.println("获取项目的站点名:"+webapp);
获取请求参数
/* 获取请求的参数 */
//获取指定名称的参数值
String uname=request.getParameter("uname");
//获取指定名称的参数的所有参数
String upwd=request.getParameter("upwd");
//获取指定名称的参数的所哟参数值,返回字符串数组(用于复选框传值
String[] hobbys=request.getParameterValues("hobby");
if(hobbys!=null&hobbys.length>0){
for(String hobby:hobbys){
System.out.println("爱好:"+hobby);
}
}
请求乱码的问题
req.setCharacterEncoding("UTF-8");
请求转发
特点:
- 地址不会发生改变
- 参数共享
- 只能在项目内跳转
- 服务端行为
- request作用域有效,session作用域有效
当客户端请求到达后,服务器进行转发,此时会将请求对象进行保存,地址栏中的URL地址不会改变,得到响应后,服务器端再讲响应发送给客户端,从始至终只有一个请求发出
通过该对象可以在一个请求中传递数据,作用范围:在一次请求中有效,即服务器跳转有效
//设置域对象内容
req.setAttribute(String name,Object value);
//获取域对象内容
req.getAttribute(String name)
//删除域对象内容
req.removeAttribute(String name);
响应数据
字节流和字符类不能同时使用
//字符流
PrintWriter writer=resp.getWriter();
writer.write("Hello");
//字节流
ServletOutputStream out=resp.getOutputStream();
out.write("Hi".getBytes());
字符流:要同时设置客户端和服务端的字符类型都支持中文,才不会乱码
resp.setCharacterEncoding("UTF-8");
resp.setHeader("content-type","text/html;charset=UTF-8");
字节流:指定客户端和服务器使用的方式一致
resp.setHeader("content-type","text/html;charset=UTF-8");
注意:设置要在打印之前执行
重定向
服务端指导,客户端的行为。
特点:
- 地址会发生改变
- 参数不会共享
- 可以跳转到http://
- 客户端行为
- request作用域无效,session作用域有效
Cookie对象
通过服务器的出现将一些只需保存在客户端,在客户端处理的数据,不需要通过网络传输。
//创建Cookie对象 Cookie cookie=new Cookie("name","admin"); //发送Cookie对象 resp.addCookie(cookie);
Cookie获取
//获取Cookie数据
Cookie[] cookies = req.getCookies();
//判断数组是否为空
if (cookies != null && cookies.length > 0) {
for (Cookie c : cookies) {
System.out.println(cookie.getName());
System.out.println(cookie.getValue());
}
}
Cookie设置到期时间
- 负整数
- 默认值是-1,表示不存储
- 正整数
- 存活的秒数
- 零
- 表示不存储
Cookie注意点
- 如果服务器发送重复的Cookie,会覆盖原有的Cookie
- 存储是有上限的
- 不能存中文
- 信息只是保存在本机上,换电脑这些信息无效,而且不能跨浏览器
如果一定要存中文,要用URLEncoder编码解码
name=URLEncoder.encode(name)
URLDecode.decode(name)
HttpSession对象
session对象的销毁
- 默认:存活时间是30min,有操作会重新计时。
- 手动设置:可以在Tomcat中的web.xml文件中进行修改
- 立即销毁:通过session.invalidate()方法
- 关闭浏览器:默认只在浏览器中存活,关闭立即失效
- 关闭服务器:session销毁
ServletContext对象
//获取ServletContext对象
ServletContext servletContext=request.getServletContext();
//设置域对象
servletContext.setAttribute("name","zhang");
//获取域对象
String name=(String)servletContext.getAttribute("name");
//移除域对象
servletContext.removeAttribute("name");
Servlet的三大域对象
- request域对象
- 在一次请求中有效,请求转发有效,重定向无效
- session域对象
- 在一次会话中有效,请求转发和重定向都有效,session销毁后无效
- ServletContext域对象
- 在整个应用程序中有效,关闭后失效
文件上传和下载
前台页面:
- 要设置form表单的enctype为:“multipart/form-data”
- 请求方式为POST
- 设置文件提交的地址
- 准备表单元素
- 设置表单元素的name属性(否则后台无法接收)
后台实现:
- 使用@MultipartConfig注解