Java Web专题攻关
servlet概念
servlet其实就是运行在服务器的一个小程序
如何去理解呢?我们访问服务器的资源包括静态资源和动态资源,其中静态资源是我们放置的模板,CSS、JS等文件,是不变的。而我们访问的动态资源,是根据我们访问的请求路径,路由到指定的类去加载,运行其对应的方法而给出浏览器资源的响应。那么凭什么我们的服务器会帮我们去加载,去运行指定的方法呢?或者说服务器是怎么做到的呢?我们一定是需要遵循服务器某些规则才行的,而规则在 Java 中 就是接口,Servlet就是被服务器(Tomcat)识别的接口,我们需要根据这个接口来定义响应的方法。
快速入门
创建一个 Java EE 项目,定义一个类,实现 Servlet 接口
public class ServletDemo1 implements Servlet
实现接口中的抽象方法
在web.xml中配置 Servlet
<!-- 配置Servlet -->
<servlet>
<servlet-name>demo1</servlet-name>
<servlet-class>org.taoguoguo.web.ServletDemo1</servlet-class>
</servlet> <servlet-mapping>
<servlet-name>demo1</servlet-name>
<url-pattern>/demo1</url-pattern>
</servlet-mapping>
Servlet执行原理:
- 当服务器接受客户端浏览器的请求后,会解析请求的URL路径,获取访问Servlet的资源路径
- 查找web.xml文件,是否有对应的
<url-pattern>标签体内容 - 如果有,则会找到对应的
<servlet-class>全类名 - tomcat 会将字节码文件加载进内存,并且创建其对象
- 调用其方法
Servlet中的生命周期:
被创建:执行 init 方法,只执行一次
Servlet 什么时候被创建?
默认情况下, 第一次访问时,Servlet被创建
可以配置Servlet的创建时机,如下:
<!-- 配置Servlet -->
<servlet>
<servlet-name>demo2</servlet-name>
<servlet-class>org.taoguoguo.web.ServletDemo2</servlet-class>
<!--指定Servlet的创建时机
1.第一次被访问时创建,创建
<load-on-startup>的值为负数,默认为 -1,或者不配置 也是在第一次访问被创建
2.在服务器启动时,创建
<load-on-startup>的值为0或正整数 一般设置为 1
-->
<load-on-startup>-1</load-on-startup>
</servlet> <servlet-mapping>
<servlet-name>demo2</servlet-name>
<url-pattern>/demo2</url-pattern>
</servlet-mapping>
注意:Servlet的 init 方法只执行一次,说明servlet在内存中只存在一个对象,是单例的,所以多个用户线程同时访问时,可能存在线程安全问题。解决方案:1.尽量不要在Servlet中定义成员变量,因为成员变量时每个线程共享的资源,可能有的线程会修改,而部分线程会去获取,容易出现数据安全问题。尽量在方法内部定义局部变量代替成员变量。因为局部变量是在每个线程方法栈中独享的一份资源,可以保证数据安全性。2.如果必须要定义成员变量,不要在方法中去修改成员变量的值,仅用于线程获取,是不会有线程安全问题的。3.在servlet解决多线程安全问题时,切记使用同步锁,因为会非常的消耗性能
提供服务: 执行 service 方法,执行多次
- 每次访问 Servlet 时,service 方法都会被调用一次
被销毁:执行 destory 方法,只执行一次
- servlet 被销毁时执行,只有服务器正常关闭时,servlet 被销毁 才会执行destory 方法。destory 是在被销毁之前执行,一般用于释放资源
Servlet 3.0:
支持注解配置,可以不需要web.xml了
使用方法:在创建的servlet 上 加上注解
@WebServlet("资源路径"),支持如下写法@WebServlet(urlPatterns = "/demo3")@WebServlet("/demo3")
HTTP
概念:
Hyper Text Transfer Protocol 超文本传输协议
传输协议:
定义了 客户端 和 服务器端通信时,发送数据的格式
特点:
1.基于TCP/IP的高级协议(安全的,需要经过三次握手)
2.默认端口号:80
3.基于 请求/响应 模型 (请求响应 一 一 对应,一次请求,一次响应)
4.无状态的:每次请求之间相互独立,不能交互数据
历史版本:
1.0:每一次请求,建立一次新的连接
1.1: 复用连接,对缓存性能较好
请求消息数据格式:
请求行
组成格式:请求方式 请求url 请求协议/版本
请求行: GET /login.html HTTP/1.1
HTTP协议有7种请求方式,常用的有两种请求头
GET
请求参数在请求行中,在url后。
请求的url长度有限制
相对安全
POST
请求参数在请求体中
请求的url是没有限制的
相对安全
请求头(客户端告知服务端自身信息)
组成格式: 请求头名称:请求头值
Host:localhost 当前主机
User-Agent:浏览器告诉服务器,我访问你使用的浏览器版本信息
//作用:服务端可以根据这个头信息区分不同浏览器解决浏览器兼容问题
Accept:告诉服务器,浏览器可以解析哪些格式类型的信息 如 text/html
Accept-Language: 告诉服务器所支持的语言环境
Accept-Encoding: 告诉服务器所支持的压缩格式
Referer:http://localhost:8080/login.html 告诉服务器,当前请求从那里来
Referer的常用场景: 1.防盗链 2.统计工作
Connection: keep-alive 连接状态 保持活跃服用
Upgrade-Insecure-Requests:1 浏览器升级信息
请求空行
- 空行,用于用于分割POST请求的请求头,和请求体的
请求体
- 封装POST请求消息的请求参数
username=zhangsan
- 封装POST请求消息的请求参数
响应消息数据格式:
服务端发送给客户端的数据
1.响应行
- 格式:协议/版本 响应状态码 状态码描述
HTTP/1.1 200 OK
响应状态码:服务器告诉客户端本次请求和响应的一个状态
状态码都是三位数字:
1xx: 服务器接收客户端消息,但是没有接收完成,等待一段时间后,发送1xx 状态码询问客户端
2xx: 成功。代表: 200
3xx: 重定向。
302(重定向) 服务器响应状态码302 并给客户端响应一个地址,客户端收到302状态码后自动请求对应地址进行重定向。 304 (访问缓存)例如请求访问图片等大二进制数据时较慢,当浏览器请求服务器,服务器返回图片资源时,浏览器会进行图片的本地缓存,当再次访问时,如果服务器资源没有变化,
并且当前浏览器存在图片缓存,可返回状态码 304 告知浏览器访问缓存,提高交互体验。
4xx: 客户端错误
404 客户端访问路径错误,未找到访问资源 - 405 请求方式没有对应的 doXXX 方法
5xx: 服务端错误
500 服务器内部异常
2.响应头
1.格式: 头名称:值
2.常见的响应头:
Content-Type: 服务器告诉客户端本次响应体数据格式及编码格式
Content-Length: 响应字节长度
Date: 响应时间
Content-disposition: 服务器告诉客户端以什么格式打开响应体数据
默认值: in-line 在当前页面打开
attachment;filename 以附件形式打开响应体,文件下载中使用
3.响应空行
4.响应体
响应体就是传输的响应数据,有整个页面的解析响应,也有二进制数据
Request
request对象和response对象的原理
- request对象和response对象是由服务器创建的,我们来使用它
- request对象是来获取请求消息,response对象是来设置响应消息的
Request对象继承体系结构
- ServletRequest -- 接口
- HttpServletRequest -- 子接口(继承 ServletRequest)
- org.apache.catalina.connector.RequestFacade 类(Tomcat)
- HttpServletRequest -- 子接口(继承 ServletRequest)
request功能:
1.获取请求消息数据
1.获取请求行数据
*GET /day14/demo1?name=zhangsan HTTP/1.1
获取请求方式:GET
String getMethod();
获取虚拟目录:/day14
String getContextPath();
获取servlet路径:/demo1
String getServletPath();
获取get方式请求参数:name=zhangsan
String getQueryString();
获取请求的URI:/day14/demo1
//URI: 统一资源标识符 共和国
//URL: 统一资源定位符 中华人民共和国
String getRequestURI(); //URI如: /day14/demo1
StringBuffer getRequestURL(); //URL如: http://localhost/day14/demo1
获取协议及版本:HTTP/1.1
String getProtocol(); //ServletRequest中的方法
获取客户机的IP地址
String getRemoteAddr();
2.获取请求头数据
String getHeader(String var1); //通过请求头的名称获取请求头的值
Enumeration<String> getHeaders(String var1); //获取所有的请求头名称
根据所有请求头获取所有的请求头对应的值:
@Override
protected void doGet(HttpServletRequest request, HttpServletResponse resp) throws ServletException, IOException {
Enumeration<String> names = request.getHeaderNames();
while(names.hasMoreElements()){
String s = names.nextElement();
System.out.println(s +":" + request.getHeader(s));
}
}
判断浏览器版本,根据请求头
1user-agent的值来判断 是否包含指定浏览器关键字,比如Chromeuser-agent:Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/83.0.4103.116 Safari/537.36
判断请求来源,用于防盗链 可根据请求头
referer的值来判断referer:http://localhost:8080/hello.html
3.获取请求体数据
请求体:只有POST请求,才有请求体,在请求体中封装了POST请求的请求参数
步骤:
1.获取流对象
BufferedReader getReader() //获取字符输入流,只能操作字符数据
获取页面表单提交数据
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//获取请求消息体-请求参数
//1.获取字符流
BufferedReader br = request.getReader();
//2.读取数据
String line = null;
while ((line = br.readLine()) != null){
System.out.println(line);
}
}
2.再从流对象中拿数据
ServletInputStream getInputStream() //获取字节输入流 可以操作所有类型数据 文件上传中使用
4.其他功能
1.获取请求参数通用方式:
不论是get 还是post 请求方式都可以使用下列方法来获取请求参数
String getParameter(String name) 根据请求参数获取参数值 username=zhangsan&password=123
@WebServlet("/demo5")
public class ServletDemo5 extends HttpServlet {
protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//post获取请求参数
String username = request.getParameter("username");
System.out.println("post");
System.out.println(username); } protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
//get获取请求参数
String username = request.getParameter("username");
System.out.println("get");
System.out.println(username);
}
}
String[] getParameter(String name) 根据请求参数获取参数值的数组 hobby=study&hobby=eat
Enumeration getParameterNames() 获取所有请求的参数名称
Map<String, String[]>getParameterMap()获取所有的请求参数和值 封装为一个Map返回
技巧:获取前台参数时,可创建一个Java Bean对象,然后使用BeanUtils.populate(Object obj,Map map)
将map集合的键值信息,封装到 Java Bean对象中
获取请求参数乱码问题
get方式:Tomcat 8 已经将get方式乱码问题解决
post方式:会乱码
解决方法:获取参数前设置一下流的编码,编码来自于请求提交页面编码格式
request.setCharacterEncoding("utf-8");
2.请求转发:
一种在服务器内部资源跳转的方式
步骤:
1.通过 request 对象获取请求转发器对象,RequestDispatcher getRequestDispatcher(String var1);
2.使用 RequestDispatcher 来进行转发:forward(ServletRequest var1, ServletResponse var2)
特点:
1.浏览器地址栏不发生变化
2.转发只能转发当前服务器的内部资源,比如不能转发到访问百度
3.转发是一次请求
3.共享数据
域对象:一个有作用范围的对象,可以在范围内共享数据。
request域:代表一次请求,一般用于一次请求转发的访问多个资源中共享数据
方法:
1.setAttribute(String name,Object obj); 存储对象
2.Object getAttribute(String name); 通过键获取值
3.removeAttribute(String name); 通过键移除值
4.获取ServletContext
ServletContext getServletContext()
Response
功能:设置响应消息
1.设置响应行
1.格式:HTTP /1.1 200
一般是设置响应状态码 :setStatus(int sc)
2.设置响应头
setHeader(String name, String value)
3.响应体
使用步骤:
1.获取输出流
1.字符输出流:PrintWriter getWriter()
2.字节输出流:ServletOutputStream getOutputStream()
2.使用输出流,将数据输出到客户端浏览器
案例:
1.完成重定向
重定向是客户端浏览器访问服务器时,服务器响应客户端访问重定向另一个资源进行处理的过程
第一种方式:
设置响应状态码为302 response.setStatus(302);
设置响应头location 值为要跳转的路径 response.setHeader("location","/demo7");
第二种方式:
使用response对象的重定向方法: response.sendRedirect("/demo7");
重定向特点
1.地址栏发生变化
2.重定向可以访问其他站点服务器资源
3.重定向是多次请求,不能使用request域对象来共享数据
路径分类
相对路径:通过相对路径不可以确定唯一资源
- 如:./index.html
- 相对路径通常以 . 开头
- 写法规则:先确定当前资源和访问的目标资源之间的相对位置关系
- ./ 代表当前目录,也可省略 , ../ 代表上一级目录
相对目录路径写法举例:
比如一个项目,有个路径为 demo1的 Servlet,访问路径为::
http://localhost:8080/project/demo1 -> 目标资源
项目的web目录下有个index.html,访问路径为:
http://localhost:8080/project/index.html -> 当前资源
当前资源和目标资源在同一个层级上,那么从 index.html访问 demo1的路径通常为
<a href="./demo1">访问Demo1</a>当前目录的 ./ 是可以省略的,所以也可以简写为:demo1绝对路径:通过绝对路径可以确定唯一资源
如:http://localhost:8080/project/demo1 这是完整的绝对路径
也可以简写,以 / 开头的路径 如:/project/demo1 因为协议和主机端口等通常都是固定的
写法规则:判断路径是给谁用的的?如何判断,就是看请求从哪里发出。客户端发出就是给客户端用,服务端发出就是给服务端用。
给客户端使用:需要加虚拟目录(项目的访问路径),常见的比如标签,
注意,重定向本质上是两次请求,服务端给出路径资源,客户端重新发起请求,所以转发时是需要使用全路径的,通常我们会采用动态获取的方式来拿到虚拟目录。
response.sendRedirect(request.getContextPath()+"/demo7"); //比如重定向
Java Web专题攻关的更多相关文章
- Java Web专题
- 【原创】三分钟教你学会MVC框架——基于java web开发(2)
没想到我的上一篇博客有这么多人看,还有几位看完之后给我留言加油,不胜感激,备受鼓励,啥都别说了,继续系列文章之第二篇.(如果没看过我第一篇博客的朋友,可以到我的主页上先浏览完再看这篇文章,以免上下文对 ...
- Java Web 高性能开发,第 2 部分: 前端的高性能
Web 发展的速度让许多人叹为观止,层出不穷的组件.技术,只需要合理的组合.恰当的设置,就可以让 Web 程序性能不断飞跃.Web 的思想是通用的,它们也可以运用到 Java Web.这一系列的文章, ...
- Java Web 高性能开发,第 1 部分: 前端的高性能
Web 发展的速度让许多人叹为观止,层出不穷的组件.技术,只需要合理的组合.恰当的设置,就可以让 Web 程序性能不断飞跃.所有 Web 的思想都是通用的,它们也可以运用到 Java Web.这一系列 ...
- 使用Intellij idea新建Java Web项目(servlet) 原理及初步使用
准备 JDK (配置JDK_HOME\bin 和 CLASSPATH) 注:JDK8下载已经需要注册了,请使用JDK11(现在是官方长期支持的版本) 对于我们新手来说,JD ...
- 大型Java进阶专题(一) 前言
前言 各位读者好,本系列为Java进阶专题,为那些有一定工作经验,做了多年业务的码农,希望突破技术瓶颈,但没有形成系统的Java只是体系,缺乏清晰的提升方法和学习路径的人,比如作者本人.该课题的是 ...
- Java登录专题-----创建用户(一)
Java登录专题-----创建用户(一) 我来填坑了 创建用户 入参 应该包括: 用户姓名,用户密码,用户手机号,用户所属机构 用户版本号,角色id 出参: 没有 数据结构: JavaBean ...
- 高效 Java Web 开发框架 JessMA v3.5.1
JessMA 是功能完备的高性能 Full-Stack Web 应用开发框架,内置可扩展的 MVC Web 基础架构和 DAO 数据库访问组件(内部已提供了 Hibernate.MyBatis 与 J ...
- 高效 Java Web 开发框架 JessMA v3.4.1
JessMA 是功能完备的高性能 Full-Stack Web 应用开发框架,内置可扩展的 MVC Web 基础架构和 DAO 数据库访问组件(内部已提供了 Hibernate.MyBatis 与 J ...
- java web 之客户关系管理系统
这个周末真的是觉得自己学会了一个比较高大上的本领,为什么这么觉得呢?那是因为星期六的时候觉得自己可以看看源码能做出来,可是让我头疼的是花费了一上午的时间还是没有弄出来,还好上天给了我机会,要是没有老师 ...
随机推荐
- OpenCV程序练习(三):图像运算
一.图像加法运算 代码 import cv2 img=cv2.imread("demoimg.jpg",0) #读取图片,参数0等价于cv2.IMREAD_GRAYSCALE,将图 ...
- 如何在 Windows 使用 Podman Desktop 取代 Docker Desktop
Podman Desktop 是 Docker Desktop 的免费替代品,是本地开发使用的另一个绝佳选择.它提供了类似的功能集,同时保持完全开源,让您避免使用 Docker 产品的许可问题.在本文 ...
- 推荐常用的Idea插件
Idea常用快捷键 删除所有空行 使用替换 Ctrl + R 点亮后面的魔法图标启用正则表达式,输入:^\s*\n,然后选择替换全部 查询指定类或方法在哪里被引用 光标点中需要查找的类名和方法名,然后 ...
- nodejs-mime类型
mime是一个互联网标准,通过设定它就可以设定文件在浏览器的打开方式. mime使用方法: 使用mime模块查询文件的MIME类型: mime.getType('/path/to/file.txt') ...
- 开源免费又好用的中式数据报表:UReport2是一款高性能的架构在Spring之上纯Java报表引擎,通过迭代单元格可以实现任意复杂的中国式报表。
北润乾.南帆软,数加发力在云端. uReport 身何安?中式报表真开源. 报表江湖之中,uReport安身立命的产品品类定位是什么? 说来很简单,uReport的价值在于填补了这样一个市场空白:开源 ...
- Vue 数组和对象更新,但视图未更新,背后的故事
在实际开发中,遇到遍历数组和对象,当property 发生改变时,并没有触发视图的更新今天来浅显的聊聊这背后的故事,有说的不对地方,还望指出! 本人博文地址:https://www.cnblogs.c ...
- Python爬虫Post请求返回值为-1000
今天写了一个简单的爬虫程序,为了爬取kfc官网的餐厅数据,代码如下 # ajax的post请求--肯德基官网 def create_request(page): url='http://www.kfc ...
- AT_arc149_a 题解
洛谷链接&Atcoder 链接 本篇题解为此题较简单做法及较少码量,并且码风优良,请放心阅读. 题目简述 求满足以下条件的小于 \(10 ^ n\) 数最大是多少? 每一位数字均相同: 是 \ ...
- 第四章:springboot整合mybatis
1,引入mybatis依赖 <dependency> <groupId>org.mybatis.spring.boot</groupId> <artifact ...
- 嘿,我使用了mp的自连接+分页查询之后,再使用条件查询居然失效了。
原因:我想通过自连接查询将一个表的两条数据放在一起,为此我重写了mp的分页查询 IPage<Indi> selectIndiShow(IPage<Indi> page, @Pa ...