JavaWeb_Servlet_Request_Response

1.Servlet

(1)Servlet是JavaEE的一个动态web资源开发技术,就是在服务器上运行的小程序,这个小程序是由服务器调用的,服务器为了能调用这个小程序,就要求这样的程序必须实现一个Servlet接口或去继承Servlet的实现类GenericServlet或HttpServlet、

(2)在web应用的配置文件web.xml中配置Servlet的对外访问路径,再将整个web应用交给虚拟机管理。

(3)Servlet的调用过程:

①服务器根据请求的包的host找到要访问的虚拟主机

②在请求包中需求的web应用资源找到对应的web应用,根据请求资源去查询web.xml 中的 url-pattern 找到对应的Servlet名字,再根据这个名字找到对应的Servlet-Class

③服务器调用这个Servlet-Class传递reqeust和response到该类中运行

④Servlet类将处理好的信息用response写入缓存中,再给服务器返回,最后交给浏览器

(4)Servlet的生命周期:从第一次被调用开始就由服务器创建出来,在创建出来后立即调用init方法初始化,直到web应用被虚拟主机移除或者服务器停掉,Servlet才会被销毁,

(5)Servlet的继承结构:

Servlet接口:所有Servlet必须实现的接口
  |
   ——GenericServlet:对Servlet接口的一个默认实现,其中包含的service方法为一个抽象方法,并且提供一些常用方法。
      |
       ——HttpServlet:专门增加了对HTTP请求的处理机制,实现了GenericServlet中的service方法,在此方法中会判断请求是以什么请求方式发送的,根据不同的请求方式会去调用响应的doXXX()方法,我们只需要继承HttpServlet,覆写其中的doGet()和doPost()方法就可以处理相应的GetPost请求了了。

(6)web.xml中配置Servlet
<web-app>
//注册一个Servlet
  <servlet>

//配置Servlet的名字

<servlet-name>AnyName</servlet-name>

//配置Servlet的对应类,注意,这里要的是一个类的全路径名,不要写成斜杠分割,也不要带.java或.class后缀名

<servlet-class>HelloServlet</servlet-class>

</servlet>

//映射Servlet的对外访问路径
  <servlet-mapping>

//配置映射的Servlet名

<servlet-name>AnyName</servlet-name>

//配置虚拟访问路径

<url-pattern>/demo/hello.html</url-pattern>

</servlet-mapping>
</web-app>

(7)web.xml额外功能标签
①<load-on-startup>用在<servlet>标签中,来表示服务器启动时就创建该类的实例,其中可以设置一个正整形值,表示启动的顺序
②一个servlet的对外访问路径被配置为/,则该Servlet就是一个缺省Servlet,其他Servlet不处理的事情就由缺省Servlet来处理。
③<init-param>元素,其中可以配置一些初始化信息,在Servlet里可以通过ServletConfig获得

(8)线程安全问题

①类变量导致线程安全问题或者多个线程同时操作Servlet一会引起线程安全问题。
②解决办法,使用同步代码块嵌套可能造成线程安全问题的代码,不过会造成访问延迟和效率低下,避免使用类变量和减少线程阻塞时间和保证同步代码块代码尽量少。
(9)Context域对象
①ServletContext代表整个应用,在整个web应用都能够被看见,利用ServletContext域可以在不同的Servlet之间传递信息。

<context-param>
    <param-name>contextData1</param-name>
    <param-value>zzzzzz</param-value>
</context-param>

②利用ServletContext获取资源:

this.getServletContext().getResourceAsStream("虚拟路径");

this.getSerlvetContext().getRealPath("虚拟路径");//返回资源的真实路径

③利用类加载器去获取资源

loader.getResource("虚拟路径在classes路径算起").getPath();

//获取资源的真实路径再创建流

 
2、response
(1)向浏览器输出一段数据

 //输出中文 没有乱码问题 底层是OutputStream输出
 PrintWriter print = new PrintWriter(response.getOutputStream(),true);
  print.println("happy!!!!!!!你们别玩了");
  
  //设置响应头的编码形式可以防止乱码

  response.setContentType("text/html;charset=utf-8");
  response.getWriter().write("你们别玩了")

(2)实现文件下载

//用URL编码防止中文乱码
  String path = URLEncoder.encode("成人脑力训练.zip", "utf-8");
  //设置消息头以附件的形式发布
  response.setHeader("Content-Disposition", "attachment;filename="+path);
  //用Context 流读取对应要下载的文件
  InputStream in = this.getServletContext().getResourceAsStream("/WEB-INF/成人脑力训练.zip");
  //关联输出流输出文件
  OutputStream out = response.getOutputStream();
  int len = 0;
  byte[] arr = new byte[1024];
  while ((len = in.read(arr)) != -1) {
   out.write(arr, 0, len);
  }
  in.close();

(3)设置响应头refresh可以实现页面的定时刷新功能

 //html下 用<meta http-equiv= "" content="">可以模拟头功能
  //定时跳转,登陆成功跳转就用的这个功能
  //response.setHeader("Refresh", "3;url=/web/HelloWorld");

(4)控制是否缓存资源

//不缓存0或-1,用的是格林威治时间,用毫秒数表示
  response.setDateHeader("Expires", 0); 

(5)实现请求重定向

//重定向,代替了设置响应头的Location和设置状态302
  //response.sendRedirect("/web/HelloWorld");

(6)验证码的制作

//设置不缓存
  response.setDateHeader("Expires", 0);
  
  //设置验证码的宽和高
  int width = 150, height = width / 3;
  //获取一个画板并设置宽和高,以及显示的色泽通道
  BufferedImage img = new BufferedImage(width, height,BufferedImage.TYPE_INT_RGB);
  //获取画布
  Graphics2D g = (Graphics2D) img.getGraphics();
  //设置颜色画背景
  g.setColor(Color.WHITE);
  g.fillRect(0, 0, width, height);
  //设置颜色画边框
  g.setColor(Color.BLUE);
  g.drawRect(0, 0, width - 1, height - 1);
  
  //设置常见汉字字符的样本
  String base = "u7684u4e00u4e86u662fu6211u4e0du5728u4ebau4eecu6709u6765u4ed6u8fd9u4e0au7740u4e2au5730u5230u5927u91ccu8bf4u5c31u53bbu5b50u5f97u4e5fu548cu90a3u8981u4e0bu770bu5929u65f6u8fc7u51fau5c0fu4e48u8d77u4f60u90fdu628au597du8fd8u591au6ca1u4e3au53c8u53efu5bb6u5b66u53eau4ee5u4e3bu4f1au6837u5e74u60f3u751fu540cu8001u4e2du5341u4eceu81eau9762u524du5934u9053u5b83u540eu7136u8d70u5f88u50cfu89c1u4e24u7528u5979u56fdu52a8u8fdbu6210u56deu4ec0u8fb9u4f5cu5bf9u5f00u800cu5df1u4e9bu73b0u5c71u6c11u5019u7ecfu53d1u5de5u5411u4e8bu547du7ed9u957fu6c34u51e0u4e49u4e09u58f0u4e8eu9ad8u624bu77e5u7406u773cu5fd7u70b9u5fc3u6218u4e8cu95eeu4f46u8eabu65b9u5b9eu5403u505au53ebu5f53u4f4fu542cu9769u6253u5462u771fu5168u624du56dbu5df2u6240u654cu4e4bu6700u5149u4ea7u60c5u8defu5206u603bu6761u767du8bddu4e1cu5e2du6b21u4eb2u5982u88abu82b1u53e3u653eu513fu5e38u6c14u4e94u7b2cu4f7fu5199u519bu5427u6587u8fd0u518du679cu600eu5b9au8bb8u5febu660eu884cu56e0u522bu98deu5916u6811u7269u6d3bu90e8u95e8u65e0u5f80u8239u671bu65b0u5e26u961fu5148u529bu5b8cu5374u7ad9u4ee3u5458u673au66f4u4e5du60a8u6bcfu98ceu7ea7u8ddfu7b11u554au5b69u4e07u5c11u76f4u610fu591cu6bd4u9636u8fdeu8f66u91cdu4fbfu6597u9a6cu54eau5316u592au6307u53d8u793eu4f3cu58ebu8005u5e72u77f3u6ee1u65e5u51b3u767eu539fu62ffu7fa4u7a76u5404u516du672cu601du89e3u7acbu6cb3u6751u516bu96beu65e9u8bbau5417u6839u5171u8ba9u76f8u7814u4ecau5176u4e66u5750u63a5u5e94u5173u4fe1u89c9u6b65u53cdu5904u8bb0u5c06u5343u627eu4e89u9886u6216u5e08u7ed3u5757u8dd1u8c01u8349u8d8au5b57u52a0u811au7d27u7231u7b49u4e60u9635u6015u6708u9752u534au706bu6cd5u9898u5efau8d76u4f4du5531u6d77u4e03u5973u4efbu4ef6u611fu51c6u5f20u56e2u5c4bu79bbu8272u8138u7247u79d1u5012u775bu5229u4e16u521au4e14u7531u9001u5207u661fu5bfcu665au8868u591fu6574u8ba4u54cdu96eau6d41u672au573au8be5u5e76u5e95u6df1u523bu5e73u4f1fu5fd9u63d0u786eu8fd1u4eaeu8f7bu8bb2u519cu53e4u9ed1u544au754cu62c9u540du5440u571fu6e05u9633u7167u529eu53f2u6539u5386u8f6cu753bu9020u5634u6b64u6cbbu5317u5fc5u670du96e8u7a7fu5185u8bc6u9a8cu4f20u4e1au83dcu722cu7761u5174u5f62u91cfu54b1u89c2u82e6u4f53u4f17u901au51b2u5408u7834u53cbu5ea6u672fu996du516cu65c1u623fu6781u5357u67aau8bfbu6c99u5c81u7ebfu91ceu575au7a7au6536u7b97u81f3u653fu57ceu52b3u843du94b1u7279u56f4u5f1fu80dcu6559u70edu5c55u5305u6b4cu7c7bu6e10u5f3au6570u4e61u547cu6027u97f3u7b54u54e5u9645u65e7u795eu5ea7u7ae0u5e2eu5566u53d7u7cfbu4ee4u8df3u975eu4f55u725bu53d6u5165u5cb8u6562u6389u5ffdu79cdu88c5u9876u6025u6797u505cu606fu53e5u533au8863u822cu62a5u53f6u538bu6162u53d4u80ccu7ec6";
  
  //设置颜色并画干扰线
  g.setColor(Color.BLACK);
  for (int x = 0; x < 10; x++) {
    g.drawLine(getNum(0, width), getNum(0, height), getNum(0, width),
      getNum(0, height));
  }
  
  //设置字体格式并把字体写入画布
  g.setFont(new Font("微软雅黑", Font.BOLD, 20));
  for(int x=0; x<4; x++)
  { 
    //获取弧度 ,弧度=角度*PI/180
    double d = getNum(-30, 30)*Math.PI/180;
    //旋转一定弧度并把字体写在相应的坐标
    g.rotate(d, 20+x*30, 30);
    g.drawString(base.charAt(getNum(0, base.length()-1))+"", 20+x*30, 30);
    g.rotate(-d, 20+x*30, 30);
  }
  
  //以jpg格式输出到浏览器中
  ImageIO.write(img, "jpg", response.getOutputStream());

(7)字节流和字符流是互斥的
 
3、request
(1)获取request中的各种信息

OutputStream out = response.getOutputStream();
  response.setContentType("text/html;charset=gb2312");
  
  //获取请求头的请求资源地址 主机地址 和端口
  StringBuffer url = request.getRequestURL();
  String uri = request.getRequestURI();
  String addr = request.getRemoteAddr();
  String host = request.getRemoteHost();
  int port = request.getRemotePort();
  String user = request.getRemoteUser();
  
  out.write((uri+"-->"+url+"-->"+addr+"-->"+host+":"+port+"-->"+user).getBytes());
  
  //获取请求头的方法,长度,类型,和编码形式
  String method = request.getMethod();
  int len = request.getContentLength();
  String type = request.getContentType();
  String path = request.getContextPath();
  String encoding = request.getCharacterEncoding();
  
  out.write("<hr/>".getBytes());
  out.write((method+"-->"+len+"-->"+type+"-->"+path+":"+encoding).getBytes());
  
  //获取请求头的名字和值
  Enumeration<String> e = request.getHeaderNames();
   while((e.hasMoreElements()))
   {
    String name =  e.nextElement();
    String value = request.getHeader(name);
    out.write("<hr/>".getBytes());
    out.write((name+":"+value).getBytes());
   }

(2)referer防盗链

  //获取来路的值
  String referer = request.getHeader("Referer");
  //拿来路匹配前面的域名。如果匹配就转发,不匹配就重定向到本域名的首页
  if(referer==null || !referer.startsWith("http://localhost"))
  {
    response.sendRedirect("/web/index.jsp");
  }
  else
    request.getRequestDispatcher("/WEB-INF/miji.html").forward(request, response);


(3)处理中文乱码

   //解决乱码,通用的方法。
   String name = request.getParameter("name");
   name = new String(name.getBytes("ISO8859-1"),"utf-8");

 

  //解决乱码,post的特有方法。
  request.setCharacterEncoding("utf-8");

(4)request域对象和转发

   //将参数带去其他的servlet,但是流要注意用。不能使用2种流
   request.setAttribute("test", "wow!!");
   request.getRequestDispatcher("/f").forward(request, response);

(5)include

   //上面的直接转发会导致流被清空然后输出不了数据,
   //使用include可以输出本面页的信息以及要转发的面页信息,可以用组装面页
   request.getRequestDispatcher("/f").include(request, response);

(6)重定向和转发的区别
①重定向:会改变浏览器的地址,2次请求和响应
②转发:不会改变浏览器的地址,一次请求和响应

(7)缓存区的研究

  OutputStream out = response.getOutputStream();
  
  byte[] arr = new byte[8192];
  
  //缓存区满了会出现 chunked块发送
  //满了后会以16进制数字标识块数据的大小,以0加空内容结束
  out.write(arr);
  
  int size = response.getBufferSize();
  
  out.write(("<hr/>缓冲区满了吗?"+response.isCommitted()+"->缓存区大小:"+size).getBytes());

 

原文地址:https://www.cnblogs.com/lingyi1111/p/4451105.html