Javaweb详解一

1.基本概念

1.1.web开发:

  • 静态web
    • html,css
    • 提供给所有人看的数据始终不会发生变化
  • 动态web:
    • 淘宝.
    • 提供给所有人看的数据始终会发生变化,每个人在不同时间不同地点看的信息各不相同
    • 技术栈: Servlet,asp,php
      在Java中,动态web资源开发的技术统称为javaweb

1.2.web应用程序

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

  • a.html,b.html…多个web资源可以被外界访问,对外界提供服务;
  • 能访问到的任何一个页面或者资源,都存在与这个世界的偶一个角落的计算机上
  • URL
  • 这个统一的web资源**(应用)**会被放在同一个文件夹下,web应用程序 --> Tomcat: 服务器
  • 一个web应用由多部分组成(静态web,动态web)
    • html,css,js
    • jsp,servlet
    • java程序
    • jar包
    • 配置文件 (Properties)
      web应用程序编写完毕后,若想提供给外界访问: 要一个服务器统一管理

1.3.静态web

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

Javaweb详解一_第1张图片

  • 静态web存在的缺点

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

1.4、动态web

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

网页架构图:

Javaweb详解一_第2张图片

jsp可以调用java程序

缺点:

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

优点:

  • Web页面可以动态更新,所有用户看到都不是同一个页面
  • 它可以与数据库交互 (数据持久化:注册,商品信息,用户信息… ; 放到数据库里面,下次登录不至于再也找不到自己,静态web没有连接数据库不能做到)

Javaweb详解一_第3张图片

动态web资源通过JDBC访问数据库的资源并返回到服务器

2、web服务器

2.1、三个技术讲解

ASP:

  • 微软:国内最早流行的就是ASP;
  • 在HTML中嵌入了VB的脚本, ASP + DOM;
  • 在ASP开发中,基本一个页面都有几千行的业务代码,页面极其换乱
  • 维护成本高!
  • 在ASP中主要用C#语言
  • IIS
    ASP代码
<h1>
    <h1><h1>
        <h1>
            <h1>
                <h1>
        <h1>
            <%
            System.out.println("hello")
            %>
            <h1>
                <h1>
   <h1><h1>
<h1>

php:

  • PHP开发速度很快,功能很强大,跨平台,代码很简单 (70% , WP)
  • 无法承载大访问量的情况(局限性)

**JSP/Servlet : **

B/S:浏览和服务器

C/S: 客户端和服务器

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

还有更多的引擎…

2.2、web服务器(web server): 用来接收用户的请求,并且负责给用户返回一些响应

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

IIS

微软的; ASP…,Windows中自带的

Tomcat 就是做一些这样的事情

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

3、Tomcat

3.1、 安装tomcat

tomcat官网:http://tomcat.apache.org/

Javaweb详解一_第4张图片

直接解压即可使用

3.2、Tomcat启动和配置

文件夹作用:

Javaweb详解一_第5张图片

启动。关闭Tomcat

Javaweb详解一_第6张图片

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

可能遇到的问题:

  1. Java环境变量(jre)没有配置
  2. 闪退问题:需要配置兼容性
  3. 乱码问题:配置文件中设置

3.3、配置

Javaweb详解一_第7张图片

在server.xml文件里可以配置启动的端口号

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

还可以配置主机的名称

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

高难度面试题:

请你谈谈网站是如何进行访问的!

  1. 输入一个域名;回车

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

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

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

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-9tScf9yS-1635248756859)(https://upload-images.jianshu.io/upload_images/24940810-f831ec8413484840.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)]

3.可以配置一下环境变量(可选性): 好处是随时启动

3.4、发布一个web网站

  • 将自己写的网站,放到服务器(Tomcat)中指定的web应用的文件夹(webapps)下,启动Tomcat就可以访问了

网站应该有的结构

这里面的文件都可以随便建,因为可以访问到html文件肯定就可以访问到其他的文件

--webapps :Tomcat服务器的web目录
	-ROOT
	-xxx :网站的目录名
		- WEB-INF
			-classes : java程序
			-lib:web应用所依赖的jar包
			-web.xml :网站配置文件
		- index.html 默认的首页
		- static 
            -css
            	-style.css
            -js
            -img
         -..... 

4、Http

4.1、什么是HTTP

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

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

Https:安全的(+s: security)

  • 默认端口: 443

4.2、两个时代

  • http1.0时代

    • HTTP/1.0:客户端可以与web服务器一次连接后,只能获得一个web资源,断开连接: 比如从百度上请求了一个index.html(一个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:14.215.177.39:443
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、请求行

主要用来控制请求的协议

  • 请求行中的请求方式:GET
  • 请求方式有哪些:Get,Post,HEAD,DELETE,PUT,TRACT…
    • get:请求能够携带的参数比较少,大小有限制,会在浏览器的URL地址栏显示数据内容,不安全,但高效
    • post:请求能够携带的参数没有限制,大小没有限制,不会在浏览器的URL地址栏显示数据内容,安全,但不高效。

2、消息(请求)头: Request Headers

Accept:告诉服务器,它所支持的数据类型
Accept-Encoding:支持哪种编码格式  GBK   UTF-8(数据库用的)   GB2312  ISO8859-1(java默认编码)
Accept-Language:告诉服务器,它的语言环境
Cache-Control:缓存控制(例如: 用户登录缓存)
Connection:告诉服务器,请求完成是断开还是保持连接
HOST:表示主机..../.

4.4、Http响应

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

百度响应给客户端:

Cache-Control:private    缓存控制
Connection:Keep-Alive    保持连接: 现在能连接上,走了HTTP/1.1
Content-Encoding:gzip    编码
Content-Type:text/html   响应回的类型

1.响应体

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

2、响应状态码 (重点)

200:请求响应成功 200

3xx:请求重定向

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

4xx:找不到资源 404 禁止访问: 403

  • 资源不存在;

5xx:服务器代码错误 500 502:网关错误: Bad Gateway 作为网关或者代理工作的服务器尝试执行请求时,从远程服务器接收到了一个无效的响应.

常见面试题:

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

5、Maven

我为什么要学习这个技术?

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

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

    由此,Maven诞生了!

5.1 Maven: 项目架构管理工具

目前用来就是方便导入jar包的!
Maven的核心思想:约定大于配置

  • 有约束,不要去违反。(别人制定的规则,必须这么做)
    能够防止出现各种各样以外情况,以及解决不了的问题

Maven会规定好你该如何去编写我们的Java代码,目录结构必须要按照这个规范来;

5.2 下载安装Maven

官网;https://maven.apache.org/

Javaweb详解一_第8张图片

下载完成后,解压即可;

5.3 配置环境变量

在我们的系统环境变量中

配置如下配置:

  • M2_HOME maven目录下的bin目录

  • MAVEN_HOME maven的目录

  • 在系统变量下的path中配置 %MAVEN_HOME%\bin

image.png

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

5.4 添加阿里云镜像

在conf配置文件夹下的setting.xml配置文件中的mirrors添加镜像

Javaweb详解一_第9张图片

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

    nexus-aliyun  
    *,!jeecg,!jeecg-snapshots  
    Nexus aliyun  
    http://maven.aliyun.com/nexus/content/groups/public 

5.5 本地仓库

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

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

  1. 先在maven文件下创建一个新文件夹maven-repo

Javaweb详解一_第10张图片

2.再在setting.xml文件中添加新的本地仓库地址

Javaweb详解一_第11张图片

5.6、在IDEA中使用Maven

  1. 启动IDEA
  2. 创建一个MavenWeb项目

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-5Xzw82xA-1635248756866)(https://upload-images.jianshu.io/upload_images/24940810-3da79cc889b2ed52.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)]

下一步:

Javaweb详解一_第12张图片

Javaweb详解一_第13张图片

下一步:

Javaweb详解一_第14张图片

Javaweb详解一_第15张图片

3.等待项目初始化完毕

  • 出现这个说项目搭建成功

Javaweb详解一_第16张图片

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

  2. IDEA中的Maven设置

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

Javaweb详解一_第17张图片

Javaweb详解一_第18张图片

6.到这里,Maven在IDEA中的配置和使用就OK了!

5.7、创建一个普通(纯净)的Maven项目

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-LkpAQn2e-1635248756875)(https://upload-images.jianshu.io/upload_images/24940810-c37e57cfac7b42de.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)]

Javaweb详解一_第19张图片

下面这个只有在Web应用(webapp)下才会有!

Javaweb详解一_第20张图片

5.8 标记文件夹功能(给文件夹赋予功能)

方式一:

Javaweb详解一_第21张图片

方式二:

Javaweb详解一_第22张图片

Javaweb详解一_第23张图片

Javaweb详解一_第24张图片

5.9 在 IDEA中配置Tomcat

Javaweb详解一_第25张图片

Javaweb详解一_第26张图片

Javaweb详解一_第27张图片

  • 若下面报ERROR,则是没有配置本地Tomcat服务器地址:Application server

Javaweb详解一_第28张图片

解决警告问题

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

Javaweb详解一_第29张图片

Javaweb详解一_第30张图片

Javaweb详解一_第31张图片

Javaweb详解一_第32张图片

5.10 pom文件

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

Javaweb详解一_第33张图片



<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.tigagroupId>
  <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.8maven.compiler.source>
    <maven.compiler.target>1.8maven.compiler.target>
  properties>

  <dependencies>

    <dependency>
      <groupId>junitgroupId>
      <artifactId>junitartifactId>
      <version>4.11version>
    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详解一_第34张图片

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

问题见于: mybatis的静态资源过滤问题
问题是: 比如在java目录下maven约定死了只能写java,如果在该目录下写xml格式文件是导不出的
需要在src/main/java目录下让他可以包含.properties,.xml这样的文件,这样java目录的这些文件就可以导出了
让他不去过滤这些文件


    
    <build>
        <resources>
            <resource>
                <directory>src/main/resourcesdirectory>
                <includes>
                    <include>**/*.propertiesinclude>
                    <include>**/*.xmlinclude>
                includes>
                <filtering>truefiltering>
            resource>
            <resource>
                <directory>src/main/javadirectory>
                <includes>
                    <include>**/*.propertiesinclude>
                    <include>**/*.xmlinclude>
                includes>
                <filtering>truefiltering>
            resource>
        resources>
    build>

5.12 IDEA操作

  • 热部署
    文件资源自动更新

5.13 解决遇到的问题

1.IDEA中每次都要重复配置Maven
在IDEA中的全局默认配置中去配置

Javaweb详解一_第35张图片

Javaweb详解一_第36张图片

  1. Maven项目中Tomcat无法配置
  2. maven默认web项目中的web.xml版本问题

Javaweb详解一_第37张图片

4.替换为webapp中的web.xml为4.0版本和tomcat服务器的web.xml一致


<web-app xmlns="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>

5.Maven仓库的使用
地址:https://mvnrepository.com/

  • 第一次导入maven依赖,如果在IDEA中搜不到就去Maven仓库去找,不知道接口名,可以从启动的地方往上推,比如说启动Tomcat需要用到servlet-api-jar包,就去搜关于这的jar包

Javaweb详解一_第38张图片

Javaweb详解一_第39张图片

6、Servlet

什么是Servlet?
Servlet是JavaWeb开发的基石,与平台无关的服务器组件,他是运行在Servlet容器/Web应用服务器/Tomcat,负责与客户端进行通信.
Servlet的功能:
1.创建并返回基于客户端请求的动态HTML页面.
2.与数据库进行通信.

Javaweb详解一_第40张图片

  • 如何使用Servlet?
    Servlet本身是一组接口,就是java程序中描述某一个东西具备某一种功能的操作,是抽象化的,是可以作为一个后台应用的功能,这个接口由java提供存放在 javax.servlet中,java.lang,java.util,java指的是最基础的那套东西,javax指的是后来扩展出来的东西.
    自定义一个类,并实现Servlet接口,这个类就具备了接收客户端请求以及做出响应的功能.
public class MyServlet implements Servlet {
    @Override
    public void init(ServletConfig servletConfig) throws ServletException {

    }

    @Override
    public ServletConfig getServletConfig() {
        return null;
    }

    @Override
    public void service(ServletRequest servletRequest, ServletResponse servletResponse) throws ServletException, IOException {
        // 在Servlet容器中处理消息
        System.out.println("我是Servlet容器,我成功接收到请求了");
        String id = servletRequest.getParameter("id");
        servletResponse.setContentType("text/html;charset=utf-8");
        // 通过servletResponse对象调用输出方法,就能把消息返回给客户端
        servletResponse.getWriter().println("接收到的参数是: " + id);
        servletResponse.getWriter().write("客户端你好,我是servlet容器");
    }

    @Override
    public String getServletInfo() {
        return null;
    }

    @Override
    public void destroy() {

    }
}

浏览器不能直接访问Servlet文件,只能通过映射的方式来间接访问Servlet,映射需要开发者手动配置,有两种配置方式.

  • 基于xml文件的配置方式

  • 基于注解的配置方式

6.1、Servlet简介

  • Servlet就是sun公司开发动态web的一门技术
  • Sun在这些API中提供一个接口叫做:Servlet,如果你想开发一个Servlet程序,只需要完成两个小步骤:
    • 编写一个类,实现Servlet接口
    • 把开发好的Java类部署到web服务器中。

把实现了Servlet接口的Java程序叫做,Servlet

6.2、HelloServlet

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

HttpServlet中继承的services 方法里面做的事情是调用doGet或者doPost方法

doGet方法是在浏览器顶部栏中输入被调用
doPost方法是在表单中输入被调用

  1. 构建一个普通的Maven项目,删掉里面的src目录,以后我们的学习就在这个项目里面建立Module;因为IDEA每次只能展示一个project这个空的工程就是Maven主工程;

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

    父项目中会有

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

    子项目会有

没有parent的原因: 去父工程的依赖中删除scope标签,重建创建子模块即可

    <parent>
        <artifactId>javaweb-02-servletartifactId>
        <groupId>com.tigagroupId>
        <version>1.0-SNAPSHOTversion>
    parent>

父项目中的java子项目可以直接使用

son extends father
  1. Maven环境优化

    1. 修改web.xml为最新的
    2. 将maven的结构搭建完整(在main目录下创建java和resources目录并标记功能)

4.编写一个Servlet程序
在主工程的module下的java目录下创建package(命名规范: 里面放什么就用其命名)

    1. 编写一个普通类
    1. 间接实现Servlet接口,这里我们直接继承HttpServlet
public class HelloServlet extends HttpServlet {
    // 重写HttpServlet中的doGet和doPost方法
    // 由于get或post只是请求(request)实现的不同的方式,所以可以互相调用,业务逻辑都一样
    // 假设方法里有存在的业务,在里面写jdbc代码让他往里存数据库,两种方法都一样,不需要再把同样的代码在另一种方法抄一遍
    // 最快的方法是在另一个方法调用当前的方法,这两个方法谁调谁都无所谓,所以代码在一种方法里写就行了
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        System.out.println("进入doGet方法");// 在控制台打印
        // 文件流(响应流)写出需要response的对象
        // 创建响应流(输出流)
        // 下面这段要注释掉,否则浏览器会报500错误
        // ServletOutputStream sos = resp.getOutputStream();
        PrintWriter writer = resp.getWriter();
        writer.print("hello Servlet");
    }

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

5.编写Servlet的映射

为什么需要映射:我们写的是JAVA程序,但是要通过浏览器访问,而浏览器需要连接web服务器,所以我们需要再web服务中注册我们写的Servlet,还需给他一个浏览器能够访问的路径;

 
    <servlet>

        <servlet-name>helloservlet-name>

        <servlet-class>com.tiga.servlet.HelloServletservlet-class>
    servlet>

    <servlet-mapping>

        <servlet-name>helloservlet-name>
        

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

6.配置Tomcat

  • 添加Tomcat Server - Local
  • 添加Deployment的war包
  • 添加发布路径
    注意:配置项目发布的路径就可以了

7.启动测试,OK!
启动后出现target,会生成一个网站,生成一个war包(就是一个网站),war包上面的文件夹就是其解压包,解压包的WEB-INF下的classes文件夹里就是java写的类;
lib文件夹里存放的是导入的jar包
首先会访问Tomcat配置的路径下默认的index.jsp, 然后在浏览器访问地址后加上在web.xml配置的Servlet映射的url-pattern地址,访问后他会到web.xml下找url-pattern,沿着往上找到了name,最后找到映射Servlet注册的处理类指向的java的类

6.3、Servlet原理

就是我把请求给你,你去处理,然后再把结果给我

早期的web应用主要应用于浏览静态网页,HTTP服务器(比如Apache、Nginx)向浏览器返回静态HTML,浏览器负责解析HTML,将结果呈现给用户。
随着互联网发展,我们已不满足于仅仅浏览静态网页,还希望通过一些交互操作来获取动态结果,因此也就需要一些扩展机制能够让Http服务器调用服务端程序。
于是Sun公司推出了Servlet容器。可以把servlet简单理解为运行在服务端的java小程序,但是servlet没有main方法,不能独立运行,因此必须把它部署到serlet容器,由容器来实例化并调用servlet。
而Tomcat和jetty就是一个Servlet容器。为了方便使用,他们也具有Http协议服务器的功能,因此Tomcat或jetty就是“HTTP服务器+Servlet容器”,我们也叫它Web容器。

  • Tomcat 与 Servlet 配合工作

步骤:
Web Client 向Servlet容器**(Tomcat)**发出Http请求
Servlet容器接收Web Client的请求
Servlet容器创建一个HttpRequest对象,将Web Client请求的信息封装到这个对象中。
Servlet容器创建一个HttpResponse对象
Servlet容器调用HttpServlet对象的service方法,把HttpRequest对象与HttpResponse对象作为参数传给 HttpServlet 对象。
HttpServlet调用HttpRequest对象的有关方法,获取Http请求信息。
HttpServlet调用HttpResponse对象的有关方法,生成响应数据。
Servlet容器把HttpServlet的响应结果传给Web Client。

  • Servlet工作原理

1、首先简单解释一下Servlet接收和响应客户请求的过程,首先客户发送一个请求,Servlet是调用service()方法对请求进行响应的,通过源代码可见,service()方法中对请求的方式进行了匹配,选择调用doGet,doPost等这些方法,然后再进入对应的方法中调用逻辑层的方法,实现对客户的响应。在Servlet接口和GenericServlet中是没有doGet()、doPost()等等这些方法的,HttpServlet中定义了这些方法,但是都是返回error信息,所以,我们每次定义一个Servlet的时候,都必须实现doGet或doPost等这些方法。

2、每一个自定义的Servlet都必须实现Servlet的接口,Servlet接口中定义了五个方法,其中比较重要的三个方法涉及到Servlet的生命周期,分别是上文提到init(),service(),destroy()方法。GenericServlet是一个通用的,不特定于任何协议的Servlet,它实现了Servlet接口。而HttpServlet继承于GenericServlet,因此HttpServlet也实现了Servlet接口。所以我们定义Servlet的时候只需要继承HttpServlet即可。

3、Servlet接口和GenericServlet是不特定于任何协议的,而HttpServlet是特定于HTTP协议的类,所以HttpServlet中实现了service()方法,并将请求ServletRequest、ServletResponse 强转为HttpRequest 和 HttpResponse。

  • 创建Servlet对象的时机:

Servlet容器启动时:读取web.xml配置文件中的信息,构造指定的Servlet对象,创建ServletConfig对象,同时将ServletConfig对象作为参数来调用Servlet对象的init方法。

在Servlet容器启动后:客户首次向Servlet发出请求,Servlet容器会判断内存中是否存在指定的Servlet对象,如果没有则创建它,然后根据客户的请求创建HttpRequest、HttpResponse对象,从而调用Servlet 对象的service方法。

Servlet容器在启动时自动创建Servlet,这是由在web.xml文件中为Servlet设置的属性决定的。从中我们也能看到同一个类型的Servlet对象在Servlet容器中以单例的形式存在。

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

Javaweb详解一_第41张图片

6.4、Mapping问题

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

2.一个Servlet可以指定多个映射路径

 <servlet-mapping>
        <servlet-name>helloservlet-name>
        <url-pattern>/hellourl-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>
    <servlet-mapping>
        <servlet-name>helloservlet-name>
        <url-pattern>/hello5url-pattern>
    servlet-mapping>

3.一个Servlet可以指定通用映射路径

 <servlet-mapping>
        <servlet-name>helloservlet-name>
        <url-pattern>/hello/*url-pattern>
    servlet-mapping>

4.默认请求路径**(尽量少用)**


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

5.指定一些后缀或者前缀等等….


    <servlet-mapping>
        <servlet-name>helloservlet-name>
        <url-pattern>*.tigaurl-pattern>
    servlet-mapping>

6.优先级问题
假如精准匹配,越准确优先级越高
指定了固有的映射路径优先级最高,如果找不到就会走默认的处理请求;


    <servlet>
        <servlet-name>errorservlet-name>
        <servlet-class>com.tiga.servlet.ErrorServletservlet-class>
    servlet>

    <servlet-mapping>
        <servlet-name>errorservlet-name>

        <url-pattern>/*url-pattern>
    servlet-mapping>

6.5、ServletContext

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

1、共享数据

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

Javaweb详解一_第42张图片

  • 会被request,session替代,这种方法尽量少用,因为服务器多了就会承受不住
public class HelloServlet extends HttpServlet {
    // 重写doGet,doPost方法
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        // 当前类的对象调用继承父类的方法
        // this.getInitParameter();// 初始化参数
        // this.getServletConfig();// Servlet配置
        // this.getServletContext();// Servlet上下文环境
        // 每个web工程中只有一个ServletContext对象
        // 代表当前web应用,是web程序的映射,这个网站由这个对象来管理,公共的,可共享数据
        ServletContext context = this.getServletContext();
        String username = "tiga";
        // 先存
        // 将一个数据保存在ServletContext中,(key值)名字为: username,(value)值为: username对象
        context.setAttribute("username",username); (这种方法尽量少用)
    }
    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        doGet(req,resp);
    }
}
public class HelloServlet extends HttpServlet {
    // 重写doGet,doPost方法
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        // 当前类的对象调用继承父类的方法
        // this.getInitParameter();// 初始化参数
        // this.getServletConfig();// Servlet配置
        // this.getServletContext();// Servlet上下文环境
        // 每个web工程中只有一个ServletContext对象
        // 代表当前web应用,是web程序的映射,这个网站由这个对象来管理,公共的,可共享数据
        ServletContext context = this.getServletContext();
        String username = "tiga";
        // 先存
        // 将一个数据保存在ServletContext中,(key值)名字为: username,(value)值为: username对象
        context.setAttribute("username",username);
    }
    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        doGet(req,resp);
    }
}

    <servlet>
        <servlet-name>helloservlet-name>
        <servlet-class>com.tiga.servlet.HelloServletservlet-class>
    servlet>

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

    <servlet>
        <servlet-name>getcservlet-name>
        <servlet-class>com.tiga.servlet.GetServletservlet-class>
    servlet>
    <servlet-mapping>
        <servlet-name>getcservlet-name>
        <url-pattern>/getcurl-pattern>
    servlet-mapping>

2、获取初始化参数


    <context-param>

        <param-name>urlparam-name>


        <param-value>jdbc:mysql//localhost:3306/mybatisparam-value>
    context-param>
@Override
    // 从req参数接收请求处理,再到resp参数请求响应回去
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        // context的应用
        ServletContext context = this.getServletContext();
        // 可以在web.xml中配置一些web应用初始化参数
        // 在web.xml中配置完获取初始化的参数, 形参为web.xml中初始化参数的参数名,键值对方式
        String url = context.getInitParameter("url");
        resp.getWriter().print(url);
    }

3、请求转发

  • 转发路径不变,页面改变; 重定向路径改变,页面也改变

这里注意重定向时A(客户端)是发送了两次请求,请求转发时A只发送了一次请求,而服务器端对该请求进行了转发

public class ServletDemo02 extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        // 获取ServletContext对象
        ServletContext context = this.getServletContext();
        // 转发和重定向的区别: 转发像中间商,重定向像媒婆介绍新人
        // 这里只是做了转发的作用
        // 获取请求转发,方法形参是要转发的地址,forward(): 代表转发的意思
        // RequestDispatcher requestDispatcher = context.getRequestDispatcher("/gp");// 转发的请求路径
        // 调用forward实现请求转发,参数1: 请求对象 参数2: 响应对象
        // requestDispatcher.forward(req,resp);
        // 直接合并实现请求转发
        // 记住这个请求转发,这是springmvc dispatchServlet的实现原理
        context.getRequestDispatcher("/gp").forward(req,resp);
    }

转发和重定向

Javaweb详解一_第43张图片

Javaweb详解一_第44张图片

4、读取资源文件

Properties

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

Servlet运行时,在解压触点war包下的classes,目录下的java下创建的properties文件没有生成,由于maven中配置pom.xml约定大于配置的原因,资源导出可能存在问题,所以要在pom.xml中添加build配置防止导出失败

成功导出后,发现:都被打包到了同一个路径下:classes,我们俗称这个路径为classpath:
思路:需要一个文件流;

username=root613516
password=1234ioregroe
public class PropertiesServlet extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        // 获取Servlet上下文环境,并把资源变成流
        // 路径格式: /: 不能省略,代表当前web应用(项目),或当前文件夹,这个项目指的是打包完之后的项目
        // 用SservletContext类的对象获得流,把文件变成流通过Perproties类的对象文件读取出来
        InputStream is = this.getServletContext().getResourceAsStream("/WEB-INF/classes/com/tiga/servlet/aa.properties");
        Properties prop = new Properties();
        // prop对象加载文件流
        prop.load(is);
        // 获取Properties类prop对象的属性
        String user = prop.getProperty("username");
        String pwd = prop.getProperty("password");
        // 修改响应编码格式
        resp.setContentType("text/html;charset=utf-8");
        // 响应请求到客户端
        resp.getWriter().write(user + ":" + pwd);
        // 去webapp下注册映射地址
    }

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

6.6、HttpServletResponse

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

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

1、简单分类

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

ServletOutputStream getOutputStream() throws IOException;
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、下载文件

public class FileServlet extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        // ServletContext对象是Tomcat创造的
        // 1.获取Servlet上下文对象,和绝对路径,参数为路径,/:代表当前web下去找jpg
        // 错误获取方式:只是从本地磁盘上获取路径,而不是从Tomcat服务器的web应用上获取路径
        // String realPath = this.getServletContext().getRealPath("/比伯.jpg");
        // 正确获取方式:
        // 写死绝对路径,也可以写相对路径:相对路径:"/WEB-INF/classes/比伯.jpg"
        String realPath = "F:\\JavaWeb\\javaweb-02-servlet\\response\\target\\classes\\比伯.jpg";
        // 通过反射方法获取路径
        // String realPath = this.getClass().getClassLoader().getResource("比伯.jpg").getPath();
        System.out.println("要下载的文件路径: " + realPath);
        // 2.获取下载文件名,用substring方法截取字符串,截取/后的字符串一定是文件名
        // lastIndexOf(): 获取路径的的最后一个字符串, \\: 表示转义, +1: 表示截取/的下一位就是文件名了
        String fileName = realPath.substring(realPath.lastIndexOf("\\") + 1);
        // 3.设置下载的头消息:设置让浏览器能支持(Content-Disposition)下载指定的东西;
        // 响应回客户端使其能够下载东西,根据attachment(附件)的名称,用文件编码转换方法: URLEncoder.encode编码,否则有可能乱码
        resp.setHeader("Content-Disposition","attachment;filename=" + URLEncoder.encode(fileName,"UTF-8"));
        // 4.获取要下载文件的输入流,输出流
        FileInputStream fis = new FileInputStream(realPath);
        // 客户端要下载就要以响应的方式获取输出流
        ServletOutputStream sos = resp.getOutputStream();
        // 5.创建缓冲区: 读入写出操作
        int len;
        byte[] buffer = new byte[1024];
        while ((len = fis.read(buffer)) != -1) {
            // 写出到客户端
            sos.write(buffer,0,len);
        }
        // 资源关闭
        sos.close();
        fis.close();
    }
    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        doGet(req,resp);
    }
}

3、验证码功能

验证怎么来的?

  • 前端实现
  • 后端实现,需要用到 Java 的图片类,生产一个图片
public class ImageServlet extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        // setHeader()的常用功能
        // 让浏览器3秒自动刷新一次
        resp.setHeader("refresh", "3");
        // 在内存中创建一个图片对象: 参数分别为: 宽,高,颜色: RGB:三原色
        BufferedImage image = new BufferedImage(80, 20, BufferedImage.TYPE_INT_RGB);
        // 得到图片
        // 创建一个画笔
        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");
        // 网站存在缓存,缓存策略: 不让浏览器缓存
        resp.setDateHeader("expires", -1);
        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 sb = new StringBuffer();
        // 保证数组元素有七位数,不足七位数的位置用0来填充
        for (int i = 0; i < 7 - num.length(); i++) {
            sb.append(0);
        }
        num = sb.toString() + num;
        return num;
    }
    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        doGet(req,resp);
    }
}

你可能感兴趣的:(笔记,服务器,java,javaweb)