文章目录
1.基本概念
1.1 前言
静态Web:
提供给所有人看数据不会发生变化!
HTML,CSS
动态Web:
有数据交互,登录账号密码,网站访问人数等
技术栈:Servlet/JSP,ASP,PHP
在Java中,动态web资源开发的技术统称为JavaWeb;
1.2 Web 应用程序
Web 应用程序:可以提供浏览器访问的程序;
这个统一的web资源会被放在同一个文件夹下,Web 应用程序 —> Tomcat:服务器
一个 Web 应用由多部分组成(静态Web、动态Web)
HTML,CSS,JavaScript
JSP,Servlet
Java 程序
jar 包
配置文件 (Properties)
Web 应用程序编写完毕后,若想提供给外界访问:需要一个服务器来统一管理;
1.3动态 Web 的访问过程
![](https://img-blog.csdnimg.cn/img_convert/c96809db242914663f2142b7af7ed25e.png)
浏览器发送 HTTP 请求,服务器 Tomcat 接收请求,Servlet 容器从磁盘加载 Servlet 程序处理请求 request ,处理结束返回 response。
2. Web 服务器
2.1 技术讲解
PHP:
作为开发速度很快,功能很强大,跨平台
无法承载大访问量的情况
JSP/Servlet:
基于 Java 语言
![](https://img-blog.csdnimg.cn/img_convert/b08ec43b7a99551ece9a2406d3d2af1a.png)
Tomcat是Apache 软件基金会(Apache Software Foundation)的Jakarta 项目中的一个核心项目。
Tomcat 技术先进、性能稳定,而且免费。
3.Tomcat
3.1 安装 Tomcat
Tomcat官网:http://tomcat.apache.org/
![](https://img-blog.csdnimg.cn/img_convert/c5130f256be7a0a4cc42513afd987377.png)
![](https://img-blog.csdnimg.cn/img_convert/56180b2e91eddf889c1d274e031c2b2a.png)
3.2 Tomcat 启动和配置
文件夹:
![](https://img-blog.csdnimg.cn/img_convert/160ac450dce58971018b66d6766f150d.png)
启动和关闭 Tomcat:
![](https://img-blog.csdnimg.cn/img_convert/7596373143e85fac63d059ec1b5c2584.png)
可能遇到的问题:
1.Java 环境变量没有配置导致闪退
2.乱码问题:可在配置文件中配置
12
3.3 配置
Servlet 核心配置文件目录如下:
![](https://img-blog.csdnimg.cn/img_convert/b2eea58b08607a475a0465464eb32cae.png)
可以配置启动的端口号
tomcat的默认端口号为:8080
mysql:3306
http:80
https:443
<Connectorport="8081"protocol="HTTP/1.1"connectionTimeout="20000"redirectPort="8443"/>123
可配置主机名称
默认的主机名为:localhost->127.0.0.1
默认网站应用存放的位置为:webapps
<Hostname="www.qinjiang.com"appBase="webapps"unpackWARs="true"autoDeploy="true">12
面试题:
在浏览器输入一个域名,回车;
本机查看 C:\Windows\System32\drivers\etc\hosts 配置文件是否有相应域名的映射。
case1: 若有,则直接映射到对应的 IP 地址,进行访问。
case2: 若无,则去 DNS 服务器上查找对应的 IP ,找到就返回相应的 IP,找不到就不返回。
![](https://img-blog.csdnimg.cn/img_convert/a41c3ca0071c69018a0a8156348c2e94.png)
![](https://img-blog.csdnimg.cn/img_convert/974026a6901b836b72c240edb082ab55.png)
3.4 发布一个 Web 网站
网站程序的结构:
--webapps :Tomcat服务器的web目录
-ROOT
-kuangstudy :网站的目录名
- WEB-INF
-classes : java程序
-lib:web应用所依赖的jar包
-web.xml :网站配置文件
- index.html 默认的首页
-static-css
-style.css
-js
-img
-.....1234567891011121314
4.HTTP
4.2 什么是 HTTP?
HTTP(hypertext transport protocol) 超文本传输协议。
文本 :HTML,字符串…
超文本:图片,音乐,视频,定位,地图…
80 端口
HTTPS(Hyper Text Transfer Protocol over SecureSocket Layer):是以安全为目标的 HTTP 通道,在 HTTP 的基础上通过传输加密和身份认证保证了传输过程的安全性。HTTPS 在HTTP 的基础下加入SSL 层,HTTPS 的安全基础是 SSL。
443 端口
4.2 两个时代
HTTP 1.0:
HTTP/1.0:客户端与 Web 服务器连接后,只能获得一个Wen 资源,然后就断开连接,加入某个页面有多个图片资源需要加载,那么需要连接多次,影响服务器和客户端的性能。
HTTP 2.0:
HTTP/1.1:客户端可以与web服务器连接后,可以获得多个web资源。
4.3 HTTP 请求
客户端 -> 发送请求(Request)->服务器
百度:
Request URL:https://www.baidu.com/ 请求地址
Request Method:GET get方法/post方法
Status Code:200 OK 状态码:200
Remote(远程) Address:14.215.177.39:4431234
Accept:text/html
Accept-Encoding:gzip, deflate, br
Accept-Language:zh-CN,zh;q=0.9 语言
Cache-Control:max-age=0
Connection:keep-alive
12345
1、请求行
请求行中的请求方式:GET
请求方式:Get,Post,HEAD,DELETE,PUT,TRACT…
get:请求能够携带的参数比较少,大小有限制,会在浏览器的URL地址栏显示数据内容,不安全,但高效
post:请求能够携带的参数没有限制,大小没有限制,不会在浏览器的URL地址栏显示数据内容,安全,但不高效。
2、消息头
Accept:告诉浏览器,它所支持的数据类型
Accept-Encoding:支持哪种编码格式 GBK UTF-8 GB2312 ISO8859-1
Accept-Language:告诉浏览器,它的语言环境
Cache-Control:缓存控制
Connection:告诉浏览器,请求完成是断开还是保持连接
HOST:主机..../.123456
4.4 HTTP 响应
服务器 -> 响应(response) ->客户端
百度:
Cache-Control:private 缓存控制
Connection:Keep-Alive 连接
Content-Encoding:gzip 编码
Content-Type:text/html 类型
1234
1、响应体
Accept:告诉浏览器,它所支持的数据类型
Accept-Encoding:支持哪种编码格式 GBK UTF-8 GB2312 ISO8859-1
Accept-Language:告诉浏览器,它的语言环境
Cache-Control:缓存控制
Connection:告诉浏览器,请求完成是断开还是保持连接
HOST:主机..../.
Refresh:告诉客户端,多久刷新一次;
Location:让网页重新定位;
12345678
2、响应状态码
200:请求响应成功 200
3xx:请求重定向
重定向:你重新到我给你新位置去;
4xx:找不到资源 404
5xx:服务器代码错误 500 502:网关错误
常见面试题:
当你的浏览器中地址栏输入地址并回车的一瞬间到页面能够展示回来,经历了什么?
5. Maven
在 JavaWeb 开发中,需要使用大量的 jar 包,我们手动去导入;
如何能够让一个东西自动帮我导入和配置这个jar包。
由此,Maven诞生了!
5.1 Maven 项目架构管理工具
Maven 的核心思想:约定大于配置
有约束,不要去违反。
Maven 会规定好你该如何去编写我们的 Java 代码,必须要按照这个规范来;
5.2 下载安装 Maven
点击我查看安装教程 https://editor.csdn.net/md/?articleId=107500020
5.3 在 IDEA 中使用 Maven
启动 IDEA
创建一个 MavenWeb 项目
![](https://img-blog.csdnimg.cn/img_convert/ae28d4532823dc4721d4b32082d06e57.png)
![](https://img-blog.csdnimg.cn/img_convert/aacd61cebba7c5e996a0683eb5910a3f.png)
![](https://img-blog.csdnimg.cn/img_convert/8ff0e575b92544fb796aebeefaa4c339.png)
![](https://img-blog.csdnimg.cn/img_convert/23e9764fd91baaac0e0f6e2ccce19cdc.png)
![](https://img-blog.csdnimg.cn/img_convert/7e8a2da1fb9661dfbdded024c12b02b9.png)
3. 等待项目初始化完毕
![](https://img-blog.csdnimg.cn/img_convert/61fdc31375a106a764858f1341acbae6.png)
![](https://img-blog.csdnimg.cn/img_convert/f2c8c538ca9c184eca4a2a60dad335e7.png)
4. 观察 maven 仓库中多了什么东西?
5. IDEA 中的 Maven 设置
注意:IDEA 项目创建成功后,看一眼 Maven 的配置
![](https://img-blog.csdnimg.cn/img_convert/596bd802b68d5fcb8bb5d48c151cb441.png)
![](https://img-blog.csdnimg.cn/img_convert/05fe5094255485f2cf0ccf454b81b10f.png)
6. 到这里,Maven 在 IDEA 中的配置和使用就 OK 了!
5.4 创建一个普通的 Maven 项目
![](https://img-blog.csdnimg.cn/img_convert/974098587cb9e8118d8ab464d789ba21.png)
![](https://img-blog.csdnimg.cn/img_convert/b2a28b0b23bd7d4ad47d7613e04e3baf.png)
这个只有在Web应用下才会有!
![](https://img-blog.csdnimg.cn/img_convert/d718c48f45058a4162a7e7fc21fcc483.png)
5.5 标记文件夹功能
![](https://img-blog.csdnimg.cn/img_convert/ad5e65c227d4b7a235a4429bcf038af0.png)
![](https://img-blog.csdnimg.cn/img_convert/674b90873b2a8330fed96ad2b3b1814a.png)
![](https://img-blog.csdnimg.cn/img_convert/b9edb7f344fb218ce72ab2a65b705d75.png)
5.6 在 IDEA 中配置 Tomcat
![](https://img-blog.csdnimg.cn/img_convert/7367a709efa960e53da45135f6163ee8.png)
![](https://img-blog.csdnimg.cn/img_convert/5e03cf0406cd0319ea3744f8dff1a0e1.png)
![](https://img-blog.csdnimg.cn/img_convert/e5abc7912e07c723178679563dbcc133.png)
解决警告问题
必须要的配置:为什么会有这个问题:我们访问一个网站,需要指定一个文件夹名字;
![](https://img-blog.csdnimg.cn/img_convert/de69c81e8664cf9a448cf0913951b5ce.png)
![](https://img-blog.csdnimg.cn/img_convert/d475675075f00a4c010b7dee98688e45.png)
![](https://img-blog.csdnimg.cn/img_convert/6ea51e10947275fee89001e9456b985a.png)
5.7 pom 文件
pom.xml 是Maven的核心配置文件
![](https://img-blog.csdnimg.cn/img_convert/2f2adcc5ac364d974998559afc7b3c18.png)
<?xml version="1.0" encoding="UTF-8"?><!--Maven版本和头文件--><projectxmlns="http://maven.apache.org/POM/4.0.0"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"><modelVersion>4.0.0</modelVersion><!--这里就是我们刚才配置的GAV--><groupId>com.kuang</groupId><artifactId>javaweb-01-maven</artifactId><version>1.0-SNAPSHOT</version><!--Package:项目的打包方式
jar:java应用
war:JavaWeb应用
--><packaging>war</packaging><!--配置--><properties><!--项目的默认构建编码--><project.build.sourceEncoding>UTF-8</project.build.sourceEncoding><!--编码版本--><maven.compiler.source>1.8</maven.compiler.source><maven.compiler.target>1.8</maven.compiler.target></properties><!--项目依赖--><dependencies><!--具体依赖的jar包配置文件--><dependency><groupId>junit</groupId><artifactId>junit</artifactId><version>4.11</version></dependency></dependencies><!--项目构建用的东西--><build><finalName>javaweb-01-maven</finalName><pluginManagement><!-- lock down plugins versions to avoid using Maven defaults (may be moved to parent pom) --><plugins><plugin><artifactId>maven-clean-plugin</artifactId><version>3.1.0</version></plugin><!-- see http://maven.apache.org/ref/current/maven-core/default-bindings.html#Plugin_bindings_for_war_packaging --><plugin><artifactId>maven-resources-plugin</artifactId><version>3.0.2</version></plugin><plugin><artifactId>maven-compiler-plugin</artifactId><version>3.8.0</version></plugin><plugin><artifactId>maven-surefire-plugin</artifactId><version>2.22.1</version></plugin><plugin><artifactId>maven-war-plugin</artifactId><version>3.2.2</version></plugin><plugin><artifactId>maven-install-plugin</artifactId><version>2.5.2</version></plugin><plugin><artifactId>maven-deploy-plugin</artifactId><version>2.8.2</version></plugin></plugins></pluginManagement></build></project>12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576
![](https://img-blog.csdnimg.cn/img_convert/80997e5102e2833c340b8d00bb799864.png)
Maven 由于他的约定大于配置,我们之后可以能遇到我们写的配置文件,无法被导出或者生效的问题,解决方案:
<!--在build中配置resources,来防止我们资源导出失败的问题--><build><resources><resource><directory>src/main/resources</directory><includes><include>**/*.properties</include><include>**/*.xml</include></includes><filtering>true</filtering></resource><resource><directory>src/main/java</directory><includes><include>**/*.properties</include><include>**/*.xml</include></includes><filtering>true</filtering></resource></resources></build>123456789101112131415161718192021
5.8 IDEA 查看依赖树
![](https://img-blog.csdnimg.cn/img_convert/3fdbb2e4a8a8d0c398642b34153ca902.png)
![](https://img-blog.csdnimg.cn/img_convert/4b796b8aa508588eec2284163c77b07c.png)
5.9 解决遇到的问题
Maven 3.6.2
解决方法:降级为 3.6.1
![](https://img-blog.csdnimg.cn/img_convert/82bcfc766ea2bf2dd30f0d52cd41cec8.png)
Tomcat 闪退
IDEA中每次都要重复配置Maven
在IDEA中的全局默认配置中去配置
![](https://img-blog.csdnimg.cn/img_convert/9058edfd6d489320374094eed4e49c47.png)
![](https://img-blog.csdnimg.cn/img_convert/96f68b7c5babd2baa7300af5504abb6a.png)
Maven 默认 Web 项目中的 web.xml 版本问题
替换为 webapp4.0 版本和 Tomcat 一致
<?xml version="1.0" encoding="UTF-8"?><web-appxmlns="http://xmlns.jcp.org/xml/ns/javaee"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee
http://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsd"version="4.0"metadata-complete="true"></web-app>1234567891011
6. Servlet 简介
Servlet 就是 Sun 公司开发动态 Web 的一门技术
Sun 在这些 API (Application Programming Interface,应用程序接口)中提供一个接口叫做:Servlet,如果你想开发一个Servlet程序,只需要完成两个小步骤:
编写一个类,实现Servlet接口。
把开发好的Java类部署到web服务器中。
把实现了 Servlet 接口的 Java 程序叫做,Servlet
6.2 HelloServlet
Serlvet 接口 Sun 公司有两个默认的实现类:HttpServlet,GenericServlet
构建一个普通的Maven项目,删掉里面的src目录,以后我们的学习就在这个项目里面建立Moudel;这个空的工程就是Maven主工程(建一个WebApp Maven项目,勾选模板);
关于Maven父子工程的理解:
父项目中会有:
<modules><module>servlet-01</module></modules>123
子项目中会有:
<parent><artifactId>javaweb-02-servlet</artifactId><groupId>com.kuang</groupId><version>1.0-SNAPSHOT</version></parent>12345
父项目中的 Java 子项目可以直接使用
son extends father
1
Maven 环境优化
修改 web.xml 为最新的
将 maven 的结构搭建完整
![](https://img-blog.csdnimg.cn/img_convert/b1b642e58e4b6cae10fc476f2d5165b6.png)
4.编写一个Servlet程序
![](https://img-blog.csdnimg.cn/img_convert/16cbee386c3e0f7f6bf95f3acf13ae92.png)
S1:编写一个普通类
S2:实现 Servlet 接口,这里我们直接继承 HttpServlet
publicclassHelloServletextendsHttpServlet{//由于get或者post只是请求实现的不同的方式,可以相互调用,业务逻辑都一样;@OverrideprotectedvoiddoGet(HttpServletRequest req, HttpServletResponse resp)throws ServletException, IOException {//ServletOutputStream outputStream = resp.getOutputStream();
PrintWriter writer = resp.getWriter();//响应流
writer.print("Hello,Serlvet");}@OverrideprotectedvoiddoPost(HttpServletRequest req, HttpServletResponse resp)throws ServletException, IOException {doGet(req, resp);}}123456789101112131415
编写 Servlet 的映射
为什么需要映射:我们写的是 Java 程序,但是要通过浏览器访问,而浏览器需要连接 Web 服务器,所以我们需要再 Web 服务中注册我们写的 Servlet,还需给他一个浏览器能够访问的路径;
<!--注册Servlet--><servlet><servlet-name>hello</servlet-name><servlet-class>com.kuang.servlet.HelloServlet</servlet-class></servlet><!--Servlet的请求路径--><servlet-mapping><servlet-name>hello</servlet-name><url-pattern>/hello</url-pattern></servlet-mapping>1234567891011
配置 Tomcat
点击编辑,+ 号,选择本地的 Tomcat
![](https://img-blog.csdnimg.cn/img_convert/376e279f4b897bc1295fed066e03b261.png)
启动测试!
6.3 Servlet 原理
Servlet 是由 Web 服务器调用,Web 服务器在收到浏览器请求之后,会:
![](https://img-blog.csdnimg.cn/img_convert/f184518afd76c0e8cab1c93ce8dba301.png)
6.4 Mapping 问题
一个 Servlet 可以指定一个映射路径
<servlet-mapping><servlet-name>hello</servlet-name><url-pattern>/hello</url-pattern></servlet-mapping>1234
一个 Servlet 可以指定一个映射路径
<servlet-mapping><servlet-name>hello</servlet-name><url-pattern>/hello</url-pattern></servlet-mapping><servlet-mapping><servlet-name>hello</servlet-name><url-pattern>/hello2</url-pattern></servlet-mapping><servlet-mapping><servlet-name>hello</servlet-name><url-pattern>/hello3</url-pattern></servlet-mapping><servlet-mapping><servlet-name>hello</servlet-name><url-pattern>/hello4</url-pattern></servlet-mapping><servlet-mapping><servlet-name>hello</servlet-name><url-pattern>/hello5</url-pattern></servlet-mapping>1234567891011121314151617181920
一个Servlet可以指定通用映射路径
<servlet-mapping><servlet-name>hello</servlet-name><url-pattern>/hello/*</url-pattern></servlet-mapping>1234
默认请求路径
<!--默认请求路径--><servlet-mapping><servlet-name>hello</servlet-name><url-pattern>/*</url-pattern></servlet-mapping>12345
指定一些后缀或者前缀等等….
<!--可以自定义后缀实现请求映射
注意点,*前面不能加项目映射的路径
hello/sajdlkajda.qinjiang
--><servlet-mapping><servlet-name>hello</servlet-name><url-pattern>*.qinjiang</url-pattern></servlet-mapping>12345678
优先级问题
指定了固有的映射路径优先级最高,如果找不到就会走默认的处理请求;
<!--404--><servlet><servlet-name>error</servlet-name><servlet-class>com.kuang.servlet.ErrorServlet</servlet-class></servlet><servlet-mapping><servlet-name>error</servlet-name><url-pattern>/*</url-pattern></servlet-mapping>123456789
6.5 ServletContext
Web 容器在启动的时候,它会为每个 Web 程序都创建一个对应的 ServletContext 对象,它代表了当前的 Web 应用;
6.5.1 共享数据
在一个 servlet 中保存的数据,可以在另一个 servlet 中拿到;
publicclassHelloServletextendsHttpServlet{@OverrideprotectedvoiddoGet(HttpServletRequest req, HttpServletResponse resp)throws ServletException, IOException {//this.getInitParameter() 初始化参数//this.getServletConfig() Servlet配置//this.getServletContext() Servlet上下文
ServletContext context =this.getServletContext();
String username ="秦疆";//数据
context.setAttribute("username",username);//将一个数据保存在了ServletContext中,名字为:username 。值 username}}123456789101112131415
publicclassGetServletextendsHttpServlet{@OverrideprotectedvoiddoGet(HttpServletRequest req, HttpServletResponse resp)throws ServletException, IOException {
ServletContext context =this.getServletContext();
String username =(String) context.getAttribute("username");
resp.setContentType("text/html");
resp.setCharacterEncoding("utf-8");
resp.getWriter().print("名字"+username);}@OverrideprotectedvoiddoPost(HttpServletRequest req, HttpServletResponse resp)throws ServletException, IOException {doGet(req, resp);}}1234567891011121314151617
<servlet><servlet-name>hello</servlet-name><servlet-class>com.kuang.servlet.HelloServlet</servlet-class></servlet><servlet-mapping><servlet-name>hello</servlet-name><url-pattern>/hello</url-pattern></servlet-mapping><servlet><servlet-name>getc</servlet-name><servlet-class>com.kuang.servlet.GetServlet</servlet-class></servlet><servlet-mapping><servlet-name>getc</servlet-name><url-pattern>/getc</url-pattern></servlet-mapping>123456789101112131415161718
测试访问结果;
6.5.2 获取初始化参数
<!--配置一些web应用初始化参数--><context-param><param-name>url</param-name><param-value>jdbc:mysql://localhost:3306/mybatis</param-value></context-param>12345
protectedvoiddoGet(HttpServletRequest req, HttpServletResponse resp)throws ServletException, IOException {
ServletContext context =this.getServletContext();
String url = context.getInitParameter("url");
resp.getWriter().print(url);}12345
6.5.3 请求转发
@OverrideprotectedvoiddoGet(HttpServletRequest req, HttpServletResponse resp)throws ServletException, IOException {
ServletContext context =this.getServletContext();
System.out.println("进入了ServletDemo04");//RequestDispatcher requestDispatcher = context.getRequestDispatcher("/gp"); //转发的请求路径//requestDispatcher.forward(req,resp); //调用forward实现请求转发;
context.getRequestDispatcher("/gp").forward(req,resp);}12345678
![](https://img-blog.csdnimg.cn/img_convert/e7ec76d7656d147c39b68ee7a2316b33.png)
6.5.4 读取资源文件
Properties
在 java 目录下新建 properties
在 resources 目录下新建 properties
发现:都被打包到了同一个路径下:classes,我们俗称这个路径为classpath:
思路:需要一个文件流;
uername=sjmp
password=123456
12
publicclassServletDemo05extendsHttpServlet{@OverrideprotectedvoiddoGet(HttpServletRequest req, HttpServletResponse resp)throws ServletException, IOException {
InputStream is =this.getServletContext().getResourceAsStream("/WEB-INF/classes/com/kuang/servlet/aa.properties");
Properties prop =newProperties();
prop.load(is);
String user = prop.getProperty("username");
String pwd = prop.getProperty("password");
resp.getWriter().print(user+":"+pwd);}@OverrideprotectedvoiddoPost(HttpServletRequest req, HttpServletResponse resp)throws ServletException, IOException {doGet(req, resp);}}1234567891011121314151617181920
6.6 HttpServletResponse
Web 服务器接收到客户端的 http 请求,针对这个请求,分别创建一个代表请求的 HttpServletRequest 对象,代表响应的一个 HttpServletResponse;
如果想要获取客户端请求过来的参数:找 HttpServletRequest
如果要给客户端响应一些信息:找 HttpServletResponse
6.6.1 简单分类
负责向浏览器发送数据的方法
ServletOutputStream getOutputStream()throws IOException;
PrintWriter getWriter()throws IOException;12
负责向浏览器发送响应头的方法
voidsetCharacterEncoding(String var1);voidsetContentLength(int var1);voidsetContentLengthLong(long var1);voidsetContentType(String var1);voidsetDateHeader(String var1,long var2);voidaddDateHeader(String var1,long var2);voidsetHeader(String var1, String var2);voidaddHeader(String var1, String var2);voidsetIntHeader(String var1,int var2);voidaddIntHeader(String var1,int var2);12345678910111213141516171819
响应的状态码
int SC_CONTINUE =100;int SC_SWITCHING_PROTOCOLS =101;int SC_OK =200;int SC_CREATED =201;int SC_ACCEPTED =202;int SC_NON_AUTHORITATIVE_INFORMATION =203;int SC_NO_CONTENT =204;int SC_RESET_CONTENT =205;int SC_PARTIAL_CONTENT =206;int SC_MULTIPLE_CHOICES =300;int SC_MOVED_PERMANENTLY =301;int SC_MOVED_TEMPORARILY =302;int SC_FOUND =302;int SC_SEE_OTHER =303;int SC_NOT_MODIFIED =304;int SC_USE_PROXY =305;int SC_TEMPORARY_REDIRECT =307;int SC_BAD_REQUEST =400;int SC_UNAUTHORIZED =401;int SC_PAYMENT_REQUIRED =402;int SC_FORBIDDEN =403;int SC_NOT_FOUND =404;int SC_METHOD_NOT_ALLOWED =405;int SC_NOT_ACCEPTABLE =406;int SC_PROXY_AUTHENTICATION_REQUIRED =407;int SC_REQUEST_TIMEOUT =408;int SC_CONFLICT =409;int SC_GONE =410;int SC_LENGTH_REQUIRED =411;int SC_PRECONDITION_FAILED =412;int SC_REQUEST_ENTITY_TOO_LARGE =413;int SC_REQUEST_URI_TOO_LONG =414;int SC_UNSUPPORTED_MEDIA_TYPE =415;int SC_REQUESTED_RANGE_NOT_SATISFIABLE =416;int SC_EXPECTATION_FAILED =417;int SC_INTERNAL_SERVER_ERROR =500;int SC_NOT_IMPLEMENTED =501;int SC_BAD_GATEWAY =502;int SC_SERVICE_UNAVAILABLE =503;int SC_GATEWAY_TIMEOUT =504;int SC_HTTP_VERSION_NOT_SUPPORTED =505;1234567891011121314151617181920212223242526272829303132333435363738394041
6.6.2 下载文件
给浏览器输出消息
下载文件
1.要获取下载文件的路径
2.下载的文件名是什么?
3.设置浏览器使其支持下载的内容
4.获取下载文件的输入流
5.创建缓冲区
6.获取 OutputStream 对象
7.将 FileOutputStream 写入到 buffer缓冲区
8.使用 OutputStream 将缓冲区中的数据输出到客户端
@OverrideprotectedvoiddoGet(HttpServletRequest req, HttpServletResponse resp)throws ServletException, IOException {// 1. 要获取下载文件的路径
String realPath ="F:\\班级管理\\西开【19525】\\2、代码\\JavaWeb\\javaweb-02-servlet\\response\\target\\classes\\秦疆.png";
System.out.println("下载文件的路径:"+realPath);// 2. 下载的文件名是啥?
String fileName = realPath.substring(realPath.lastIndexOf("\\")+1);// 3. 设置想办法让浏览器能够支持(Content-Disposition)下载我们需要的东西,中文文件名URLEncoder.encode编码,否则有可能乱码
resp.setHeader("Content-Disposition","attachment;filename="+URLEncoder.encode(fileName,"UTF-8"));// 4. 获取下载文件的输入流
FileInputStream in =newFileInputStream(realPath);// 5. 创建缓冲区int len =0;byte[] buffer =newbyte[1024];// 6. 获取OutputStream对象
ServletOutputStream out = resp.getOutputStream();// 7. 将FileOutputStream流写入到buffer缓冲区,使用OutputStream将缓冲区中的数据输出到客户端!while((len=in.read(buffer))>0){
out.write(buffer,0,len);}
in.close();
out.close();}123456789101112131415161718192021222324
6.6.3 实现重定向
![](https://img-blog.csdnimg.cn/img_convert/697758dd8e3542541b166c117b2de1ea.png)
Web 资源 B 收到客户端 A 请求后,通知 A 访问另一个 Web 资源 C ,这个过程叫做重定向
常见场景:
用户登录
voidsendRedirect(String var1)throws IOException;1
@OverrideprotectedvoiddoGet(HttpServletRequest req, HttpServletResponse resp)throws ServletException, IOException {/*
resp.setHeader("Location","/r/img");
resp.setStatus(302);
*/
resp.sendRedirect("/r/img");//重定向}123456789
面试题:重定向与转发的区别?
相同点:
页面都会实现跳转
不同点:
请求转发的时候,URL 不会发生变化
重定向时候,URL 地址栏会发生变化;
![](https://img-blog.csdnimg.cn/img_convert/890c555de0aa788c58c6ffc1d0378d89.png)
6.6.4 简单实现登录重定向
<%--这里提交的路径,需要寻找到项目的路径--%>
<%--${pageContext.request.contextPath}代表当前的项目--%>
<formaction="${pageContext.request.contextPath}/login"method="get">
用户名:<inputtype="text"name="username"><br>
密码:<inputtype="password"name="password"><br><inputtype="submit"></form>12345678
@OverrideprotectedvoiddoGet(HttpServletRequest req, HttpServletResponse resp)throws ServletException, IOException {//处理请求
String username = req.getParameter("username");
String password = req.getParameter("password");
System.out.println(username+":"+password);//重定向时候一定要注意,路径问题,否则404;
resp.sendRedirect("/r/success.jsp");}123456789101112
<servlet><servlet-name>requset</servlet-name><servlet-class>com.kuang.servlet.RequestTest</servlet-class></servlet><servlet-mapping><servlet-name>requset</servlet-name><url-pattern>/login</url-pattern></servlet-mapping>12345678
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html><head><title>Title</title></head><body><h1>Success</h1></body></html>1234567891011
6.7 HttpServletRequest 获取参数,请求转发
HttpServletRequest 代表客户端的请求,用户通过 HTTP 协议访问服务器,HTTP 请求中的所有信息会被封装到 HttpServletRequest ,通过这个HttpServletRequest 的方法,获得客户端的所有信息;
![](https://img-blog.csdnimg.cn/img_convert/14c13ee3abbeb1fc26420cf3bdd386f8.png)
![](https://img-blog.csdnimg.cn/img_convert/d911329fa9b4784ca5a88a3628983be5.png)
获取参数,请求转发
![](https://img-blog.csdnimg.cn/img_convert/636b6823cd0544754e92656018073900.png)
@OverrideprotectedvoiddoGet(HttpServletRequest req, HttpServletResponse resp)throws ServletException, IOException {
req.setCharacterEncoding("utf-8");
resp.setCharacterEncoding("utf-8");
String username = req.getParameter("username");
String password = req.getParameter("password");
String[] hobbys = req.getParameterValues("hobbys");
System.out.println("=============================");//后台接收中文乱码问题
System.out.println(username);
System.out.println(password);
System.out.println(Arrays.toString(hobbys));
System.out.println("=============================");
System.out.println(req.getContextPath());//通过请求转发//这里的 / 代表当前的web应用
req.getRequestDispatcher("/success.jsp").forward(req,resp);}1234567891011121314151617181920212223
7 Cookie、Session
7.1 会话
会话:用户打开一个浏览器,点击了很多超链接,访问多个 Web 资源,关闭浏览器,这个过程可以称之为会话;
有状态会话:一个同学来过教室,下次再来教室,我们会知道这个同学,曾经来过,称之为有状态会话;
你能怎么证明你是学生?
你 学校
发票 学校开的发票
学校登记 校牌
你 | 学校 |
发票 | 学校开的发票 |
学校的登记 | 学校给你的校牌 |
一个网站,怎么证明你来过?
客户端 服务端
服务端给客户端一个 信件,客户端下次访问服务端带上信件就可以了; cookie
客户端 : 服务端给客户端一个 信件,客户端下次访问服务端带上信件就可以了; cookie
服务端:服务器登记你来过了,下次你来的时候我来匹配你; seesion
7.2 保存会话的两种技术
cookie
客户端技术(响应,请求)
session
服务器技术,利用这个技术,可以保存用户的会话信息?可将信息或数据放在 session 中
7.3 Cookie
![](https://img-blog.csdnimg.cn/img_convert/0be2aa5813c9f530ed68ef9b00d00352.png)
从请求中拿到 cookie 信息
服务器响应给客户端 cookie
Cookie[] cookies = req.getCookies();//获得Cookie
cookie.getName();//获得cookie中的key
cookie.getValue();//获得cookie中的vlauenewCookie("lastLoginTime", System.currentTimeMillis()+"");//新建一个cookie
cookie.setMaxAge(24*60*60);//设置cookie的有效期
resp.addCookie(cookie);//响应给客户端一个cookie123456
cookie : 一般会保存在本地的用户目录下 appdata ;
一个 Web 站点可以给浏览器发送多个 Cookie,最多存放 20 个 cookie;
cookie 大小有限制 4kb;
300 个 cookie 浏览器上限
删除 Cookie;
不设置有效期,关闭浏览器,自动失效;
设置有效期为 0;
编码解码:
URLEncoder.encode("秦疆","utf-8")
URLDecoder.decode(cookie.getValue(),"UTF-8")
12
7.4 Session(重点)
![](https://img-blog.csdnimg.cn/img_convert/59676437b389779b5069335e967e0776.png)
什么是 Session ?
服务器会给每一个用户(浏览器)创建一个 Session 对象;
一个 Session 独占一个浏览器,只要浏览器没有关闭,这个 Session 就存在;
用户登录之后,整个网站它都可以访问!-> 保存用户的信息;保存购物车的信息…
![](https://img-blog.csdnimg.cn/img_convert/485701645489949d44f1903be05b0a43.png)
Session 和 cookie 的区别:
Cookie是把用户的数据写给用户的浏览器,浏览器保存 (可以保存多个)
Session把用户的数据写到用户独占Session中,服务器端保存 (保存重要的信息,减少服务器资源的浪费)
Session 对象由服务创建;
使用场景:
保存一个登录用户的信息;
购物车信息;
在整个网站中经常会使用的数据,我们将它保存在 Session 中;
使用 Session:
package com.kuang.servlet;import com.kuang.pojo.Person;import javax.servlet.ServletException;import javax.servlet.http.*;import java.io.IOException;publicclassSessionDemo01extendsHttpServlet{@OverrideprotectedvoiddoGet(HttpServletRequest req, HttpServletResponse resp)throws ServletException, IOException {//解决乱码问题
req.setCharacterEncoding("UTF-8");
resp.setCharacterEncoding("UTF-8");
resp.setContentType("text/html;charset=utf-8");//得到Session
HttpSession session = req.getSession();//给Session中存东西
session.setAttribute("name",newPerson("秦疆",1));//获取Session的ID
String sessionId = session.getId();//判断Session是不是新创建if(session.isNew()){
resp.getWriter().write("session创建成功,ID:"+sessionId);}else{
resp.getWriter().write("session以及在服务器中存在了,ID:"+sessionId);}//Session创建的时候做了什么事情;// Cookie cookie = new Cookie("JSESSIONID",sessionId);// resp.addCookie(cookie);}@OverrideprotectedvoiddoPost(HttpServletRequest req, HttpServletResponse resp)throws ServletException, IOException {doGet(req, resp);}}//得到Session
HttpSession session = req.getSession();
Person person =(Person) session.getAttribute("name");
System.out.println(person.toString());
HttpSession session = req.getSession();
session.removeAttribute("name");//手动注销Session
session.invalidate();123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354
会话自动过期:web.xml 配置
<!--设置Session默认的失效时间--><session-config><!--15分钟后Session自动失效,以分钟为单位--><session-timeout>15</session-timeout></session-config>12345
![](https://img-blog.csdnimg.cn/img_convert/903d0934cfee9f14c3f832260d7f3a45.png)
8 JSP
8.1 什么 JSP
Java Server Pages : Java服务器端页面,也和Servlet一样,用于动态Web技术!
最大的特点:
写JSP就像在写HTML
区别:
HTML只给用户提供静态的数据
JSP 页面中可以嵌入Java 代码,为用户提供动态数据;
8.2 JSP 原理
服务器内部工作:
Tomcat 中有一个 work 工作目录;
IDEA 中使用 Tomcat 的会在 IDEA 中 Tomcat 中生产一个 work 目录
![](https://img-blog.csdnimg.cn/img_convert/4f8c5ae2ff3996bc75dc4c4f00314eb2.png)
C:\Users\Administrator\.IntelliJIdea2018.1\system\tomcat\Unnamed_javaweb-session-cookie\work\Catalina\localhost\ROOT\org\apache\jsp
1
发现页面转变成了 Java 程序
![](https://img-blog.csdnimg.cn/img_convert/cc72b7b903df6fd63d1cfb1b87fa7491.png)
浏览器向服务器发送请求,不管访问什么资源,起始都是在访问 Servlet !
JSP 最终也会被转换成一个 Java 类!
JSP 本质上就是一个 Servlet
//初始化publicvoid_jspInit(){}//销毁publicvoid_jspDestroy(){}//JSPServicepublicvoid_jspService(.HttpServletRequest request,HttpServletResponse response)123456789
判断请求
内置一些对象
final javax.servlet.jsp.PageContext pageContext;//页面上下文
javax.servlet.http.HttpSession session = null;//sessionfinal javax.servlet.ServletContext application;//applicationContextfinal javax.servlet.ServletConfig config;//config
javax.servlet.jsp.JspWriter out = null;//outfinal java.lang.Object page =this;//page:当前
HttpServletRequest request //请求
HttpServletResponse response //响应12345678
输出页面前增加的代码
response.setContentType("text/html");//设置响应的页面类型
pageContext = _jspxFactory.getPageContext(this, request, response,
null,true,8192,true);
_jspx_page_context = pageContext;
application = pageContext.getServletContext();
config = pageContext.getServletConfig();
session = pageContext.getSession();
out = pageContext.getOut();
_jspx_out = out;123456789
4.以上这些对象可直接在 JSP 中使用
![](https://img-blog.csdnimg.cn/img_convert/cfe7a8dc27f64ff60705fe56b81040cd.png)
在JSP页面中;
只要是 JAVA代码就会原封不动的输出;
如果是HTML代码,就会被转换为:
out.write("<html>\r\n");
1
这样的格式,输出到前端!
8.3 JSP 基础语法
JSP 表达式
<%--JSP表达式
作用:用来将程序的输出,输出到客户端
<%= 变量或者表达式%>
--%>
<%= new java.util.Date()%>
12345
JSP 脚本片段
<%--jsp脚本片段--%>
<%
int sum = 0;
for (int i = 1; i <=100 ; i++) {
sum+=i;
}
out.println("<h1>Sum="+sum+"</h1>");
%>
12345678
脚本片段的再实现
<%
int x = 10;
out.println(x);
%>
<p>这是一个JSP文档</p>
<%
int y = 2;
out.println(y);
%>
<hr>
<%--在代码嵌入HTML元素--%>
<%
for (int i = 0; i < 5; i++) {
%>
<h1>Hello,World <%=i%> </h1>
<%
}
%>
123456789101112131415161718192021
JSP 声明
<%!
static {
System.out.println("Loading Servlet!");
}
private int globalVar = 0;
public void kuang(){
System.out.println("进入了方法Kuang!");
}
%>
1234567891011
JSP 声明: 会被编译到 JSP 生成 Java 的类中! 其他的,就会被生成 _jspService 方法中!
<%%>
<%=%>
<%!%>
<%--注释--%>
12345
JSP 的注释,不会在客户端显示,HTML就会!
8.4 JSP 指令
<%@page args.... %>
<%@include file=""%>
<%--@include会将两个页面合二为一--%>
<%@include file="common/header.jsp"%>
<h1>网页主体</h1>
<%@include file="common/footer.jsp"%>
<hr>
<%--jSP标签
jsp:include:拼接页面,本质还是三个
--%>
<jsp:includepage="/common/header.jsp"/><h1>网页主体</h1><jsp:includepage="/common/footer.jsp"/>12345678910111213141516171819
8.5 九大内置对象
PageContext 存东西
Request 存东西
Response
Session 存东西
Application 【ServletContext】 存东西
config 【ServletConfig】
out
page
exception
pageContext.setAttribute("name1","秦疆1号");//保存的数据只在一个页面中有效
request.setAttribute("name2","秦疆2号");//保存的数据只在一次请求中有效,请求转发会携带这个数据
session.setAttribute("name3","秦疆3号");//保存的数据只在一次会话中有效,从打开浏览器到关闭浏览器
application.setAttribute("name4","秦疆4号");//保存的数据只在服务器中有效,从打开服务器到关闭服务器1234
request:客户端向服务器发送请求,产生的数据,用户看完就没用了,比如:新闻,用户看完没用的!
session:客户端向服务器发送请求,产生的数据,用户用完一会还有用,比如:购物车;
application:客户端向服务器发送请求,产生的数据,一个用户用完了,其他用户还可能使用,比如:聊天数据;
8.6 JSP标签、JSTL标签、EL表达式
<!-- JSTL表达式的依赖 --><dependency><groupId>javax.servlet.jsp.jstl</groupId><artifactId>jstl-api</artifactId><version>1.2</version></dependency><!-- standard标签库 --><dependency><groupId>taglibs</groupId><artifactId>standard</artifactId><version>1.1.2</version></dependency>123456789101112
EL 表达式: ${}
获取数据
执行运算
获取 Web 开发的常用对象
JSP 标签
<%--jsp:include--%>
<%--
http://localhost:8080/jsptag.jsp?name=kuangshen&age=12
--%>
<jsp:forwardpage="/jsptag2.jsp"><jsp:paramname="name"value="kuangshen"></jsp:param><jsp:paramname="age"value="12"></jsp:param></jsp:forward>12345678910
JSTL表达式
JSTL 标签库的使用就是为了弥补 HTML 标签的不足;它自定义许多标签,可以供我们使用,标签的功能和 Java 代码一样!
格式化标签
SQL标签
XML 标签
核心标签 (掌握部分)
![](https://img-blog.csdnimg.cn/img_convert/a681a43804495b886b6b2c512c40d8ee.png)
JSTL标签库使用步骤
引入对应的 taglib
使用其中的方法
在 Tomcat 也需要引入 JSTL 的包,否则会报错:JSTL 解析错误
c: if
<head><title>Title</title></head><body><h4>if测试</h4><hr><formaction="coreif.jsp"method="get">
<%--
EL表达式获取表单中的数据
${param.参数名}
--%>
<inputtype="text"name="username"value="${param.username}"><inputtype="submit"value="登录"></form>
<%--判断如果提交的用户名是管理员,则登录成功--%>
<c:iftest="${param.username=='admin'}"var="isAdmin"><c:outvalue="管理员欢迎您!"/></c:if>
<%--自闭合标签--%>
<c:outvalue="${isAdmin}"/></body>12345678910111213141516171819202122232425262728
c:choose c:when
<body>
<%--定义一个变量score,值为85--%>
<c:setvar="score"value="55"/><c:choose><c:whentest="${score>=90}">
你的成绩为优秀
</c:when><c:whentest="${score>=80}">
你的成绩为一般
</c:when><c:whentest="${score>=70}">
你的成绩为良好
</c:when><c:whentest="${score<=60}">
你的成绩为不及格
</c:when></c:choose></body>123456789101112131415161718192021
c:forEach
<%
ArrayList<String> people = new ArrayList<>();
people.add(0,"张三");
people.add(1,"李四");
people.add(2,"王五");
people.add(3,"赵六");
people.add(4,"田六");
request.setAttribute("list",people);
%>
<%--
var , 每一次遍历出来的变量
items, 要遍历的对象
begin, 哪里开始
end, 到哪里
step, 步长
--%>
<c:forEachvar="people"items="${list}"><c:outvalue="${people}"/><br></c:forEach><hr><c:forEachvar="people"items="${list}"begin="1"end="3"step="1"><c:outvalue="${people}"/><br></c:forEach>12345678910111213141516171819202122232425262728
9 JavaBean
实体类
JavaBean有特定的写法:
必须要有一个无参构造
属性必须私有化
必须有对应的get/set方法;
一般用来和数据库的字段做映射 ORM;
ORM :对象关系映射
表—>类
字段–>属性
行记录---->对象
id | name | age | address |
1 | sjmp01 | 3 | 西安 |
2 | sjmp02 | 18 | 西安 |
3 | sjmp03 | 100 | 川 |
classPeople{privateint id;private String name;privateint id;private String address;}classA{newPeople(1,"秦疆1号",3,"西安");newPeople(2,"秦疆2号",3,"西安");newPeople(3,"秦疆3号",3,"西安");}123456789101112
过滤器实现登录拦截功能,放一个常量标记是否已登录。
10 MVC 三层架构
什么是 MVC : Model View Controller 模型、视图、控制器
10.1 以前
![](https://img-blog.csdnimg.cn/img_convert/a4ffc5b7e188059b20dfdc61023f968b.png)
用户直接访问控制层,控制层就可以直接操作数据库;
servlet--CRUD-->数据库
弊端:程序十分臃肿,不利于维护
servlet的代码中:处理请求、响应、视图跳转、处理JDBC、处理业务代码、处理逻辑代码
架构:没有什么是加一层解决不了的!
程序猿调用
|
JDBC
|
Mysql Oracle SqlServer ....
12345678910
10.2 MVC 三层架构
![](https://img-blog.csdnimg.cn/img_convert/c091a22fe5dfd7b6ee948cd50a22880e.png)
Model
业务处理:业务逻辑(Service)
数据持久层:CRUD
View
展示数据
提供链接发起 Servlet 请求(a,form,img…)
Controller (Servlet)
接收用户的请求:(req:请求参数、Session 信息…)
交给业务层处理对应的代码
控制试图的跳转
登录--->接收用户的登录请求--->处理用户的请求(获取用户登录的参数,username,password)---->交给业务层处理登录业务(判断用户名密码是否正确:事务)--->Dao层查询用户名和密码是否正确-->数据库
1
11 Filter(重点)
Filter:过滤器,用来过滤网站的数据;
处理中文乱码
登录验证
![](https://img-blog.csdnimg.cn/img_convert/ec9b0843e0b68387a839925e1a444f10.png)
Filter 开发步骤:
导包
编写过滤器
导包不要错
![](https://img-blog.csdnimg.cn/img_convert/49fde31949e961f03235385917625bf2.png)
publicclassCharacterEncodingFilterimplementsFilter{//初始化:web服务器启动,就以及初始化了,随时等待过滤对象出现!publicvoidinit(FilterConfig filterConfig)throws ServletException {
System.out.println("CharacterEncodingFilter初始化");}//Chain : 链/*
1. 过滤中的所有代码,在过滤特定请求的时候都会执行
2. 必须要让过滤器继续同行
chain.doFilter(request,response);
*/publicvoiddoFilter(ServletRequest request, ServletResponse response, FilterChain chain)throws IOException, ServletException {
request.setCharacterEncoding("utf-8");
response.setCharacterEncoding("utf-8");
response.setContentType("text/html;charset=UTF-8");
System.out.println("CharacterEncodingFilter执行前....");
chain.doFilter(request,response);//让我们的请求继续走,如果不写,程序到这里就被拦截停止!
System.out.println("CharacterEncodingFilter执行后....");}//销毁:web服务器关闭的时候,过滤会销毁publicvoiddestroy(){
System.out.println("CharacterEncodingFilter销毁");}}12345678910111213141516171819202122232425262728
在 web.xml 中配置Filter
<filter><filter-name>CharacterEncodingFilter</filter-name><filter-class>com.kuang.filter.CharacterEncodingFilter</filter-class></filter><filter-mapping><filter-name>CharacterEncodingFilter</filter-name><!--只要是 /servlet的任何请求,会经过这个过滤器--><url-pattern>/servlet/*</url-pattern><!--<url-pattern>/*</url-pattern>--></filter-mapping>12345678910
12 监听器
实现一个监听器;
编写一个监听器
实现监听器的接口…
//统计网站在线人数 : 统计sessionpublicclassOnlineCountListenerimplementsHttpSessionListener{//创建session监听: 看你的一举一动//一旦创建Session就会触发一次这个事件!publicvoidsessionCreated(HttpSessionEvent se){
ServletContext ctx = se.getSession().getServletContext();
System.out.println(se.getSession().getId());
Integer onlineCount =(Integer) ctx.getAttribute("OnlineCount");if(onlineCount==null){
onlineCount =newInteger(1);}else{int count = onlineCount.intValue();
onlineCount =newInteger(count+1);}
ctx.setAttribute("OnlineCount",onlineCount);}//销毁session监听//一旦销毁Session就会触发一次这个事件!publicvoidsessionDestroyed(HttpSessionEvent se){
ServletContext ctx = se.getSession().getServletContext();
Integer onlineCount =(Integer) ctx.getAttribute("OnlineCount");if(onlineCount==null){
onlineCount =newInteger(0);}else{int count = onlineCount.intValue();
onlineCount =newInteger(count-1);}
ctx.setAttribute("OnlineCount",onlineCount);}/*
Session销毁:
1. 手动销毁 getSession().invalidate();
2. 自动销毁
*/}123456789101112131415161718192021222324252627282930313233343536373839404142434445464748
在 web.xml 中注册监听器
<!--注册监听器--><listener><listener-class>com.kuang.listener.OnlineCountListener</listener-class></listener>