Javaweb

文章目录

  • JavaWeb
    • 1、基本概念
      • 1.1、前言
      • 1.2、web应用程序
      • 1.3、静态web
      • 1.4、动态web
    • 2、web服务器
      • 2.1、技术讲解
      • 2.2、web服务器
    • 3、Tomcat
      • 3.1、安装Tomcat
      • 3.2、Tomact启动和配置
      • 3.3、配置
      • **高难度面试题:**
      • 3.4、发布一个web网站
    • 4、http
      • 4.1、什么是http
      • 4.2、两个时代
      • 4.3、http请求
        • 1、请求行
        • 2、消息头
      • 4.4、http响应
        • 1、响应体
        • 2、响应状况码 (重点)
    • 5、maven
      • 5.1、maven架构管理工具
      • 5.2、下载安装maven
      • 5.3、配置环境变量
      • 5.4、阿里云镜像
      • 5.5、本地仓库
      • 5.6、在IDEA中使用Maven
      • 5.7、创建一个普通的Maven项目
      • 5.8、标记文件夹功能
      • 5.9、在IDEA中配置Tomcat
      • 5.10、pom文件
      • 5.12、IDEA操作
      • 5.13、遇到的问题
    • 6、Servlet
      • 6.1、Servlet简介
      • 6.2、HelloServlet
      • 6.3、Servlet原理
      • 6.4、Mapping问题
      • 6.5、ServletContext 获取服务上下文
        • 1、共享数据
        • 2、获取初始化参数
        • 3、请求转发
        • 4.读取资源文件
      • 6.6、HttpServletResponse
        • 1、简单分类
        • 2、常见应用(下载文件)
        • 3、验证码功能
        • 4、实现重定向(掌握)
        • 6.7、HttpServletRequest
        • 1、获取前端传递的参数 并且 请求转发
    • 7、Cookie、Session
      • 7.1、会话
      • 7.2、保存会话的两种技术
      • 7.3、Cookie
      • 7.4、Session(重点)
    • 8、JSP
      • 8.1、什么是JSP
      • 8.2、JSP原理
      • 8.3、JSP基础语法
        • **JSP表达式**
        • **jsp脚本片段**
        • JSP声明
      • 8.4、JSP指令
      • 8.5、9大内置对象
      • 8.6、JSP标签、JSTL标签、EL表达式
    • 9、JavaBean
    • 10、MVC三层架构
      • 10.1、早些年
      • 10.2、MVC三层架构
    • 11、Filter(过滤器 重点)
    • 12、监听器
    • 13、过滤器、监听器常见应用
      • 登录监听功能
      • 思路:步骤!
    • 14、JDBC
      • 什么是JDBC:
      • **实验环境搭建**
      • **JDBC 固定步骤:**
      • Statement
      • PreparedStatement 预编译SQL

Javaweb

JavaWeb

1、基本概念

1.1、前言

web开发:

  • web,网页的意思,www.caidu.com
  • 静态web
    • HTML,css
    • 提供给所有人开的数据始终不会发生变化!
  • 动态web
    • 淘宝,几乎所有的网站;
    • 提供给所有人开的数据始终会发生变化!
    • 技术栈:Servlet/JSP,ASP ,PHP

在java中,动态web资源开发的技术统称为JavaWeb;

1.2、web应用程序

什么是程序:按照自己编写的代码去执行的动作

web应用程序:可以提供浏览器访问的程序;

  • a.html、b.html……多个web资源,这些web资源可以被外部访问,对外界提供服务;

  • 我们能访问的任何一个页面或者资源,都存在于这个世界的某一个角落的计算机上!

  • URL(资源定位符)

  • 这个统一的web资源会被放在同一个文件夹下,web应用程序,依赖于:Tomcat:服务器

  • 一个web应用有多部分组成(静态web,动态web)

    • HTML,css,js
    • jsp文件,servlet程序
    • java程序
    • jar包
    • 配置文件(properti)

    web应用程序编写完毕后,若想提供给外界访问:需要一个服务器来统一管理;

1.3、静态web

  • *.html , *.html这些都是网页的后缀,如果服务器上一直存在这些东西,我们就可以直接进行读取。通络

Javaweb_第1张图片

  • 静态web存在的缺点
    • web页面无法动态更新,所有用户看到都是同一个页面
      • 为了静态页面比较好看:轮播图,点击特效:伪动态
      • JavaScript [实际开发中]
      • VBScript
    • 它无法和数据库交互(数据无法持久化,用户无法交互)

1.4、动态web

页面会动态展示:“Web的页面展示的效果因人而异”;

Javaweb_第2张图片

缺点:

  • 加入服务器的动态web资源出现了错误,我们需要重新编写我们的后台程序,重新发布;
    • 停机维护

优点:

  • web页面可以动态更新,所有用户看到都是不同一个页面
  • 它可以和数据库交互(数据持久化:注册,商品信息,用户信息……)

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-hWtX8XCn-1636270388219)(javaWeb.assets/image-20210531193331954.png)]

新手村: - - 魔鬼训练(分析原理、看源码)- ->PK场

2、web服务器

实现web手段,ASP ,JSP,PHP

2.1、技术讲解

ASP:

  • 微软:国内最早流行的是ASP

  • 在HTML中嵌入了VB的脚本,ASP +COM

  • 在ASP开发中,基本一个页面都有几千行的业务代码,页面极其混乱

  • 维护成本高!

  • 主要用的C#

  • iis服务器

    Javaweb_第3张图片

php:

  • PHP开发速度很快,功能很强大,跨平台,代码很简单(无法承载大访问量的,中国百分之70%网站都是中小型的,开源的模板引擎WP)
  • 无法承载大访问量的情况(局限性)

JSP/Servlet:

JSP本质是 Servlet

B/S:浏览和服务器

C/S:客户端和服务器

  • sun公司主推的B/S架构
  • 基于java语言的(所有的大公司,或者一些开源的组件,都使用java写的)
  • 可以承载三高(高并发,高可用,高性能)问题带来的影响:
  • 语法想ASP, ASP–>JSP,加强市场强度;

……

2.2、web服务器

服务器是一种被动的操作,用来处理用户的一些请求给用户一些响应信息

IIS

微软的;ASP… Windows中自带的

Tomcat

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-A58R7D0B-1636270388222)(javaWeb.assets/b3b7d0a20cf431adfe004e4e4e36acaf2fdd98f2)]

面向百度编程

Tomcat是Apache 软件基金会(Apache Software Foundation)的Jakarta 项目中的一个核心项目,最新的Servlet 和JSP 规范总是能在Tomcat 中得到体现,因为Tomcat 技术先进、性能稳定,而且免费,因而深受Java 爱好者的喜爱并得到了部分软件开发商的认可,成为目前比较流行的Web 应用服务器。

Tomcat 服务器是一个免费的开放源代码的Web 应用服务器,属于轻量级应用服务器,在中小型系统和并发访问用户不是很多的场合下被普遍使用,是开发和调试JSP 程序的首选。对于一个java初学web的人来说,它是最佳的选择

Tomcat 实际上运行JSP 页面和Servlet。目前Tomcat最新版本为10.0.5。

……

工作3-5年之后 ,可以尝试手写Tomcat服务器

下载Tomcat:

  1. 安装 or 解压
  2. 了解配置文件及目录结构! java的配置文件及目录结构
  3. 作用

3、Tomcat

3.1、安装Tomcat

Tomact官网:https://tomcat.apache.org/

Javaweb_第4张图片

3.2、Tomact启动和配置

文件夹信息

Javaweb_第5张图片

启动、关闭 Tomact

Javaweb_第6张图片

访问测试:http://localhost:8080/

可能遇到的问题:

  1. java环境变量没有配置 会闪退 (Tomact依赖于java,是java写的)
  2. 闪退问题:需要配置兼容性
  3. 乱码问题:配置文件中设置

3.3、配置

Javaweb_第7张图片

可以配置启动的端口号

  • Tomcat的默认端口号为:8080
  • MySQL:3306
  • http:80
  • https:443
<Connector port="8080" redirectPort="8443" connectionTimeout="20000" protocol="HTTP/1.1"/>

可以配置主机的名称

  • 默认的主机名为 localhost 等价于 127.0.0.1
  • 默认网站应用存放的位置为:webapps
-<Host name="localhost" autoDeploy="true" unpackWARs="true" appBase="webapps">

<Valve className="org.apache.catalina.valves.AccessLogValve" pattern="%h %l %u %t "%r" %s %b" suffix=".txt" prefix="localhost_access_log" directory="logs"/>

Host>

高难度面试题:

请你谈谈网站是如何进行访问的!(阿里四面,真题)

  1. 输入一个域名;回车

  2. 检查本机的 C:\Windows\System32\drivers\etc**\hosts** 配置文件下有没有这个域名映射;

    1. 有:直接返回对应的ip地址,这个地址中,有我们需要访问的web程序,可以直接访问

      127.0.0.1      www.xiaoming.com
      
    2. 没有:去DNS服务器找,找到的话就返回,找不到就返回找不到

      Javaweb_第8张图片

  3. 可以配置一个环境变量(可选)

3.4、发布一个web网站

不会就模仿

  • 将自己写的网站,放在服务器(Tomact)中指定的web应用的文件夹(webapps)下,就可以访问了
-- webapps // Tomcat服务器的web目录
  -POOT // 项目的名字
  -study // 网站的目录名
  	- WEB-INF // 放网站程序的
  		-classes // java程序
  		-lib  // web 应用所依赖的jar包
  		-web.xml
   	- index.html // 默认的首页
  	- static // 可以放资源文件
  		-css
  			-style.css
  		-js
  		-img
  	-......

4、http

4.1、什么是http

超文本传输协议(Hypertext Transfer Protocol,HTTP)是一个简单的请求-响应协议,它通常运行在TCP之上。

  • 文本:HTML,字符串 ,~……
  • 超文本:图片,音乐,视频,定位,地图……
  • 默认端口:80

Https:安全的

  • 默认端口号:443

4.2、两个时代

  • http1.0
    • HTTP/1.0:客户端可以与web服务器连接后,只能获得一个web资源,断开连接
  • http2.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: 39.156.66.14:443 // 远程地址 ,真实访问走的这个地址
// Referrer Policy: strict-origin-when-cross-origin  // 协议 能访问到什么东西
请求 URL: https://www.baidu.com/
请求方法: GET
状态代码: 200 OK
远程地址: 39.156.66.14:443
引用站点策略: strict-origin-when-cross-origin

请求标头

Accept:text/html   // 类型
Accept-Encoding: gzip, deflate, br  // 类型的编码
Accept-Language: zh-CN,zh;q=0.9  //  类型的语言
Cache-Control: max-age=0 	// 缓存控制
Connection: keep-alive // 连接

1、请求行

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-kFp7h7pP-1636270388227)(javaWeb.assets/image-20210609185530425.png)]

  • 请求行中的方式: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: 主机...

4.4、http响应

  • 服务器 - - -> 响应 - - - -> 给客户端

百度

响应头

Cache-Control: private  								缓存-控制
Connection: keep-alive   								连接:保持连接
Content-Encoding: gzip   								内容-编码 
Content-Type: text/html;charset=utf-8   内容-类型 html

1、响应体

Accept: 告诉浏览器,他所支持的数据类型
Accept-Encoding: 支持哪种编码格式  GBK  UTF-8  GB2312  ISO8859-1
Accept-Language: 告诉浏览器,它的语言环境
Cache-Control: 缓存控制
Connection: 告诉浏览器,请求完成是断开还是保持连接
host: 主机...
refresh: 告诉客户端,多久刷新一次
Location: 让网页重新定义;

2、响应状况码 (重点)

200:请求响应成功 200

3XX:请求重定向 303

  • 重定向:你重新到我给你新位置去;

4XX:找不到资源 404

  • 资源不存在;

5XX:服务器代码错误 500 505:网关错误

Javaweb_第9张图片

常见面试题

当你的浏览器中地址栏输入地址并回车的一瞬间到页面能够展示回来,经历了什么?

5、maven

为什么要学习这个技术?

  1. 在Javaweb开发中,需要使用大量的jar包,手动去导入

  2. 如何能够让一个东西自动帮我导入和配置这个jar包。

    由此,maven诞生了!

5.1、maven架构管理工具

目前用它来方便导入jar包的

maven的核心思想:约定大于配置

  • 有约束,不要去违反。

maven会规定好你如何去编写java代码,必须按照这个来;

5.2、下载安装maven

官网:http://maven.apache.org/download.cgi

Javaweb_第10张图片

建议:电脑上的所有环境都放在一个文件夹下,方便管理

5.3、配置环境变量

在体统环境变量中

配置如下:

  • M2_HOME maven目录下的bin目录
  • MAVEN_HOME maven的目录
  • 在系统的path中配置 %MAVEN_HOME%\bin

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-fr5yJeyE-1636270388229)(javaWeb.assets/image-20210609203817053.png)]

测试Maven是否安装成功,保证必须配置完毕!

5.4、阿里云镜像

  • 镜像:mirrors
    • 作用:加速我们的下载
  • 国内建议使用阿里云镜像

    alimaven
    central
    aliyun maven
    http://maven.aliyun.com/nexus/content/groups/public/

5.5、本地仓库

在本地的仓库,远程仓库;

**建立一个本地仓库:**localRepository


D:\学习资料\MarkDown学习\javaWeb\apache-maven-3.8.1-bin\apache-maven-3.8.1\maven-repo

5.6、在IDEA中使用Maven

  1. 启动IDEA

  2. 创建一个MavenWeb项目

    Javaweb_第11张图片

    Javaweb_第12张图片

    Javaweb_第13张图片

    Javaweb_第14张图片

    Javaweb_第15张图片

    Javaweb_第16张图片

  3. 等待项目初始化完毕(自动导入包)

  1. 观察Maven仓库中多了什么东西

    Javaweb_第17张图片

  2. IDEA中的Maven设置

    注意: IDEA项目创建成功后,看一眼Maven的配置,

    Javaweb_第18张图片

  3. 到这里,Maven在IEDA中的配置和使用就OK了

    Javaweb_第19张图片

5.7、创建一个普通的Maven项目

Javaweb_第20张图片

Javaweb_第21张图片

只有在web应用下才会有!

Javaweb_第22张图片

5.8、标记文件夹功能

Javaweb_第23张图片

Javaweb_第24张图片

Javaweb_第25张图片

Javaweb_第26张图片

Javaweb_第27张图片

5.9、在IDEA中配置Tomcat

Javaweb_第28张图片

Javaweb_第29张图片

Javaweb_第30张图片

Javaweb_第31张图片

Javaweb_第32张图片

Javaweb_第33张图片

Javaweb_第34张图片

解决警告问题

必须要的配置: 为什么会有这个问题:我们访问一个网站,需要指定一个文件夹名字

Javaweb_第35张图片

Javaweb_第36张图片

Javaweb_第37张图片

Javaweb_第38张图片

5.10、pom文件

pom.xml 是Maven的核心配置文件

Javaweb_第39张图片




<project xmlns="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.0modelVersion>

  
  <groupId>com.kuanggroupId>
  <artifactId>javaweb-01-mavenartifactId>
  <version>1.0-SNAPSHOTversion>
  
  <packaging>warpackaging>

  
  <name>javaweb-01-maven Maven Webappname>
  
  <url>http://www.example.comurl>

  
  <properties>
    
    <project.build.sourceEncoding>UTF-8project.build.sourceEncoding>
    
    <maven.compiler.source>1.7maven.compiler.source>
    <maven.compiler.target>1.7maven.compiler.target>
  properties>

  
  <dependencies>
    
    <dependency>
      <groupId>junitgroupId>
      <artifactId>junitartifactId>
      <version>4.11version>
      <scope>testscope>
    dependency>
  dependencies>

  
  <build>
    <finalName>javaweb-01-mavenfinalName>
    <pluginManagement>
      <plugins>
        <plugin>
          <artifactId>maven-clean-pluginartifactId>
          <version>3.1.0version>
        plugin>
        
        <plugin>
          <artifactId>maven-resources-pluginartifactId>
          <version>3.0.2version>
        plugin>
        <plugin>
          <artifactId>maven-compiler-pluginartifactId>
          <version>3.8.0version>
        plugin>
        <plugin>
          <artifactId>maven-surefire-pluginartifactId>
          <version>2.22.1version>
        plugin>
        <plugin>
          <artifactId>maven-war-pluginartifactId>
          <version>3.2.2version>
        plugin>
        <plugin>
          <artifactId>maven-install-pluginartifactId>
          <version>2.5.2version>
        plugin>
        <plugin>
          <artifactId>maven-deploy-pluginartifactId>
          <version>2.8.2version>
        plugin>
      plugins>
    pluginManagement>
  build>
project>

Javaweb_第40张图片

maven由于他的约定大于配置,我们之后可以能遇到我们写的配置文件,无法被导出或者生效的问题,解决方案:

Javaweb_第41张图片

5.12、IDEA操作

Javaweb_第42张图片

Javaweb_第43张图片

Javaweb_第44张图片

5.13、遇到的问题

  1. Maven3.6.2 版本问题:建一个项目,所有依赖无法导入,导入会报一个错,

错误:[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-fUb7lZrz-1636270388255)(javaWeb.assets/image-20210611162829166.png)]

解决方案:更新上一个版本

  1. Tomcat闪退

  2. IDEA中每次都要重复配置Maven

    在IDEA中的全局默认配置中去配置

    Javaweb_第45张图片

    Javaweb_第46张图片

  3. Maven项目中Tomcat没法配置

  4. maven默认web项目的web.xml版本问题

    Javaweb_第47张图片

  5. 替换为webapp 的版本和Tomcat一致

    <web-app xmlns="https://jakarta.ee/xml/ns/jakartaee"         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"         xsi:schemaLocation="https://jakarta.ee/xml/ns/jakartaee                      https://jakarta.ee/xml/ns/jakartaee/web-app_5_0.xsd"         version="5.0"         metadata-complete="true">  <display-name>Welcome to Tomcatdisplay-name>  <description>    Welcome to Tomcat  description>web-app>
    
  6. Maven仓库的使用

    地址 :https://mvnrepository.com/

    Javaweb_第48张图片

Javaweb_第49张图片

Javaweb_第50张图片

Javaweb_第51张图片

6、Servlet

6.1、Servlet简介

  • Servlet就是sun公司开发动态web的一门技术

    • 什么是动态web:每个人去访问同一个网页,看到的内容不一样,因为程序可以生产不一样的内容
  • Sun在这些API中提供一个接口叫做:Servlet,如果你想开发一个Servlet程序,只需要完成两个小步骤:

    • 编写一个类,实现Servlet接口
    • 把开发好的Java类部署到web服务器中。

什么是Servlet:把实现了Servlet接口的Java程序叫做,Servlet

servlet的本质是什么,它是如何工作的?

参考知乎:https://www.zhihu.com/question/21416727/answer/690289895

6.2、HelloServlet

Serlvet接口在Sun公司有两个默认的实现类:HttpServlet,GenericServlet

  1. 构建一个普通的Maven项目,删掉里面的src目录,以后我们学习就在这个项目里面建立Model;这个空的工程就是Maven主工程;

  2. 关于Maven父子工程的理解:

    父项目中会有

   <modules> 
    <module>servlet-01module>
   modules>

子项目中会有

Javaweb_第52张图片

注意:maven3.6.0刚开始有parent后面就没有了,变成了自动继承了,

  1. Maven环境优化

    1. 修改web.xml为最新的

      Javaweb_第53张图片

    2. 将maven的结构搭建完整

      Javaweb_第54张图片

  2. 编写一个Servlet程序

    1. 编写一个普通类

    2. 实现Servlet接口,在这里直接继承 HttpServlet

package com.kuang.servlet;

import javax.servlet.ServletException;
import javax.servlet.ServletOutputStream;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.PrintWriter;

/**
 * 第一个Servlet
 *
 * @author <[email protected]>
 * @since 2021/6/13 9:04
 */
public class HelloServlet extends HttpServlet {
     

//    重写方法

//    由于get或者post只是请求实现的不同的方式,可以相互调用,因为业务逻辑都一样

    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
     
//        代码在这里面写

//        输入
//        获得输入流
//        req.getInputStream()


//        输出
//        IO流分两种:1、resp.getWriter(); 2、resp.getOutputStream()
//        ServletOutputStream outputStream = resp.getOutputStream();
        PrintWriter writer = resp.getWriter();//响应流

        writer.print("HelloServlet");
			// 一个类搞定了      
    }

    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
     
        doGet(req, resp);
    }
}

分析源码:

Javaweb_第55张图片

  1. 编写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>
  1. 配置Tomact

    Javaweb_第56张图片

    注意:配置项目发布的路径

    Javaweb_第57张图片

  2. 启动测试

6.3、Servlet原理

Servlet是由Web服务器调用,web服务器在收到 浏览器请求之后,会:

Javaweb_第58张图片

Javaweb_第59张图片

Javaweb_第60张图片

6.4、Mapping问题

Mapping:它是请求映射的一个路径

  1. 一个Servlet可以指定一个映射路径
   
   
   <servlet-mapping>  
   <servlet-name>Helloservlet-name> 
    <url-pattern>/hellourl-pattern>
    servlet-mapping>
  1. 一个Servlet可以指定多个映射路径


<servlet-mapping>
  <servlet-name>helloservlet-name>
  <url-pattern>/hellourl-pattern>
servlet-mapping>

<servlet-mapping>
  <servlet-name>helloservlet-name>
  <url-pattern>/hello1url-pattern>
servlet-mapping>

<servlet-mapping>
  <servlet-name>helloservlet-name>
  <url-pattern>/hello2url-pattern>
servlet-mapping>

<servlet-mapping>
  <servlet-name>helloservlet-name>
  <url-pattern>/hello3url-pattern>
servlet-mapping>

<servlet-mapping>
  <servlet-name>helloservlet-name>
  <url-pattern>/hello4url-pattern>
servlet-mapping>
  1. 一个Servlet可以指定通用映射路径

   <servlet-mapping> 
    <servlet-name>helloservlet-name>
      
        <url-pattern>/hello/*url-pattern>
servlet-mapping>
  1. 默认请求路径

    <servlet-mapping> 
     <servlet-name>helloservlet-name> 
      <url-pattern>/*url-pattern>
 servlet-mapping>
  1. 指定一些后缀或者前缀等等……
  [自定义]

<servlet-mapping>
  <servlet-name>helloservlet-name>
  <url-pattern>*.yinurl-pattern>
servlet-mapping>
  1. 优先级问题

    指定了固有的映射路径优先级最高,如果找不到就会走默认的处理请求



<servlet>
  <servlet-name>errorservlet-name>
  <servlet-class>com.kuang.servlet.ErrorServletservlet-class>
servlet>
<servlet-mapping>
  <servlet-name>errorservlet-name>
  <url-pattern>/*url-pattern>
servlet-mapping>

通过这种方式去处理一些错误的请求

package com.kuang.servlet;

import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.PrintWriter;

/**
 * 请求,专门来处理404页面找不到
 *
 * @author <[email protected]>
 * @since 2021/6/13 11:31
 */
public class ErrorServlet extends HttpServlet {
     

    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
     

//        输出
//        以HTML展示 ,并且编码为 utf-8
        resp.setContentType("text/html");
        resp.setCharacterEncoding("utf-8");

//        展示完给他个输出对象
        PrintWriter writer = resp.getWriter();
        writer.print("

404

"
); } @Override protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { doGet(req, resp); } }

解决乱码问题:

  • 打开 Tomact文件夹 —–> conf文件夹 ——> logging.properties 文件
  java.util.logging.ConsoleHandler.encoding = utf-8   
  改为:java.util.logging.ConsoleHandler.encoding = GBK

6.5、ServletContext 获取服务上下文

web容器在启动的时候,它会为每一web程序都创建一个对应的ServletContext对象,他代表了当前的web应用

1、共享数据

我在这个Servlet中保存的数据,可以在另外一个servlet中拿到;

Javaweb_第61张图片

  • 首先需要设置一个放数据的类
package com.kunag.servlet;

import javax.servlet.ServletContext;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;


public class HelloServlet extends HttpServlet {
     
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
     

//       this.getInitParameter() 获得初始化的参数   初始化参数
//          this.getServletConfig() 获取服务配置    Servlet配置
//          this.getServletContext() 获取服务上下文  Servlet上下文
        ServletContext servletContext = this.getServletContext();

        String username = "小尹";// 数据
        servletContext.setAttribute("username",username);//将一个数据保存在了ServletContext中,名字为:username,值 username

    }
}

  • 还需要一个去读取它的类
public class GetServlet extends HttpServlet {
     
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
     
        ServletContext servletContext = this.getServletContext();

        String username = (String) servletContext.getAttribute("username");

        resp.setContentType("text/html");
        resp.setCharacterEncoding("utf-8");
        resp.getWriter().print("名字"+username);
    }

    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
     
        doGet(req, resp);
    }
}

  • 还需要配置一个web.xml
<servlet>
  <servlet-name>helloservlet-name>
  <servlet-class>com.kunag.servlet.HelloServletservlet-class>
servlet>
<servlet-mapping>
  <servlet-name>helloservlet-name>
  <url-pattern>/hellourl-pattern>
servlet-mapping>

<servlet>
  <servlet-name>getcservlet-name>
  <servlet-class>com.kunag.servlet.GetServletservlet-class>
servlet>
<servlet-mapping>
  <servlet-name>getcservlet-name>
  <url-pattern>/getcurl-pattern>
servlet-mapping>
  • 测试访问结果:

Javaweb_第62张图片

2、获取初始化参数

  1. 在web.xml中配置
   
   <context-param>
     <param-name>urlparam-name>
     <param-value>jdbc:mysql://localhost:3306/mabatisparam-value>
   context-param>
  1. 在ServletDemo03类中拿到它
public class ServletDemo03 extends HttpServlet {
     
  @Override
  protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
     
    ServletContext servletContext = this.getServletContext();

    //        获取参数
    String url = servletContext.getInitParameter("url");
    resp.getWriter().print(url);


  }

  @Override
  protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
     
    doGet(req, resp);
  }
}

3、请求转发

Javaweb_第63张图片

  1. 创建一个ServletDemo04类 请求转发

public class ServletDemo04 extends HttpServlet {
     
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
     
        ServletContext servletContext = this.getServletContext();
        System.out.println("进入了ServletDemo04");
//        请求转发
//        /:代表当前项目,转发到当前的gp请求   forward:转发
        /*RequestDispatcher requestDispatcher = servletContext.getRequestDispatcher("/gp");// 转发的请求路径
        requestDispatcher.forward(req,resp);// 调用 forward 方法实现请求转发*/
//        两句话合并成一句话, 不用去写返回值
        servletContext.getRequestDispatcher("/jp").forward(req,resp);
    }

    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
     
        doGet(req, resp);
    }
}

  1. 在web.xml中配置
<servlet>
  <servlet-name>sd4servlet-name>
  <servlet-class>com.kunag.servlet.ServletDemo04servlet-class>
servlet>
<servlet-mapping>
  <servlet-name>sd4servlet-name>
  <url-pattern>/sd4url-pattern>
servlet-mapping>

4.读取资源文件

Properties类

  • 在java目录下新建properties
  • 在resources目录下新建properties

Javaweb_第64张图片

发现:都被打包到了同一个路径下:classes,俗称这个路径为classpath;类路径

Javaweb_第65张图片

6.6、HttpServletResponse

响应:

web服务器接收到客户端的http请求,针对这个请求,分别创建一个代表请求的HttpServletRequest对象,代表响应的一个HttpServletResponse;

  • 如果要获得客户端请求过来的参数:找HttpServletRequest
  • 如果要给客户端响应一些信息:找HttpServletResponse

1、简单分类

负责向浏览器发送数据的方法

ServletOutputStream getOutputStream() throws IOException;
// 写中文用 getWriter() 写平常的流用 getOutputStream()
PrintWriter getWriter() throws IOException;

负责向浏览器发送响应头的方法

void setCharacterEncoding(String var1);

void setContentLength(int var1);

void setContentLengthLong(long var1);

void setContentType(String var1);


void setDateHeader(String var1, long var2);

void addDateHeader(String var1, long var2);

void setHeader(String var1, String var2);

void addHeader(String var1, String var2);

void setIntHeader(String var1, int var2);

void addIntHeader(String var1, int var2);

响应的状态码

    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;

2、常见应用(下载文件)

  1. 向浏览器输出消息
//ServletInputStream inputStream = req.getInputStream();// 获得输入流
//IO流分为两种
//ServletOutputStream outputStream = resp.getOutputStream(); //有输入一定有这两个东西
PrintWriter writer = resp.getWriter(); // 响应流 向外面写东西
writer.print("Hello,Serlvet");// 往外写东西
  1. 下载文件

    1. 要获取下载文件的路径
    2. 下载的文件名是啥?
    3. 设置想办法让浏览器能够支持下载我们需要的东西
    4. 获取下载文件的输入流
    5. 输出:创建缓冲区
    6. 获取OutputStream对象
    7. 将FileOuputStream流写入到buffer缓冲区
    8. 使用OutputStream将缓冲区中的数据输出到客户端!
package com.kuang;

import javax.servlet.ServletException;
import javax.servlet.ServletOutputStream;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.FileInputStream;
import java.io.IOException;
import java.net.URLEncoder;

/**
 * 下载文件
 * @author <[email protected]>
 * @since 2021/6/14 14:41
 */
public class servlet extends HttpServlet {
     
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
     
//        1. 要获取下载文件的路径
        String realPath = "D:\\IDEA数据\\javaweb\\javaweb-02-01-servlet\\response\\src\\main\\resources\\云.jpg";
//        D:\IDEA数据\javaweb\javaweb-02-01-servlet\response\src\main\resources\云.jpg
        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 = new FileInputStream(re alPath);
//        5. 输出:创建缓冲区
        int len = 0;
        byte[] buffer = new byte[1024];
//        6. 获取OutputStream对象
        ServletOutputStream out = resp.getOutputStream();
//        7. 将FileOuputStream流写入到buffer缓冲区
        while ((len=in.read(buffer))>0){
     
            out.write(buffer,0,len);
        }
//        8. 使用OutputStream将缓冲区中的数据输出到客户端!
        in.close();
        out.close();
    }

    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
     
        doGet(req, resp);
    }
}

Javaweb_第66张图片

3、验证码功能

验证怎么来的?

  • 前端实现, JavaScript
  • 后端实现 ,java ,需要用到 Java 的图片类,生成一个图片

新建一个类

package com.kuang;

import javax.imageio.ImageIO;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.awt.*;
import java.awt.image.BufferedImage;
import java.io.BufferedInputStream;
import java.io.IOException;
import java.util.Random;

/**
 * 验证码功能
 * @author <[email protected]>
 * @since 2021/6/14 15:36
 */
public class ImageServlet extends HttpServlet {
     
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
     

        // 如何让浏览器3秒自动刷新一次
        // refresh 让当前函数刷新
        resp.setHeader("refresh","3");

        // 在内存中创建一个图片
        BufferedImage image = new BufferedImage(80,20,BufferedImage.TYPE_INT_RGB);
        // 得到图片
        //在图片上写东西  2D的笔
        Graphics2D g = (Graphics2D) image.getGraphics(); // 笔
        // 设置图片的背景颜色
            //白色的背景
        g.setColor(Color.white);
            //背景颜色
        g.fillRect(0,0,80,20);
        // 给图片写数据
        g.setColor(Color.blue);
        // 字体
        g.setFont(new Font(null,Font.BOLD,20));
        // 写东西
        g.drawString(makeNum(),0,20);

        /*告诉浏览器,这个请求用图片的方式打开*/
        resp.setContentType("image/jpg");
        //网站存在缓存 ,不然浏览器缓存

        // 设置它得头类型 让他为-1不缓存
        resp.setDateHeader("expires",-1);
        //告诉浏览器 Cache-Control:缓存控制  no-cache :不缓存
        resp.setHeader("Cache-Control","no-cache");

        resp.setHeader("Pragma","no-cache");

        //把图片写给浏览器
            // ImageIO是图片类,专门写图片的
        ImageIO.write(image,"jpg",resp.getOutputStream());

    }

    // 生成随机数
    private String makeNum(){
     
        // 创建随机类
        Random random = new Random();
        // 生成一个随机数
        String num = random.nextInt(9999999) + "";
        // 随机数往外边写 StringBuffer可变长字符串
        StringBuffer sb = new StringBuffer();
        // 遍历
        for (int i = 0; i < 7-num.length(); i++) {
     
            // 追加: append
            sb.append("0");
        }
        num = sb.toString() + num;
        return num;
    }

    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
     
        doGet(req, resp);
    }
}

在xml下配置

<servlet>
  <servlet-name>ImageServletservlet-name>
  <servlet-class>com.kuang.ImageServletservlet-class>
servlet>
<servlet-mapping>
  <servlet-name>ImageServletservlet-name>
  <url-pattern>/imgurl-pattern>
servlet-mapping>

Javaweb_第67张图片

4、实现重定向(掌握)

Javaweb_第68张图片

一个web资源(B)收到客户端(A)请求后,它(B)会通知客户端(A)去访问另外一个web资源(C),这个过程叫做重点向

常见场景:

  • 用户登录(用户首先访问登录页面,登录成功跳转到另一个页面)

方法

void sendRedirect(String var1) throws IOException;

测试

package com.kuang;

public class RedirectServlet extends HttpServlet {
     
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
     

        // 发送一个请求
        /*
        原理
        resp.setHeader("Location","/r/img");
        resp.setStatus(302);
        * */

        resp.sendRedirect("/r/img"); // 重定向
    }

    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
     
        doGet(req, resp);
    }
}

<servlet>
  <servlet-name>RedirectServletservlet-name>
  <servlet-class>com.kuang.RedirectServletservlet-class>
servlet>
<servlet-mapping>
  <servlet-name>RedirectServletservlet-name>
<url-pattern>/redurl-pattern>

面试题:重定向和转发的区别?

相同点

  • 页面都会实现跳转

不同点

  • 请求转发的时候,URL不会产生变化
  • 重定向时候,URL地址栏会发生变化

Javaweb_第69张图片

重定向跳转

package com.kuang;

import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;

/**
 * @author <[email protected]>
 * @since 2021/6/14 16:58
 */
public class RequestTest extends HttpServlet {
     
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
     
        System.out.println("进入这个请求");

        // 处理请求 从请求里面获取一个参数
        String username = req.getParameter("username");
        String password = req.getParameter("password");


        System.out.println(username+":"+password);
        // 重定向 跳转  注意路径问题,否则404没找到页面 500 代表代码错了
        resp.sendRedirect("/r/success.jsp");
    }

    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
     
        doGet(req, resp);
    }
}

配置xml

<servlet>
  <servlet-name>requestservlet-name>
  <servlet-class>com.kuang.RequestTestservlet-class>
servlet>
<servlet-mapping>
  <servlet-name>requestservlet-name>
  <url-pattern>/loginurl-pattern> 
servlet-mapping>

jsp跳转前



Hello World!

<%--这里提交的路径,需要寻找到项目的路径--%> <%--${pageContext.request.contextPath}代表 :当前项目--%>
用户名:
密码:
<%--提交表单--%>

jsp先带入jar包

jsp跳转后

<%--
  Created by IntelliJ IDEA.
  User: Lenovo
  Date: 2021/6/14
  Time: 17:52
  To change this template use File | Settings | File Templates.
--%>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>


    Title



Success

6.7、HttpServletRequest

Javaweb_第70张图片

Javaweb_第71张图片

重点在request中有两个应用场景

1、获取前端传递的参数 并且 请求转发

Javaweb_第72张图片

package com.kuang.servlet;

import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.util.Arrays;

/**
 * @author <[email protected]>
 * @since 2021/6/14 20:05
 */
public class LoginServlet extends HttpServlet {
     


    @Override
    protected void doGet(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("=====================");



      //方式一、resp.sendRedirect("/r/img"); // 重定向

        //方式二、this.getServletContext() 转发
        //方式三、通过请求转发
        //通过请求转发
        //说明这里的 / 代表当前的web应用
        req.getRequestDispatcher(req.getContextPath()+"/success.jsp").forward(req,resp);

    }

    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
     
        doGet(req, resp);
    }
}

xml文件

<web-app xmlns="https://jakarta.ee/xml/ns/jakartaee"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="https://jakarta.ee/xml/ns/jakartaee
                      https://jakarta.ee/xml/ns/jakartaee/web-app_5_0.xsd"
         version="5.0"
         metadata-complete="true">
    <servlet>
        <servlet-name>LoginServletservlet-name>
        <servlet-class>com.kuang.servlet.LoginServletservlet-class>
    servlet>
    <servlet-mapping>
        <servlet-name>LoginServletservlet-name>
        <url-pattern>/loginurl-pattern>
    servlet-mapping>

web-app>

转发前的jsp代码

<%--
  Created by IntelliJ IDEA.
  User: Lenovo
  Date: 2021/6/14
  Time: 20:03
  To change this template use File | Settings | File Templates.
--%>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>


    登录


登录

<%--action 请求的地址 ${pageContext.request.contextPath}/login :代表外部项目,下面有了login --%> <%--表单的意思:以post方式提交表单,提交到我们的login请求--%>
用户名:
密码:
爱好: 女孩 代码 唱歌 电影 <%--提交--%>

转发后的jsp代码

<%--
  Created by IntelliJ IDEA.
  User: Lenovo
  Date: 2021/6/14
  Time: 20:11
  To change this template use File | Settings | File Templates.
--%>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>


    Title


成功

请求转发和重定向的区别

相同点

  • 页面都会实现跳转

不同点

  • 请求转发的时候,URL不会产生变化 编码307
  • 重定向时候,URL地址栏会发生变化 编码302

7、Cookie、Session

7.1、会话

**会话:**用户打开一个浏览器,点击了很多超链接,访问多个web资源,关闭浏览器,这个过程可以称之为会话。

**有状态会话:**一个同学来过教室,下次再来教室,我们会知道这个同学,曾经来过,称之为有状态会话;(记录了某些东西)

现实生活中:你能怎么证明你是学生?

你 学校

  1. 发票 学校发票
  2. 学籍 学校登记 学校标记你来过了

网络中:一个网站,怎么证明你来过?

客户端 服务端

  1. 服务端给客户端一个 信件 ,客户端下次访问服务端带上信件就可以了; cookie(饼干)
  2. 服务器登记你来过了,下次你来的时候我来匹配你;seesion

7.2、保存会话的两种技术

cookie

  • 客户端技术 (通过响应发给服务端,客户端通过请求带到服务器上)

session

  • 服务器技术 ,利用这个技术,可以保存用户的会话信息 。我们可以把信息或者数据放在Session中!

常见场景:网站登录之后,下次不用再登录了,第二次访问直接就进去了

7.3、Cookie

一个网站,怎么证明你来过?

Javaweb_第73张图片

Javaweb_第74张图片

  1. 从请求中拿到cookie信息
  2. 服务器响应给客户端cookie

Cookie方法

Cookie[] cookies = req.getCookies();//获得cookie
cookie.getName();//获得cookie中的key
cookie.getValue();//获得cookie中的值
new Cookie("lastLoginTime", System.currentTimeMillis()+"");//新建一个cookie
cookie.setMaxAge(24*60*60);//设置cookie的有效期
resp.addCookie(cookie);//响应给客户端一个cookie

cookie:一般会保存在本地的 用户目录下 appdata;

问题:一个网站cookie是否存在上限!细节问题

  • 一个Cookie只能保存一个信息;
  • 一个web站点可以给浏览器发送多个cookie,每个站点最多存放20个cookie
  • cookie大小有限制 4kb
  • 300个cookie浏览器上限

删除cookie

  • 不设置有效期,关闭浏览器,自动失效;
  • 设置有效期时间为0;

编码解码:

URLEncoder.encode("小明","utf-8")  //编码
URLDecoder.decode(cookie.getValue(),"UTF-8"); // 解码

代码

保存用户上一次访问的时间

package com.kuang.servlet;

import javafx.scene.chart.PieChart;

import javax.servlet.ServletException;
import javax.servlet.http.Cookie;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.PrintWriter;
import java.util.Date;

/**
 * 保存用户上一次访问的时间
 * @author <[email protected]>
 * @since 2021/6/15 16:57
 */
public class CookieDemo01 extends HttpServlet {
     
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
     
        //服务器,告诉你,你来的时间,把这个时间封装成为一个 信件,你下次带来,我就知道你来了

        //解决中文乱码问题
        req.setCharacterEncoding("utf-8");
        resp.setCharacterEncoding("utf-8");

        //客户端返回一些字符串
        // 需要输出对象  响应
        PrintWriter out = resp.getWriter();

        //Cookie ,服务器端从客户端获取
        Cookie[] cookies = req.getCookies();//这里返回数组,说明Cookie可能存在多个

        //判断,Cookie是否存在
        if (cookies!=null){
     
            //如果存在怎么办? 遍历数组
            out.write("你上一次访问的时间是:");

            /*for (Cookie cookie : cookies) {
                
            }*/
                //两种区别 ,下边的这个更灵活,可以取到每一个的下标
            for (int i = 0; i < cookies.length; i++) {
     
                Cookie cookie = cookies[i];
                //获取cookie名字
                if (cookie.getName().equals("lastLoginTime")){
     
                    //如果拿到的名字跟他一样
                    //获取cookie中的值 判断是否一样
                    /*cookie.getValue();*/
                    //解析为长整型,在解析为字符串      把字符串变成一个时间戳了
                    long lastLoginTime = Long.parseLong(cookie.getValue());
                    //把时间戳变成一个对象  常用类知识
                    Date date = new Date(lastLoginTime);
                    //输出             本地时间格式化
                    out.write(date.toLocaleString());

                }

            }

        }else {
     
            out.write("这是您第一次访问本站");
        }

        //服务器给客户端响应(发送)创建一个cookie                          //获取时间
        Cookie cookie = new Cookie("lastLoginTime", System.currentTimeMillis()+"");

        //设置cookie的有效期 1天
        cookie.setMaxAge(24*60*60);

        //添加一个cookie, 响应给客户端
        resp.addCookie(cookie);

    }

    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
     
        doGet(req, resp);
    }
}

删除上一次访问时间

package com.kuang.servlet;

import javax.servlet.ServletException;
import javax.servlet.http.Cookie;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.PrintWriter;
import java.util.Date;

/**
 * 删除上一次访问时间
 * @author <[email protected]>
 * @since 2021/6/15 16:57
 */
public class CookieDemo02 extends HttpServlet {
     
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
     

        //创建一个cookie
        Cookie cookie = new Cookie("lastLoginTime", System.currentTimeMillis()+"");
        //设置有效期 ,立马过期
        cookie.setMaxAge(0);
        //响应给客户端
        resp.addCookie(cookie);


    }

    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
     
        doGet(req, resp);
    }
}

中文数据传递

package com.kuang.servlet;

import javax.servlet.ServletException;
import javax.servlet.http.Cookie;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.PrintWriter;
import java.net.URLDecoder;
import java.net.URLEncoder;
import java.util.Date;

/**
 * 中文数据传递
 *
 * @author <[email protected]>
 * @since 2021/6/15 19:51
 */
public class CookieDemo03 extends HttpServlet {
     
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
     

        //解决请求和响应的中文乱码问题
        req.setCharacterEncoding("utf-8");
        resp.setCharacterEncoding("utf-8");

        //输出

        // 需要输出对象  响应
        PrintWriter out = resp.getWriter();

        //Cookie ,服务器端从客户端获取
        Cookie[] cookies = req.getCookies();//这里返回数组,说明Cookie可能存在多个

        //判断,Cookie是否存在
        if (cookies!=null){
     
            //如果存在怎么办? 遍历数组 取出数据
            out.write("你上一次访问的时间是:");

            /*for (Cookie cookie : cookies) {

            }*/
            //两种区别 ,下边的这个更灵活,可以取到每一个的下标
            for (int i = 0; i < cookies.length; i++) {
     
                Cookie cookie = cookies[i];
                //获取cookie名字
                if (cookie.getName().equals("lastLoginTime")){
     

                    /*解码*/
                    out.write(URLDecoder.decode(cookie.getValue(),"utf-8"));


                }

            }

        }else {
     
            out.write("这是您第一次访问本站");
        }


        // 需要编码和解码        最好用URLEncoder.encode() 编码
         Cookie cookie = new Cookie("name", URLEncoder.encode("小明","utf-8"));
        // 取出数据
        out.write(cookie.getValue());
        // 响应给客户端
        resp.addCookie(cookie);

    }

    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
     
        doGet(req, resp);
    }
}

Javaweb_第75张图片

7.4、Session(重点)

一个网站,怎么证明你来过?

Javaweb_第76张图片

Javaweb_第77张图片

什么是Session:

  • 服务器会给每一个用户(浏览器)创建一个Session对象;
  • 一个Session独占一个浏览器,只有浏览器没有关闭,这个Session就存在;
  • 用户登录之后,整个网站它都可以访问! —–> 保存用户的信息;保存购物车的信息……

常用的Session方法

Javaweb_第78张图片

Session 和 Cookie 区别:

  • Cookie是把用户的数据写给用户的浏览器,浏览器保存 (可以保存多个)
  • Session把用户的数据写到用户独占Session中,(Session每个浏览器都不一样的,针对一个用户来的),服务器端保存(保存重要的信息,减少服务器资源的浪费)
  • Session对象由服务器创建;

使用场景:

  • 保存一个登陆用户的信息;
  • 保存一个购物车信息;
  • 在整个网站中经常会使用的数据,我们将它保存在Session中;(不用每次去get了,在session中放了,每次去取就可以了,不用去new一个新的)

使用Session:

package com.kuang.servlet;

import com.kuang.pojo.Person;

import javax.servlet.ServletException;
import javax.servlet.http.*;
import java.io.IOException;

/**
 * Session
 * @author <[email protected]>
 * @since 2021/6/16 14:48
 */
public class SessionDemo01 extends HttpServlet {
     

    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
     

        /* 1、解决乱码问题*/
        req.setCharacterEncoding("utf-8");
        resp.setCharacterEncoding("utf-8");
        // 响应回去是个HTML页面    设置浏览器响应的格式
        resp.setContentType("text/html;charset=utf-8");

        /*2、得到Session*/
        // 从请求里面拿!
        HttpSession session = req.getSession();

        /*3、 给Session中存东西 ~ */
        session.setAttribute("name",new Person("小尹",1));

        /*方法*/
        // 获取Session的ID
        String Sessionid = session.getId();
        // 判断Session是不是新创建的
        if (session.isNew()){
     
                    // write: 往外写出去  print:打印
            resp.getWriter().write("Session创建成功,ID:"+Sessionid);
        }else {
     
            resp.getWriter().write("Session已经在服务器中存在了,ID:"+Sessionid);
        }

        /*Session创建的时候做了什么事情?*/
                         //       键:JSESSIONID  值:Sessionid
//        Cookie cookie = new Cookie("JSESSIONID",Sessionid);
//        // Session在创建的时候做了这样一个cookie,把cookie响应回去了,不然不可能在cookie中看到这个东西
//        resp.addCookie(cookie);



    }

    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
     
        doGet(req, resp);
    }
}

得到Session

package com.kuang.servlet;

import com.kuang.pojo.Person;

import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession;
import java.io.IOException;

/**
 * 取Session
 * @author <[email protected]>
 * @since 2021/6/16 14:48
 */
public class SessionDemo02 extends HttpServlet {
     

    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
     

        /* 1、解决乱码问题*/
        req.setCharacterEncoding("utf-8");
        resp.setCharacterEncoding("utf-8");
        // 响应回去是个HTML页面    设置浏览器响应的格式
        resp.setContentType("text/html;charset=utf-8");

        /*2、得到Session*/
        // 从请求里面拿!
        HttpSession session = req.getSession();

        /*3、获取Session*/
        Person name = (Person) session.getAttribute("name");
        System.out.println(name.toString());
    }

    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
     
        doGet(req, resp);
    }
}

注销

package com.kuang.servlet;

import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession;
import java.io.IOException;

/**
 * 注销 Session
 *
 * @author <[email protected]>
 * @since 2021/6/16 15:53
 */
public class SessionDemo03 extends HttpServlet {
     

    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
     

        // 获得Session
        HttpSession session = req.getSession();
        // 取消
        session.removeAttribute("name");
        //  手动注销 Session
        session.invalidate();
    }

    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
     
        doGet(req, resp);
    }
}

会话自动过期:web.xml中配置


<session-config>
  
  
  
  <session-timeout>1session-timeout>
session-config>

理解!

Javaweb_第79张图片

8、JSP

8.1、什么是JSP

Java Server Pages :Java服务器端页面,也和Servlet一样,用于开发动态Web技术!

最大的特点:

  • 写JSP就像在写HTML
  • 区别:
    • HTML只给用户提供静态的数据
    • JSP页面中可以嵌入Java代码,为用户提供动态数据;

8.2、JSP原理

思路:JSP到底怎么执行的!

  • 代码层面没有任何问题

  • 服务器内部工作

    Tomact中有一个work目录;

    IDEA中使用Tomact的话会在IDEA的Tomcat中生产一个work目录

    Javaweb_第80张图片

Javaweb_第81张图片

地址:C:\Users\Lenovo\AppData\Local\JetBrains\IntelliJIdea2021.1\tomcat

电脑的地址

C:\Users\Lenovo\AppData\Local\JetBrains\IntelliJIdea2021.1
\tomcat\52743147-623a-422a-935b-
ca4218af9099\work\Catalina\localhost\ROOT\org\apache\jsp

页面转变成了 java 程序

Javaweb_第82张图片

浏览器向服务器发送请求,不管访问什么资源,其实都是在访问Servlet!

JSP最终也会被转换成为一个Java类!

JSP本质上就是一个Servlet (继承关系)

源码剖析

// 初始化
public void _jspInit() {
     
}
// 销毁
public void _jspDestroy() {
     
}
// JSPService (JSP的服务)
public void _jspService(HttpServletRequest request, HttpServletResponse response)
// Tomact每个它都要走这个请求服务,会去处理请求和响应,请求和响应可以做事情

请求和响应做的事情

  1. 判断请求
if (!javax.servlet.DispatcherType.ERROR.equals(request.getDispatcherType())) {
     
  final java.lang.String _jspx_method = request.getMethod();
  if ("OPTIONS".equals(_jspx_method)) {
     
    response.setHeader("Allow","GET, HEAD, POST, OPTIONS");
    return;
  }
  if (!"GET".equals(_jspx_method) && !"POST".equals(_jspx_method) && !"HEAD".equals(_jspx_method)) {
     
    response.setHeader("Allow","GET, HEAD, POST, OPTIONS");
    response.sendError(HttpServletResponse.SC_METHOD_NOT_ALLOWED, "JSP 只允许 GET、POST 或 HEAD。Jasper 还允许 OPTIONS");
    return;
  }
}
  1. 内置了一些对象 (记住!)
final javax.servlet.jsp.PageContext pageContext; // pageContext 页面上下文
javax.servlet.http.HttpSession session = null; // session
final javax.servlet.ServletContext application; // applicationContext 它本质:ServletContext  它的作用域非常高
final javax.servlet.ServletConfig config; // config 配置
javax.servlet.jsp.JspWriter out = null; // out 输出
final java.lang.Object page = this; // page 当前页
HttpServletRequest request  // request 请求
HttpServletResponse response // response 响应
  1. 输出页面前增加的代码
response.setContentType("text/html"); // 设置它得页面响应类型: text/html
pageContext = _jspxFactory.getPageContext(this, request, response,
                  null, true, 8192, true);// getPageContext 初始化创造对象
_jspx_page_context = pageContext; // pageContext 给父类一个值
application = pageContext.getServletContext(); // 
config = pageContext.getServletConfig(); // 获取了一个配置对象
session = pageContext.getSession();
out = pageContext.getOut();
_jspx_out = out;
  1. 以上的这些对象我们可以在JSP页面中直接使用!

jsp本质就是servlet,只不过简化了一个东西,只需要写页面部分就可以

JSP原理流程图:

Javaweb_第83张图片

JSP代码


<%--
  Created by IntelliJ IDEA.
  User: Lenovo
  Date: 2021/6/16
  Time: 17:49
  To change this template use File | Settings | File Templates.
--%>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>


    Title



<%
    // java代码
    String name = "小尹";

%>
<%--使用java代码
    输出
--%>
<%--name输出到页面上--%>
name:<%=name%>






Javaweb_第84张图片

源码:

Javaweb_第85张图片

结论:

在JSP页面中;

只要是Java代码就会原封不动的输出;

如果是HTML代码,就会被转换为

out.write("\r\n"); 

这样的格式,输出到前端

jsp帮我们简化页面这个块的编写

8.3、JSP基础语法

1、新建一个普通项目

Javaweb_第86张图片

  • maven项目没有web功能

  • web项目没有普通功能

  • 两个互补

  • maven(普通)项目不支持web,新建一个web

  • [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-7YRmTE2s-1636270388280)(javaWeb.assets/image-20210617202442468.png)]

  • Javaweb_第87张图片

  • Javaweb_第88张图片

  • 建议之前的形式,maven用web项目的比较正规

2、操作

  • 配置xml文件
 <dependencies>
        
        <dependency>
            <groupId>javax.servletgroupId>
            <artifactId>servlet-apiartifactId>
            <version>2.5version>
        dependency>

        
        <dependency>
            <groupId>javax.servlet.jspgroupId>
            <artifactId>javax.servlet.jsp-apiartifactId>
            <version>2.3.1version>
        dependency>

        
        <dependency>
            <groupId>javax.servlet.jsp.jstlgroupId>
            <artifactId>jstl-apiartifactId>
            <version>1.2version>
        dependency>

        
        <dependency>
            <groupId>taglibsgroupId>
            <artifactId>standardartifactId>
            <version>1.1.2version>
        dependency>


    dependencies>

3、新建jsp文件

任何语言都有自己的语法,Java中有。JSP作为java技术的一种应用,它拥有一些自己扩充的语法(了解,知道即可!),java所有语法都支持!

JSP表达式

<%--JSP表达式
  作用:用来将程序的输出,输出到客户端
  公式:<%= 变量或者表达式%>
--%>

<%-- 时间--%>
<%= new java.util.Date()%>

jsp脚本片段

<%--jsp脚本片段--%>
<%
int sun = 0;
for (int i = 1; i <100 ; i++) {
  sun+=i;
}
//输出一句话
out.println("

Sum="+sun+"

"); %>

脚本片段的再实现

<%--脚本片段的再实现--%>
  <%
    int x =10;
    out.println(x);
  %>
  

这是JSP文档

<% int y=20; out.println(y); %>
<%--进阶:在代码嵌入HTML元素--%> <% for (int i = 0; i <5 ; i++) { %>

Hello,word <%= i%>

<% } %>

JSP声明

<%--jsp声明--%>
  <%!
    //static 静态代码块
    static {
      System.out.println("Loading Servlet");
    }
    // 全局变量
    private int globalVar = 0;

    public void kuang(){
      System.out.println("进入了方法Kuang!");
    }
%>

Javaweb_第89张图片

作用域更高了

区别:

JSP声明:会被编译到JSP生成Java的类中!其他的,就会被生成到_jspService方法中!

Javaweb_第90张图片

方法里面的可以调用方法外边的

在JSP中,嵌入Java代码即可!

优化

<%--进阶:在代码嵌入HTML元素--%>
<%--EL表达式--%>
<% for (int i = 0; i <5 ; i++) { %>

Hello,word <%= i%>

<% } %>

总结

<%%> 片段
<%= %> 表达式,输出一个值
<%! %> 定义全局的 

<%--注释--%>

JSP的注释,不会再客户端显示,HTML的注释会! JSP安全性高

8.4、JSP指令

<%@page pageEncoding="utf-8" %>
<%@page page... %>

<%--提取公共页面--%>
<%@include file=""%> include包含一个文件


<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<%--主要页面--%>


    Title



  <%--脚本语言--%>
  <%--file :文件--%>

  <%--@include会将两个页面合二为一--%>
  <%@include file="common/header.jsp"%>
  

网页主体

<%-- <%int i =10; %> 页面报错 栈错误 重名了 --%> <%@include file="common/footer.jsp"%>
<%--JSP标签--%> <%--page:页面--%> <%-- jsp:include 拼接页面 本质还是三个 --%>

网页主体

<%-- <% int i =10; // 直接报错! %>--%>

<%–JSP标签–%>

Javaweb_第91张图片

<%–脚本语言–%>

Javaweb_第92张图片

Javaweb_第93张图片

跳转到500或404页面

  1. 新建一个包存图片

  2. 新建一个404或500.jsp文件

   500
   404
  1. 配置错误的页面

<error-page>
  <error-code>404error-code>
  <location>/error/404.jsplocation>
error-page>

<error-page>
  <error-code>500error-code>
  <location>/error/500.jsplocation>
error-page>

Javaweb_第94张图片

8.5、9大内置对象

  • PageContext 页面的上下文 存东西
  • Request 参数 存东西
  • Response 参数
  • Session 存东西
  • Application【SerlvetContext】 存东西
  • config 【SerlvetConfig】
  • out
  • page 不用了解
  • exception 和java异常一样的
//存东西
pageContext.setAttribute("name1", "小尹1号");//保存的数据只在一个页面中有效
request.setAttribute("name2","小尹2号");     //保存的数据只在一次请求中有效,请求转发会携带这个数据
session.setAttribute("name3","小尹3号");     //保存的数据只在一次会话中有效,从打开浏览器到关闭浏览器
application.setAttribute("name4","小尹4号"); //保存的数据只在服务器中有效,从打开服务器到关闭服务器

request:客户端向服务器发送请求,产生的数据,用户看完了就没用了,比如:新闻,用户看完就没用的了

session:客户端向服务器发送请求,产生的数据,用户看完了还会用,比如:购物车;

application:客户端向服务器发送请求,产生的数据,一个用户用完了,其他用户还可以 使用,比如:聊天数据;

作用域的概念 重点!

Javaweb_第95张图片

8.6、JSP标签、JSTL标签、EL表达式

JSTL标签 需要导包


<dependency>
  <groupId>javax.servlet.jsp.jstlgroupId>
  <artifactId>jstl-apiartifactId>
  <version>1.2version>
dependency>


<dependency>
  <groupId>taglibsgroupId>
  <artifactId>standardartifactId>
  <version>1.1.2version>
dependency>

LE表达式:${ }

  • 获取数据
  • 执行运算
  • 获取web开发的常用对象

jsp标签

<%--标签--%>

<%--包含上下文--%>
<%--


<%--
    相当于转发的时候携带了两个参数
http://localhost:8080/jstag.jsp?name=xiaoyin&age=12
--%>
<%--跳转到那个页面--%>
    <%--参数--%>
    <%--转发的时候是可以写东西的--%>


    
    

JSTL表达式

JSTL标签库的使用就是为了弥补HTML标签的不足;它自定义许多标签,可以供我们使用,标签的功能和java代码一样!

格式化标签

SQL标签

XML 标签

核心标签(掌握部分)

Javaweb_第96张图片

JSTL标签库使用步骤

  • 引入对应的 taglib (标签库)
  • 使用其中的方法
  • 在Tomact 中也需要引入jstl的包,否则会报错:JSTL解析错误

c:if 重点!



if测试

<%-- EL表达式获取表单中的数据 ${param.参数名} --%> <%--提交的按钮--%>
<%--判断如果提交的用户名是管理员,则登录成功--%> <%-- test:必须要写的 var :接收他的返回值的 scope:作用域 --%> <%--自闭和标签--%>

c:choose c:when


        <%--用c:set 保存一些数据--%>
<%--定义一个变量score,值为85--%>


<%--判断--%>
        <%--choose:选择--%>

        
            你的成绩为优秀
        

    
        你的成绩为良好
    

    
        你的成绩为一般
    
    
        你的成绩为不及格
    



c:forEach 数据库大量用到它!



<%--建一个对象--%>
<%
    ArrayList people = new ArrayList<>();
//    people :人  add:添加
    people.add(0,"张三");
    people.add(1,"李四");
    people.add(2,"王五");
    people.add(3,"田七");
//    放的request里 当前这个页面  有个list集合
    request.setAttribute("list",people);
%>
<%--遍历出来--%>
<%--
var:每一次遍历出来的变量
items: 要遍历的对象
begin: 哪里开始 默认从0开始
end: 到哪里 默认最后一个
step:步长 默认是1
--%>

     


9、JavaBean

Bean:豆子

实体类

JavaBean

  • 必须要有一个无参构造
  • 属性必须私有化
  • 必须有对应的get/set方法

一般用来和数据库的字段做映射 ORM;

ORM:对象关系映射

  • 数据库中的一张表 —> 对应 java的 类
  • 表里的字段 —> 对应 类的属性
  • 表里的记录 – 对应 类的 对象

数据库的一张表

people表

id name age address
1 小尹1号 12 山东
2 小尹2号 24 山东
3 小尹3号 30 山东

拆成一个java类

class people{
     
  	private int id;
  	private String name;
  	private int id;
  	private String address;
}

class A{
     
  new people(1,"小尹1号",12,"山东");
  new people(2,"小尹2号",24,"山东");
  new people(3,"小尹3号",30,"山东");
}

10、MVC三层架构

什么是MVC:Model view Controller 模型、试图、控制器

10.1、早些年

Javaweb_第97张图片

用户直接访问控制层,控制层就可以直接操作数据库

servlet --> CRUD(增删改查) -->数据库
  弊端:程序十分臃肿,不利于维护
  servlet的代码中:处理请求、响应、试图跳转、处理JDBC、处理业务代码、处理逻辑代码
  
  在中间在加一层
  架构思想中:没有什么是加一层解决不了的! 如果一层解决不了,再加一层
  加一层解决不了的方案:
  jdbc
  程序员调用jdbc
  jdbc
  |
  |
 MySQL  Oracle  sqlservlet……

10.2、MVC三层架构

Javaweb_第98张图片

有三层

Model

  • 业务处理:业务逻辑 (Servlet)
  • 数据持久层:CRUD 增删改查 (Dao)

View 视图

  • 展示数据
  • 提供链接发起Servlet请求 (a,form,img……)

Controller :(Servlet) 控制层

  • 接收用户的请求:(req :请求参数、Session信息……)

  • 交给业务层处理对应的代码

  • 控制试图的跳转

登录-->接收用户的登录请求 --> 处理用户的请求(获取用户登录的参数,username,password)--->交给业务层处理登录业务(判断用户密码是否正确:事务) --> Dao层查询用户名和密码是否正确-->数据库

11、Filter(过滤器 重点)

Filter:过滤器,用来过滤网站的数据

  • 处理中文乱码
  • 登录验证……

Javaweb_第99张图片

Filter开发步骤:

  1. 导包

    在pom.xml下

<dependencies>
  
  <dependency>
    <groupId>javax.servletgroupId>
    <artifactId>servlet-apiartifactId>
    <version>2.5version>
  dependency>
  
  <dependency>
    <groupId>javax.servlet.jspgroupId>
    <artifactId>javax.servlet.jsp-apiartifactId>
    <version>2.3.1version>
  dependency>
  
  <dependency>
    <groupId>javax.servlet.jsp.jstlgroupId>
    <artifactId>jstl-apiartifactId>
    <version>1.2version>
  dependency>
  
  <dependency>
    <groupId>taglibsgroupId>
    <artifactId>standardartifactId>
    <version>1.1.2version>
  dependency>

  
  <dependency>
    <groupId>mysqlgroupId>
    <artifactId>mysql-connector-javaartifactId>
    <version>8.0.24version>
  dependency>
dependencies>
  1. 编写过滤器

    1. 导包不要错

      Javaweb_第100张图片

    2. 实现Filter接口,重写对应的方法即可

package com.kuang.filter;

import javax.servlet.*;
import java.io.IOException;

/**
 * 字符集编码过滤器
 * @author <[email protected]>
 * @since 2021/6/21 18:33
 *
 * 过滤器 总结
 * 1.过滤器中的所有代码,在过滤特点请求的时候都会执行
 * 2.必须要让过滤器 继续同行 :把这个请求往下转交  转交的代码
 *   filterChain.doFilter(servletRequest,servletResponse);  固定的死代码
 *
 */
public class CharacterEncodingFilter implements Filter {
     

    @Override
//    初始化 :web服务器启动,就已经初始化了,随时等待过滤器对象出现!  他和服务器一同启动的
    public void init(FilterConfig filterConfig) throws ServletException {
     

    }

    @Override
//    FilterChain  Chain :链
    public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
     
        // 过来的请求等于utf-8编码
        servletRequest.setCharacterEncoding("utf-8");
        // 过去的请求等于utf-8编码
        servletResponse.setCharacterEncoding("utf-8");
        // 上下文
        servletResponse.setContentType("text/html;charset=UTF-8");

        System.out.println("CharacterEncodingFilter执行前");
//        让我们的请求继续走,如果不写,程序到这里就拦截停止了    如:两个过滤器
        filterChain.doFilter(servletRequest,servletResponse);
        System.out.println("CharacterEncodingFilter执行前后");
    }

    @Override
//    销毁  web服务器关闭的时候,过滤器会销毁
    public void destroy() {
     
        System.out.println("CharacterEncodingFilter销毁");
    }
}

  1. 在web.xml中配置 servlet 和 filter
<servlet>
  <servlet-name>ShowServletservlet-name>
  <servlet-class>com.kuang.servlet.ShowServletservlet-class>
servlet>
<servlet-mapping>
  <servlet-name>ShowServletservlet-name>
  <url-pattern>/servlet/showurl-pattern>
servlet-mapping>
<servlet-mapping>
  <servlet-name>ShowServletservlet-name>
  <url-pattern>/showurl-pattern>
servlet-mapping>



<filter>
  <filter-name>CharacterEncodingFilterfilter-name>
  <filter-class>com.kuang.filter.CharacterEncodingFilterfilter-class>
filter>
<filter-mapping>
  <filter-name>CharacterEncodingFilterfilter-name>
  
  <url-pattern>/servlet/*url-pattern>
filter-mapping>

package com.kuang.servlet;

import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;

/**
 * 展示Servlet
 * @author <[email protected]>
 * @since 2021/6/21 18:35
 */
public class ShowServlet extends HttpServlet {
     
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
     

        //以前的处理方式
//        resp.setCharacterEncoding("utf-8");
        // 乱码
        resp.getWriter().write("你好,世界");
    }

    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
     
        doGet(req, resp);
    }
}


结论:

当路径是 localhost:8080/servlet/show的时候 显示 你好,世界 过滤器启动了

当路径是localhost:8080/show的时候,显示 ??? 乱码 过滤器未启动

过滤器 总结

  • 1.过滤器中的所有代码,在过滤特点请求的时候都会执行

  • 2.必须要让过滤器 继续同行 :把这个请求往下转交 转交的代码

  • filterChain.doFilter(servletRequest,servletResponse); 固定的死代码

12、监听器

用代码来讲是:实现一个监听器的接口 (有N种)

统计网上在线人数: 监听 统计 session

  1. 编写一个监听器

    实现监听器的接口…

package com.kuang.listener;

import javax.servlet.ServletContext;
import javax.servlet.http.HttpSessionEvent;
import javax.servlet.http.HttpSessionListener;

/**
 * 统计网上在线人数: 监听  统计 session
 *  listener :听众
 * @author <[email protected]>
 * @since 2021/6/21 19:26
 */
public class OnlineCountListener implements HttpSessionListener {
     
  @Override
  //    创建session 监听 :看你的一举一动
  //    一旦创建一个session 就会触发一次这个事件
  public void sessionCreated(HttpSessionEvent httpSessionEvent) {
     
    // getSession() 拿到Session  getServletContext() 拿到网站上下文
    ServletContext ctx = httpSessionEvent.getSession().getServletContext();
    // 输出一下session的ID
    System.out.println(httpSessionEvent.getSession().getId());

    //        取东西
    Integer onlineCountListener = (Integer) ctx.getAttribute("OnlineCountListener");

    if (onlineCountListener==null){
     
      onlineCountListener = new Integer(1);
    }else {
     
      int count = onlineCountListener.intValue();

      onlineCountListener = new Integer(count+1);
    }
    ctx.setAttribute("OnlineCountListener",onlineCountListener);

  }

  @Override
  //     销毁 session 监听
  //    一旦销毁一个session 就会触发一次这个事件
  public void sessionDestroyed(HttpSessionEvent httpSessionEvent) {
     

    // getSession() 拿到Session  getServletContext() 拿到网站上下文
    ServletContext ctx = httpSessionEvent.getSession().getServletContext();

    //        1.手动销毁
    //        httpSessionEvent.getSession().invalidate();

    //        取东西
    Integer onlineCountListener = (Integer) ctx.getAttribute("OnlineCountListener");

    if (onlineCountListener==null){
     
      onlineCountListener = new Integer(0);
    }else {
     
      int count = onlineCountListener.intValue();

      onlineCountListener = new Integer(count-1);
    }
    ctx.setAttribute("OnlineCountListener",onlineCountListener);

  }
  /*
    * Session销毁的两种情况
    * 1.手动销毁 httpSessionEvent.getSession().invalidate();
    * 2.自动销毁在 web.xml中配置 session 过期时间
    * */
}

  1. 配置监听器 在web.xml中注册


<listener>
  <listener-class>com.kuang.listener.OnlineCountListenerlistener-class>
listener>
  1. 看情况是否使用

13、过滤器、监听器常见应用

监听器:GUI 编程中经常使用

GUI :图形界面编程

package com.kuang.listener;

import java.awt.*;
import java.awt.event.WindowAdapter;
import java.awt.event.WindowEvent;
import java.awt.event.WindowListener;

/**
 * GUL 监听器
 *
 * @author <[email protected]>
 * @since 2021/6/21 20:31
 */
public class TestPanel {
     
    public static void main(String[] args) {
     
//         新建一个界面      窗口的标题
        Frame frame = new Frame("快乐");
//        面板
        Panel panel = new Panel(null);

//        布局  设置窗体的布局
        frame.setLayout(null);
//        给窗体设置一些坐标
        frame.setBounds(300,300,300,300);
//        背景颜色
        frame.setBackground(new Color( 0,0,255));
//         面板 坐标
        panel.setBounds(50,50,50,50);
        panel.setBackground(new Color( 255,0,0));

//        内嵌的效果
        frame.add(panel);
//        设置可见性
        frame.setVisible(true);

//        给关闭加事件
//        监听关闭事件
        frame.addWindowListener(new WindowListener() {
     
            @Override
            public void windowOpened(WindowEvent e) {
     
//                        窗口打开的事件
                System.out.println("打开");
            }

            @Override       //窗口关闭中
            public void windowClosing(WindowEvent e) {
     

                System.out.println("关闭中");
//                把程序结束进程
//                正常终止 0 ;非正常终止 1
                System.exit(0);

            }

            @Override //窗口关闭
            public void windowClosed(WindowEvent e) {
     
                System.out.println("关闭");
            }

            @Override  // 窗体图标的事件
            public void windowIconified(WindowEvent e) {
     

            }

            @Override  //是否被激活的事件
            public void windowDeiconified(WindowEvent e) {
     
                System.out.println("激活");
            }

            @Override  // 没有被激活的事件
            public void windowActivated(WindowEvent e) {
     
                System.out.println("未激活");
            }

            @Override
            public void windowDeactivated(WindowEvent e) {
     

            }
        });

//        重写它的子类  有选择型!
        frame.addWindowListener(new WindowAdapter() {
     
            @Override
            public void windowDeiconified(WindowEvent e) {
     
                super.windowDeiconified(e);
            }
        });

    }
}

----
  输出:
未激活
未激活
未激活

Javaweb_第101张图片

登录监听功能

用户登录之后才能进入主页!用户注销后就不能进入主页了!(用过滤器来实现)

  1. 用户登录之后,向Session中放入用户的数据

  2. 进入主页的时候要判断用户是否已经登录;要求在过滤器中实现!

public void doFilter(ServletRequest req, ServletResponse resp, FilterChain filterChain) throws IOException, ServletException {
     
  // ServletRequest  HttpServletRequest   父子关系
  // 获取这个 存不存在
  HttpServletRequest request = (HttpServletRequest) req;
  HttpServletResponse response = (HttpServletResponse) resp;

  //        Object user_session = request.getSession().getAttribute("USER_SESSION");

  // 在过滤器阶段没有他
  if (request.getSession().getAttribute("USER_SESSION")==null){
     
    // 重定向回去
    response.sendRedirect("/error.jsp");
  }

  // filterChain继续往下走
  filterChain.doFilter(req,resp);
}

思路:步骤!

  1. 进入登录页面,
<%--登录页面  login 登录--%>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>Titletitle>
head>
<body>

<%--表单--%>
<%--action:提交的地址--%>
<h1>登录h1>
<form action="/servlet/login" method="post">
                <%--user:用户--%>
    <input type="text" name="username">
    <input type="submit">
form>

body>
html>
  1. 走一个请求,LoginServlet ,
package com.kuang.servlet;

import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;

/**
 * @author <[email protected]>
 * @since 2021/6/22 16:27
 */
public class LoginServlet extends HttpServlet {
     
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
     
        // 获取前端请求的参数
        String username = req.getParameter("username");

        // 登录成功
        if (username.equals("admin")){
     
        // 把用户信息放到某个地方   getSession  可以在多个页面中取到  setAttribute :放 键(常量) 和 值
            req.getSession().setAttribute("USER_SESSION",req.getSession().getId());
            // 重定向
            resp.sendRedirect("/sys/success.jsp");

        // 登录失败
        }else {
     
            resp.sendRedirect("/error.jsp");

        }
    }

    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
     
        doGet(req, resp);
    }
}

  1. 会有两种情况,要么成功,要么失败,

  2. 假设成功就进入成功的页面,

<%--主页  success :成功--%>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>


    Title


主页

<%--注销--%>

注销

  1. 失败走失败的页面
<%@ page contentType="text/html;charset=UTF-8" language="java" %>


    Title


错误

没有权限,不是管理员

用户名错误

返回登录页面
  1. 成功了就让他注销

  2. 失败了就让他返回主页

  3. 加一个权限验证

  4. 加一个过滤器:做一些用户Servlet判断,重点是放servlet和存servlet

package com.kuang.filter;

import javax.servlet.*;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;

/**
 * 过滤器
 *
 * @author <[email protected]>
 * @since 2021/6/22 18:30
 */
public class SysFilter implements Filter {
     
    @Override
    public void init(FilterConfig filterConfig) throws ServletException {
     

    }

    @Override
    public void doFilter(ServletRequest req, ServletResponse resp, FilterChain filterChain) throws IOException, ServletException {
     
        // ServletRequest  HttpServletRequest   父子关系
        // 获取这个 存不存在
        HttpServletRequest request = (HttpServletRequest) req;
        HttpServletResponse response = (HttpServletResponse) resp;

//        Object user_session = request.getSession().getAttribute("USER_SESSION");

        // 在过滤器阶段没有他
        if (request.getSession().getAttribute("USER_SESSION")==null){
     
            // 重定向回去
            response.sendRedirect("/error.jsp");
        }

        // filterChain继续往下走
        filterChain.doFilter(req,resp);
    }

    @Override
    public void destroy() {
     

    }
}

  1. 发现的问题:Servlet名字重用,提取出一个常量 优化!
public class Constant {
     
    // 修改统一
    public static String USER_SESSION = "USER_SESSION";
}
  1. web.xml中写完一个就配置一个
 <servlet>
        <servlet-name>LoginServletservlet-name>
        <servlet-class>com.kuang.servlet.LoginServletservlet-class>
    servlet>
    <servlet-mapping>
        <servlet-name>LoginServletservlet-name>
        <url-pattern>/servlet/loginurl-pattern>
    servlet-mapping>

    <servlet>
        <servlet-name>LogoutServletservlet-name>
        <servlet-class>com.kuang.servlet.LogoutServletservlet-class>
    servlet>
    <servlet-mapping>
        <servlet-name>LogoutServletservlet-name>
        <url-pattern>/servlet/logouturl-pattern>
    servlet-mapping>

    <filter>
        <filter-name>SysFilterfilter-name>
        <filter-class>com.kuang.filter.SysFilterfilter-class>
    filter>
    <filter-mapping>
        <filter-name>SysFilterfilter-name>
        <url-pattern>/sys/*url-pattern>
    filter-mapping>

14、JDBC

什么是JDBC:

J:java

DB:数据库

C:Connection 连接

Java连接数据库!

Javaweb_第102张图片

连接数据库需要两个jar包

  • java.sql
  • javax.sql
  • mysql-connector-java 连接驱动(必须要导)

实验环境搭建

-- 创建 users 表 
create table users(
		id int primary key,
		`name` varchar (40),
		`password` varchar (40),
		email varchar (60),
		birthady DATE
);

-- 插入一个数据
insert into users (id,`name`,`password`,email,birthady) values
(1,'张三','123456','[email protected]' ,'2000-01-01');
insert into users (id,`name`,`password`,email,birthady) values
(2,'李四','123456','[email protected]' ,'2000-01-01');
insert into users (id,`name`,`password`,email,birthady) values
(3,'王五','123456','[email protected]' ,'2000-01-01');

-- 查询数据
select *from users

导入数据库依赖 pom.xml下


<dependencies>
  
  <dependency>
    <groupId>mysqlgroupId>
    <artifactId>mysql-connector-javaartifactId>
    <version>8.0.24version>
  dependency>
dependencies>

IDEA中连接数据库

Javaweb_第103张图片

idea 连接 需要改时区:Asia/Shanghai

idea 连接 需要改时区:UTC

JDBC 固定步骤:

  1. 加载驱动
  2. 连接数据库
  3. 向数据库发送SQL的对象 Statement :CRUD,增删改查
  4. 编写SQL语句(根据业务,不同的SQL)
  5. 执行 SQL
  6. 关闭连接

Statement

普通的

package com.kuang.test;

import com.mysql.jdbc.Driver;

import java.sql.*;
// 查询

public class TestJdbc {
     
    public static void main(String[] args) throws ClassNotFoundException, SQLException {
     
        // 配置信息
        // 解决中文乱码 :useUnicode=true&characterEncoding=utf-8
        String url = "jdbc:mysql://localhost:3306/jdbc?useUnicode=true&characterEncoding=utf-8&serverTimezone=UTC ";
        String username = "root";
        String password = "123456";

        // 1.加载驱动
        Class.forName("com.mysql.jdbc.Driver");
        // 2.连接数据库,connection:代表数据库 , DriverManager:连接数据库驱动
        Connection connection = DriverManager.getConnection(url, username, password);
        // 3.向数据库发送SQL的对象 Statement(普通的),PreparedStatement (安全的) :CRUD,增删改查
        Statement statement = connection.createStatement();
//        PreparedStatement 安全的
        // 4.编写SQL语句
        String sql = "select * from users";
        // 5.执行查询SQL  statement:用来执行的对象    返回一个结果集 ResultSet
        ResultSet rs = statement.executeQuery(sql);
            // 判断他还有没有数据,有的话就往下遍历,遍历完就得到具体的对象
        while (rs.next()){
     
            // 从结果集里面.get某个对象,如果都不知道,get object
            System.out.println("id"+rs.getObject("id"));
            System.out.println("name"+rs.getObject("name"));
            System.out.println("password"+rs.getObject("password"));
            System.out.println("email"+rs.getObject("email"));
            System.out.println("birthady"+rs.getObject("birthady"));
        }
        // 6.关闭连接,释放资源(一定要做)先开后关
        rs.close();
        statement.close();
        connection.close();
    }
}

PreparedStatement 预编译SQL

安全的

防止SQL注入

package com.kuang.test;

import java.sql.*;

/**
 * PreparedStatement对象
 * PreparedStatement 安全的
 * @author <[email protected]>
 * @since 2021/6/23 20:07
 */
public class TestJdbc2 {
     
    public static void main(String[] args) throws ClassNotFoundException, SQLException {
     

        // 配置信息
        // 解决中文乱码 :useUnicode=true&characterEncoding=utf-8
        String url = "jdbc:mysql://localhost:3306/jdbc?useUnicode=true&characterEncoding=utf-8&serverTimezone=UTC ";
        String username = "root";
        String password = "123456";

        // 1.加载驱动
        Class.forName("com.mysql.jdbc.Driver");

        // 2.连接数据库,connection:代表数据库 , DriverManager:连接数据库驱动
        Connection connection = DriverManager.getConnection(url, username, password);

        // 3.编写SQL
        String sql = " insert into users(id, name, password, email, birthady) values (?,?,?,?,?); ";

        // 4.预编译
        PreparedStatement preparedStatement = connection.prepareStatement(sql);
        // 把具体的值赋进去
        // 第一个占位符?  的值赋值为1;
        preparedStatement.setInt(1,4);
        // 第二个占位符?  的值赋值为小明;
        preparedStatement.setString(2,"小明");
        // 第三个占位符?  的值赋值为123456;
        preparedStatement.setString(3,"123456");
        // 第四个占位符?  的值赋值为[email protected];
        preparedStatement.setString(4,"[email protected]");
        // 第五个占位符?  的值赋值为new Date(new java.util.Date().getTime());
        // 外面的Date是sql.Date ,里面是util.Date    加一个时间戳
        preparedStatement.setDate(5,new Date(new java.util.Date().getTime()));

        // 5.执行SQL
        int i = preparedStatement.executeUpdate();

        if (i>0){
     
            System.out.println("插入成功");
        }

        // 6.关闭连接,释放资源(一定要做)先开后关
        preparedStatement.close();
        connection.close();
    }
    }

事务
要么都成功,要么都失败!
特性:

ACID原则:保证数据的安全

原子性,一致性,隔离性,持久性

状态

开启事务
事务提交	commit()
事务回滚	rollback()
关闭事务

转账:
A:1000B:1000A(900)	--100-->	B(1100)  在一组事务中

Junit单元测试

依赖 jar包


<dependency>
  <groupId>junitgroupId>
  <artifactId>junitartifactId>
  <version>4.11version>
dependency>

简单使用

@Test注解只有在方法上有效,只要加了这个注解的方法,就可以直接运行!

@Test
public void test(){
     
  System.out.println("Hello");
}

Javaweb_第104张图片

失败的时候是红色

Javaweb_第105张图片

搭建一个环境,测试一波事务

模拟转账
# 开启事务
start transaction ;
#模拟转账    修改这张表
update account set money = money-100 where name ='A';
update account set money = money+100 where name ='B';

# 回滚  回到开始事务之前的状态
rollback ;

# 提交  提交完之后,转账才会执行
commit ;

Javaweb_第106张图片

用IDEA 模拟转账

package com.kuang.test;

import org.junit.Test;

import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.SQLException;

/**
 * 事务
 * 模拟转账
 * @author <[email protected]>
 * @since 2021/6/24 17:34
 */
public class TestJdbc3 {
     
    @Test
    public void test()  {
     

        // 配置信息
        // 解决中文乱码 :useUnicode=true&characterEncoding=utf-8
        String url = "jdbc:mysql://localhost:3306/jdbc?useUnicode=true&characterEncoding=utf-8&serverTimezone=UTC ";
        String username = "root";
        String password = "123456";


            // 提对象
            Connection connection = null;
        // 1.加载驱动
        try {
     
                Class.forName("com.mysql.jdbc.Driver");


            // 2.连接数据库,connection:代表数据库 , DriverManager:连接数据库驱动
            connection = DriverManager.getConnection(url, username, password);

            // 3.通知数据库开启 事务 ,false(开启)
                // 下边的这句话等价于  start transaction (开启事务);
            connection.setAutoCommit(false);

            String sql1 = "update account set money = money-100 where name ='A';";
                // 执行一下
            connection.prepareStatement(sql1).executeUpdate();

                // 制造错误
//                int i = 1/0;
            String sql2 = "update account set money = money-100 where name ='B';";
            connection.prepareStatement(sql2).executeUpdate();

            // 提交事务
            connection.commit();// 以上两条SQL都执行成功了,就提交事务!
            System.out.println("提交成功");
            System.out.println("success");
        } catch ( Exception e) {
     
                // 如果出现异常,就通知数据库回滚事务  connection.rollback(); :回滚!
            try {
     
                connection.rollback();
            } catch (SQLException throwables) {
     
                throwables.printStackTrace();
            }
            e.printStackTrace();
        }finally {
     
                // 把连接关闭掉  connection.close();
            try {
     
                connection.close();
            } catch (SQLException throwables) {
     
                throwables.printStackTrace();
            }
        }

    }
}

你可能感兴趣的:(Javaweb,java,maven,jar,java-ee)