Javaweb

1.基本概念

1.1.web開發(fā):

  • 靜態(tài)web
    • html,css
    • 提供給所有人看的數(shù)據(jù)始終不會發(fā)生變化
  • 動態(tài)web:
    • 淘寶.
    • 提供給所有人看的數(shù)據(jù)始終會發(fā)生變化,每個人在不同時間不同地點(diǎn)看的信息各不相同
    • 技術(shù)棧: Servlet,asp,php
      在Java中,動態(tài)web資源開發(fā)的技術(shù)統(tǒng)稱為javaweb

1.2.web應(yīng)用程序

web應(yīng)用程序: 可以提供瀏覽器訪問的程序;

  • a.html,b.html...多個web資源可以被外界訪問,對外界提供服務(wù);
  • 能訪問到的任何一個頁面或者資源,都存在與這個世界的偶一個角落的計算機(jī)上
  • URL
  • 這個統(tǒng)一的web資源(應(yīng)用)會被放在同一個文件夾下,web應(yīng)用程序 --> Tomcat: 服務(wù)器
  • 一個web應(yīng)用由多部分組成(靜態(tài)web,動態(tài)web)
    • html,css,js
    • jsp,servlet
    • java程序
    • jar包
    • 配置文件 (Properties)
      web應(yīng)用程序編寫完畢后,若想提供給外界訪問: 要一個服務(wù)器統(tǒng)一管理

1.3.靜態(tài)web

  • .htm,html,這些都是網(wǎng)頁的后綴,如果服務(wù)器上一直存在這些東西,就可以直接讀取,通網(wǎng)絡(luò)
image.png
  • 靜態(tài)web存在的缺點(diǎn)

    • Web頁面無法動態(tài)更新,所有用戶看到都是同一個頁面
      • 輪播圖,點(diǎn)擊特效:偽動態(tài)
      • JavaScript [實際開發(fā)中,它用的最多]
      • VBScript
  • 它無法和數(shù)據(jù)庫交互(數(shù)據(jù)無法持久化,用戶無法交互)

1.4、動態(tài)web

頁面會動態(tài)展示: “Web的頁面展示的效果因人而異”;

網(wǎng)頁架構(gòu)圖:

image.png

jsp可以調(diào)用java程序

缺點(diǎn):

  • 加入服務(wù)器的動態(tài)web資源出現(xiàn)了錯誤,我們需要重新編寫我們的后臺程序,重新發(fā)布;
    • 停機(jī)維護(hù)

優(yōu)點(diǎn):

  • Web頁面可以動態(tài)更新,所有用戶看到都不是同一個頁面
  • 它可以與數(shù)據(jù)庫交互 (數(shù)據(jù)持久化:注冊,商品信息,用戶信息........ ; 放到數(shù)據(jù)庫里面,下次登錄不至于再也找不到自己,靜態(tài)web沒有連接數(shù)據(jù)庫不能做到)
image.png

動態(tài)web資源通過JDBC訪問數(shù)據(jù)庫的資源并返回到服務(wù)器

2、web服務(wù)器

2.1、三個技術(shù)講解

ASP:

  • 微軟:國內(nèi)最早流行的就是ASP;
  • 在HTML中嵌入了VB的腳本, ASP + DOM;
  • 在ASP開發(fā)中,基本一個頁面都有幾千行的業(yè)務(wù)代碼,頁面極其換亂
  • 維護(hù)成本高!
  • 在ASP中主要用C#語言
  • IIS
    ASP代碼
<h1>
    <h1><h1>
        <h1>
            <h1>
                <h1>
        <h1>
            <%
            System.out.println("hello")
            %>
            <h1>
                <h1>
   <h1><h1>
<h1>

php:

  • PHP開發(fā)速度很快,功能很強(qiáng)大,跨平臺,代碼很簡單 (70% , WP)
  • 無法承載大訪問量的情況(局限性)

**JSP/Servlet : **

B/S:瀏覽和服務(wù)器

C/S: 客戶端和服務(wù)器

  • 是sun公司主推的B/S架構(gòu)
  • 主要基于Java語言的 (所有的大公司,或者一些開源的組件,都是用Java寫的)
  • 可以承載三高問題帶來的影響: 高并發(fā),高可用,高性能
  • 語法像ASP , ASP易轉(zhuǎn)JSP , 加強(qiáng)市場強(qiáng)度;

還有更多的引擎......

2.2、web服務(wù)器(web server): 用來接收用戶的請求,并且負(fù)責(zé)給用戶返回一些響應(yīng)

服務(wù)器是一種被動的操作,用來處理用戶的一些請求和給用戶一些響應(yīng)信息;

IIS

微軟的; ASP...,Windows中自帶的

Tomcat 就是做一些這樣的事情

Tomcat 服務(wù)器是一個免費(fèi)的開放源代碼的Web 應(yīng)用服務(wù)器,屬于輕量級應(yīng)用服務(wù)器,在中小型系統(tǒng)和并發(fā)訪問用戶不是很多的場合下被普遍使用,是開發(fā)和調(diào)試JSP 程序的首選。對于一個Java初學(xué)web的人來說,它是最佳的選擇
Tomcat 實際上運(yùn)行JSP 頁面和Servlet。

3、Tomcat

3.1、 安裝tomcat

tomcat官網(wǎng):http://tomcat.apache.org/

image.png

直接解壓即可使用

3.2、Tomcat啟動和配置

文件夾作用:

image.png

啟動。關(guān)閉Tomcat

image.png

訪問測試:http://localhost:8080/

可能遇到的問題:

  1. Java環(huán)境變量(jre)沒有配置
  2. 閃退問題:需要配置兼容性
  3. 亂碼問題:配置文件中設(shè)置

3.3、配置

image.png

在server.xml文件里可以配置啟動的端口號

  • tomcat的默認(rèn)端口號為:8080
  • mysql:3306
  • http:80
  • https:443
<Connector port="8080" protocol="HTTP/1.1"
               connectionTimeout="20000"
               redirectPort="8443" />

還可以配置主機(jī)的名稱

  • 默認(rèn)的主機(jī)名為:localhost->127.0.0.1
  • 默認(rèn)網(wǎng)站應(yīng)用存放的位置為:webapps
 <Host name="www.xxx.com"  appBase="webapps"
        unpackWARs="true" autoDeploy="true">

高難度面試題:

請你談?wù)劸W(wǎng)站是如何進(jìn)行訪問的!

  1. 輸入一個域名;回車

  2. 檢查本機(jī)的 C:\Windows\System32\drivers\etc\hosts配置文件下有沒有訪問的這個域名映射;

    1. 若有:直接返回對應(yīng)的ip地址,這個地址中,有我們需要訪問的web程序,可以直接訪問

      127.0.0.1       www.xxx.com
      
    2. 若沒有:去DNS服務(wù)器找,找到的話就返回,找不到就返回找不到;

image.png

3.可以配置一下環(huán)境變量(可選性): 好處是隨時啟動

3.4、發(fā)布一個web網(wǎng)站

  • 將自己寫的網(wǎng)站,放到服務(wù)器(Tomcat)中指定的web應(yīng)用的文件夾(webapps)下,啟動Tomcat就可以訪問了

網(wǎng)站應(yīng)該有的結(jié)構(gòu)

這里面的文件都可以隨便建,因為可以訪問到html文件肯定就可以訪問到其他的文件

--webapps :Tomcat服務(wù)器的web目錄
    -ROOT
    -xxx :網(wǎng)站的目錄名
        - WEB-INF
            -classes : java程序
            -lib:web應(yīng)用所依賴的jar包
            -web.xml :網(wǎng)站配置文件
        - index.html 默認(rèn)的首頁
        - static 
            -css
                -style.css
            -js
            -img
         -..... 

4、Http

4.1、什么是HTTP

HTTP(超文本傳輸協(xié)議)是一個簡單的請求-響應(yīng)協(xié)議,它通常運(yùn)行在TCP之上。

  • 文本:html,字符串,~ ….
  • 超文本:圖片,音樂,視頻,定位,地圖…….
  • 默認(rèn)端口: 80

Https:安全的(+s: security)

  • 默認(rèn)端口: 443

4.2、兩個時代

  • http1.0時代

    • HTTP/1.0:客戶端可以與web服務(wù)器一次連接后,只能獲得一個web資源,斷開連接: 比如從百度上請求了一個index.html(一個web資源).第二次就不能請求了
  • http2.0時代

    • HTTP/1.1:客戶端可以與web服務(wù)器連接后,可以獲得多個web資源。不用多次請求

4.3、Http請求

  • 客戶端---發(fā)請求到(Request)---> 服務(wù)器
    客戶端請求百度:
Request URL:https://www.baidu.com/   請求地址
Request Method:GET    get方法/post方法
Status Code:200 OK    狀態(tài)碼:200
Remote(遠(yuǎn)程) 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、請求行

主要用來控制請求的協(xié)議

  • 請求行中的請求方式:GET
  • 請求方式有哪些:Get,Post,HEAD,DELETE,PUT,TRACT…
    • get:請求能夠攜帶的參數(shù)比較少,大小有限制,會在瀏覽器的URL地址欄顯示數(shù)據(jù)內(nèi)容,不安全,但高效
    • post:請求能夠攜帶的參數(shù)沒有限制,大小沒有限制,不會在瀏覽器的URL地址欄顯示數(shù)據(jù)內(nèi)容,安全,但不高效。

2、消息(請求)頭: Request Headers

Accept:告訴服務(wù)器,它所支持的數(shù)據(jù)類型
Accept-Encoding:支持哪種編碼格式  GBK   UTF-8(數(shù)據(jù)庫用的)   GB2312  ISO8859-1(java默認(rèn)編碼)
Accept-Language:告訴服務(wù)器,它的語言環(huán)境
Cache-Control:緩存控制(例如: 用戶登錄緩存)
Connection:告訴服務(wù)器,請求完成是斷開還是保持連接
HOST:表示主機(jī)..../.

4.4、Http響應(yīng)

  • 服務(wù)器---響應(yīng)給-----> 客戶端

百度響應(yīng)給客戶端:

Cache-Control:private    緩存控制
Connection:Keep-Alive    保持連接: 現(xiàn)在能連接上,走了HTTP/1.1
Content-Encoding:gzip    編碼
Content-Type:text/html   響應(yīng)回的類型

1.響應(yīng)體

Accept:告訴客戶端,它所支持的數(shù)據(jù)類型
Accept-Encoding:支持哪種編碼格式  GBK   UTF-8   GB2312  ISO8859-1
Accept-Language:告訴客戶端,它的語言環(huán)境
Cache-Control:緩存控制
Connection:告訴客戶端,請求完成是斷開還是保持連接
HOST:主機(jī)..../.
Refresh:告訴客戶端,多久刷新一次;
Location:讓網(wǎng)頁重新定位;

2、響應(yīng)狀態(tài)碼 (重點(diǎn))

200:請求響應(yīng)成功 200

3xx:請求重定向

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

4xx:找不到資源 404 禁止訪問: 403

  • 資源不存在;

5xx:服務(wù)器代碼錯誤 500 502:網(wǎng)關(guān)錯誤: Bad Gateway 作為網(wǎng)關(guān)或者代理工作的服務(wù)器嘗試執(zhí)行請求時,從遠(yuǎn)程服務(wù)器接收到了一個無效的響應(yīng).

常見面試題:

當(dāng)你的瀏覽器中地址欄輸入地址并回車的一瞬間到頁面能夠展示回來,經(jīng)歷了什么?

5、Maven

我為什么要學(xué)習(xí)這個技術(shù)?

  1. 在Javaweb開發(fā)中,需要使用大量的jar包,我們需要手動去導(dǎo)入;

  2. 如何能夠讓一個東西自動幫我導(dǎo)入和配置這個jar包。

    由此,Maven誕生了!

5.1 Maven: 項目架構(gòu)管理工具

目前用來就是方便導(dǎo)入jar包的!
Maven的核心思想:約定大于配置

  • 有約束,不要去違反。(別人制定的規(guī)則,必須這么做)
    能夠防止出現(xiàn)各種各樣以外情況,以及解決不了的問題

Maven會規(guī)定好你該如何去編寫我們的Java代碼,目錄結(jié)構(gòu)必須要按照這個規(guī)范來;

5.2 下載安裝Maven

官網(wǎng);https://maven.apache.org/

image.png

下載完成后,解壓即可;

5.3 配置環(huán)境變量

在我們的系統(tǒng)環(huán)境變量中

配置如下配置:

  • M2_HOME maven目錄下的bin目錄

  • MAVEN_HOME maven的目錄

  • 在系統(tǒng)變量下的path中配置 %MAVEN_HOME%\bin

image.png

測試Maven是否安裝成功,保證必須配置完畢!

5.4 添加阿里云鏡像

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

image.png
  • 鏡像:mirrors
    • 作用:加速我們的下載
  • 國內(nèi)建議使用阿里云的鏡像
<mirror>
    <id>nexus-aliyun</id>  
    <mirrorOf>*,!jeecg,!jeecg-snapshots</mirrorOf>  
    <name>Nexus aliyun</name>  
    <url>http://maven.aliyun.com/nexus/content/groups/public</url> 
</mirror>

5.5 本地倉庫

在本地的倉庫,遠(yuǎn)程倉庫;

建立一個本地倉庫:localRepository

  1. 先在maven文件下創(chuàng)建一個新文件夾maven-repo
image.png

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

image.png

5.6、在IDEA中使用Maven

  1. 啟動IDEA
  2. 創(chuàng)建一個MavenWeb項目
image.png

下一步:

image.png
image.png

下一步:

image.png
image.png

3.等待項目初始化完畢

  • 出現(xiàn)這個說項目搭建成功
image.png
  1. 觀察maven倉庫中多了什么東西?

  2. IDEA中的Maven設(shè)置

    注意:IDEA項目創(chuàng)建成功后,看一眼Maven的配置

image.png
image.png

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

5.7、創(chuàng)建一個普通(純凈)的Maven項目

image.png
image.png

下面這個只有在Web應(yīng)用(webapp)下才會有!

image.png

5.8 標(biāo)記文件夾功能(給文件夾賦予功能)

方式一:

image.png

方式二:

image.png
image.png
image.png

5.9 在 IDEA中配置Tomcat

image.png
image.png
image.png
  • 若下面報ERROR,則是沒有配置本地Tomcat服務(wù)器地址:Application server
image.png

解決警告問題

必須要的配置:為什么會有warning這個問題:我們訪問一個網(wǎng)站,需要指定一個文件夾名字;

image.png
image.png
image.png
image.png

5.10 pom文件

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

image.png
<?xml version="1.0" encoding="UTF-8"?>
<!--Maven的版本和頭文件-->
<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.0</modelVersion>
<!--這里就是剛剛配置的組id,項目名,版本-->
  <groupId>com.tiga</groupId>
  <artifactId>javaweb-01-maven</artifactId>
  <version>1.0-SNAPSHOT</version>
<!--    Package: 項目打包方式
   jar: java應(yīng)用
   war: javaWeb應(yīng)用
-->
  <packaging>war</packaging>

  <name>javaweb-01-maven Maven Webapp</name>
  <!-- FIXME change it to the project's website -->
  <url>http://www.example.com</url>
<!--配置-->
  <properties>
<!--      項目的默認(rèn)構(gòu)建編碼-->
    <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
<!--      編碼版本-->
    <maven.compiler.source>1.8</maven.compiler.source>
    <maven.compiler.target>1.8</maven.compiler.target>
  </properties>
<!--項目依賴-->
  <dependencies>
<!--      具體依賴的jar包配置文件-->
    <dependency>
      <groupId>junit</groupId>
      <artifactId>junit</artifactId>
      <version>4.11</version>
    </dependency>
  </dependencies>
<!--項目構(gòu)建的東西-->
  <build>
    <finalName>javaweb-01-maven</finalName>
    <pluginManagement><!-- lock down plugins versions to avoid using Maven defaults (may be moved to parent pom) -->
      <plugins>
        <plugin>
          <artifactId>maven-clean-plugin</artifactId>
          <version>3.1.0</version>
        </plugin>
        <!-- see http://maven.apache.org/ref/current/maven-core/default-bindings.html#Plugin_bindings_for_war_packaging -->
        <plugin>
          <artifactId>maven-resources-plugin</artifactId>
          <version>3.0.2</version>
        </plugin>
        <plugin>
          <artifactId>maven-compiler-plugin</artifactId>
          <version>3.8.0</version>
        </plugin>
        <plugin>
          <artifactId>maven-surefire-plugin</artifactId>
          <version>2.22.1</version>
        </plugin>
        <plugin>
          <artifactId>maven-war-plugin</artifactId>
          <version>3.2.2</version>
        </plugin>
        <plugin>
          <artifactId>maven-install-plugin</artifactId>
          <version>2.5.2</version>
        </plugin>
        <plugin>
          <artifactId>maven-deploy-plugin</artifactId>
          <version>2.8.2</version>
        </plugin>
      </plugins>
    </pluginManagement>
  </build>
</project>
image.png

maven由于他的約定大于配置,我們之后可以能遇到我們寫的配置文件,無法被導(dǎo)出或者生效的問題,解決方案:

問題見于: mybatis的靜態(tài)資源過濾問題
問題是: 比如在java目錄下maven約定死了只能寫java,如果在該目錄下寫xml格式文件是導(dǎo)不出的
需要在src/main/java目錄下讓他可以包含.properties,.xml這樣的文件,這樣java目錄的這些文件就可以導(dǎo)出了
讓他不去過濾這些文件

<!--在build中配置resources,來防止我們資源導(dǎo)出失敗的問題-->
    <!--問題見于: mybatis的靜態(tài)資源過濾問題
    問題是: 比如在java目錄下maven約定死了只能寫java,如果在該目錄下寫xml格式文件是導(dǎo)不出的
    需要在src/main/java目錄下讓他可以包含.properties,.xml這樣的文件,這樣java目錄的這些文件就可以導(dǎo)出了
    讓他不去過濾這些文件-->
    <build>
        <resources>
            <resource>
                <directory>src/main/resources</directory>
                <includes>
                    <include>**/*.properties</include>
                    <include>**/*.xml</include>
                </includes>
                <filtering>true</filtering>
            </resource>
            <resource>
                <directory>src/main/java</directory>
                <includes>
                    <include>**/*.properties</include>
                    <include>**/*.xml</include>
                </includes>
                <filtering>true</filtering>
            </resource>
        </resources>
    </build>

5.12 IDEA操作

  • 熱部署
    文件資源自動更新

5.13 解決遇到的問題

1.IDEA中每次都要重復(fù)配置Maven
在IDEA中的全局默認(rèn)配置中去配置

image.png
image.png
  1. Maven項目中Tomcat無法配置
  2. maven默認(rèn)web項目中的web.xml版本問題
image.png

4.替換為webapp中的web.xml為4.0版本和tomcat服務(wù)器的web.xml一致

<?xml version="1.0" encoding="UTF-8"?>
<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/

  • 第一次導(dǎo)入maven依賴,如果在IDEA中搜不到就去Maven倉庫去找,不知道接口名,可以從啟動的地方往上推,比如說啟動Tomcat需要用到servlet-api-jar包,就去搜關(guān)于這的jar包
image.png
image.png

6、Servlet

什么是Servlet?
Servlet是JavaWeb開發(fā)的基石,與平臺無關(guān)的服務(wù)器組件,他是運(yùn)行在Servlet容器/Web應(yīng)用服務(wù)器/Tomcat,負(fù)責(zé)與客戶端進(jìn)行通信.
Servlet的功能:
1.創(chuàng)建并返回基于客戶端請求的動態(tài)HTML頁面.
2.與數(shù)據(jù)庫進(jìn)行通信.

image.png
  • 如何使用Servlet?
    Servlet本身是一組接口,就是java程序中描述某一個東西具備某一種功能的操作,是抽象化的,是可以作為一個后臺應(yīng)用的功能,這個接口由java提供存放在 javax.servlet中,java.lang,java.util,java指的是最基礎(chǔ)的那套東西,javax指的是后來擴(kuò)展出來的東西.
    自定義一個類,并實現(xiàn)Servlet接口,這個類就具備了接收客戶端請求以及做出響應(yīng)的功能.
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對象調(diào)用輸出方法,就能把消息返回給客戶端
        servletResponse.getWriter().println("接收到的參數(shù)是: " + id);
        servletResponse.getWriter().write("客戶端你好,我是servlet容器");
    }

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

    @Override
    public void destroy() {

    }
}
瀏覽器不能直接訪問Servlet文件,只能通過映射的方式來間接訪問Servlet,映射需要開發(fā)者手動配置,有兩種配置方式.
- 基于xml文件的配置方式

- 基于注解的配置方式

6.1、Servlet簡介

  • Servlet就是sun公司開發(fā)動態(tài)web的一門技術(shù)
  • Sun在這些API中提供一個接口叫做:Servlet,如果你想開發(fā)一個Servlet程序,只需要完成兩個小步驟:
    • 編寫一個類,實現(xiàn)Servlet接口
    • 把開發(fā)好的Java類部署到web服務(wù)器中。

把實現(xiàn)了Servlet接口的Java程序叫做,Servlet

6.2、HelloServlet

Serlvet接口Sun公司有兩個默認(rèn)的實現(xiàn)類:HttpServlet,GenericServlet

HttpServlet中繼承的services 方法里面做的事情是調(diào)用doGet或者doPost方法

doGet方法是在瀏覽器頂部欄中輸入被調(diào)用
doPost方法是在表單中輸入被調(diào)用

  1. 構(gòu)建一個普通的Maven項目,刪掉里面的src目錄,以后我們的學(xué)習(xí)就在這個項目里面建立Module;因為IDEA每次只能展示一個project這個空的工程就是Maven主工程;

  2. 關(guān)于Maven父子工程的理解:

    父項目中會有

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

    子項目會有

沒有parent的原因: 去父工程的依賴中刪除scope標(biāo)簽,重建創(chuàng)建子模塊即可

    <parent>
        <artifactId>javaweb-02-servlet</artifactId>
        <groupId>com.tiga</groupId>
        <version>1.0-SNAPSHOT</version>
    </parent>

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

son extends father
  1. Maven環(huán)境優(yōu)化

    1. 修改web.xml為最新的
    2. 將maven的結(jié)構(gòu)搭建完整(在main目錄下創(chuàng)建java和resources目錄并標(biāo)記功能)

4.編寫一個Servlet程序
在主工程的module下的java目錄下創(chuàng)建package(命名規(guī)范: 里面放什么就用其命名)

    1. 編寫一個普通類
    1. 間接實現(xiàn)Servlet接口,這里我們直接繼承HttpServlet
public class HelloServlet extends HttpServlet {
    // 重寫HttpServlet中的doGet和doPost方法
    // 由于get或post只是請求(request)實現(xiàn)的不同的方式,所以可以互相調(diào)用,業(yè)務(wù)邏輯都一樣
    // 假設(shè)方法里有存在的業(yè)務(wù),在里面寫jdbc代碼讓他往里存數(shù)據(jù)庫,兩種方法都一樣,不需要再把同樣的代碼在另一種方法抄一遍
    // 最快的方法是在另一個方法調(diào)用當(dāng)前的方法,這兩個方法誰調(diào)誰都無所謂,所以代碼在一種方法里寫就行了
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        System.out.println("進(jìn)入doGet方法");// 在控制臺打印
        // 文件流(響應(yīng)流)寫出需要response的對象
        // 創(chuàng)建響應(yīng)流(輸出流)
        // 下面這段要注釋掉,否則瀏覽器會報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服務(wù)器,所以我們需要再web服務(wù)中注冊我們寫的Servlet,還需給他一個瀏覽器能夠訪問的路徑;

 <!--    注冊Serlvet-->
    <servlet>
<!--        注冊名和請求路徑名要一致-->
        <servlet-name>hello</servlet-name>
<!--        一個Servlet對應(yīng)一個class(類),或者說一個入口對應(yīng)一個Serlvet,要注冊的Servlet的所屬類-->
        <servlet-class>com.tiga.servlet.HelloServlet</servlet-class>
    </servlet>
<!--    Servlet的請求(映射)路徑-->
    <servlet-mapping>
<!--        hello都是隨便起的名字,只要能映射到上面的serlet-class就可以了-->
        <servlet-name>hello</servlet-name>
        <!--    給要請求的訪問路徑-->
<!--        只要去前端請求一個/hello,他就會走HelloServlet類,而這個類里就有處理方式,他就會調(diào)里面的結(jié)構(gòu)-->
        <url-pattern>/hello</url-pattern>
    </servlet-mapping>

6.配置Tomcat

  • 添加Tomcat Server - Local
  • 添加Deployment的war包
  • 添加發(fā)布路徑
    注意:配置項目發(fā)布的路徑就可以了

7.啟動測試,OK!
啟動后出現(xiàn)target,會生成一個網(wǎng)站,生成一個war包(就是一個網(wǎng)站),war包上面的文件夾就是其解壓包,解壓包的WEB-INF下的classes文件夾里就是java寫的類;
lib文件夾里存放的是導(dǎo)入的jar包
首先會訪問Tomcat配置的路徑下默認(rèn)的index.jsp, 然后在瀏覽器訪問地址后加上在web.xml配置的Servlet映射的url-pattern地址,訪問后他會到web.xml下找url-pattern,沿著往上找到了name,最后找到映射Servlet注冊的處理類指向的java的類

6.3、Servlet原理

就是我把請求給你,你去處理,然后再把結(jié)果給我

早期的web應(yīng)用主要應(yīng)用于瀏覽靜態(tài)網(wǎng)頁,HTTP服務(wù)器(比如Apache、Nginx)向瀏覽器返回靜態(tài)HTML,瀏覽器負(fù)責(zé)解析HTML,將結(jié)果呈現(xiàn)給用戶。
隨著互聯(lián)網(wǎng)發(fā)展,我們已不滿足于僅僅瀏覽靜態(tài)網(wǎng)頁,還希望通過一些交互操作來獲取動態(tài)結(jié)果,因此也就需要一些擴(kuò)展機(jī)制能夠讓Http服務(wù)器調(diào)用服務(wù)端程序。
于是Sun公司推出了Servlet容器??梢园裺ervlet簡單理解為運(yùn)行在服務(wù)端的java小程序,但是servlet沒有main方法,不能獨(dú)立運(yùn)行,因此必須把它部署到serlet容器,由容器來實例化并調(diào)用servlet。
而Tomcat和jetty就是一個Servlet容器。為了方便使用,他們也具有Http協(xié)議服務(wù)器的功能,因此Tomcat或jetty就是“HTTP服務(wù)器+Servlet容器”,我們也叫它Web容器。

  • Tomcat 與 Servlet 配合工作

步驟:
Web Client 向Servlet容器(Tomcat)發(fā)出Http請求
Servlet容器接收Web Client的請求
Servlet容器創(chuàng)建一個HttpRequest對象,將Web Client請求的信息封裝到這個對象中。
Servlet容器創(chuàng)建一個HttpResponse對象
Servlet容器調(diào)用HttpServlet對象的service方法,把HttpRequest對象與HttpResponse對象作為參數(shù)傳給 HttpServlet 對象。
HttpServlet調(diào)用HttpRequest對象的有關(guān)方法,獲取Http請求信息。
HttpServlet調(diào)用HttpResponse對象的有關(guān)方法,生成響應(yīng)數(shù)據(jù)。
Servlet容器把HttpServlet的響應(yīng)結(jié)果傳給Web Client。

  • Servlet工作原理

1、首先簡單解釋一下Servlet接收和響應(yīng)客戶請求的過程,首先客戶發(fā)送一個請求,Servlet是調(diào)用service()方法對請求進(jìn)行響應(yīng)的,通過源代碼可見,service()方法中對請求的方式進(jìn)行了匹配,選擇調(diào)用doGet,doPost等這些方法,然后再進(jìn)入對應(yīng)的方法中調(diào)用邏輯層的方法,實現(xiàn)對客戶的響應(yīng)。在Servlet接口和GenericServlet中是沒有doGet()、doPost()等等這些方法的,HttpServlet中定義了這些方法,但是都是返回error信息,所以,我們每次定義一個Servlet的時候,都必須實現(xiàn)doGet或doPost等這些方法。

2、每一個自定義的Servlet都必須實現(xiàn)Servlet的接口,Servlet接口中定義了五個方法,其中比較重要的三個方法涉及到Servlet的生命周期,分別是上文提到init(),service(),destroy()方法。GenericServlet是一個通用的,不特定于任何協(xié)議的Servlet,它實現(xiàn)了Servlet接口。而HttpServlet繼承于GenericServlet,因此HttpServlet也實現(xiàn)了Servlet接口。所以我們定義Servlet的時候只需要繼承HttpServlet即可。

3、Servlet接口和GenericServlet是不特定于任何協(xié)議的,而HttpServlet是特定于HTTP協(xié)議的類,所以HttpServlet中實現(xiàn)了service()方法,并將請求ServletRequest、ServletResponse 強(qiáng)轉(zhuǎn)為HttpRequest 和 HttpResponse。

  • 創(chuàng)建Servlet對象的時機(jī):

Servlet容器啟動時:讀取web.xml配置文件中的信息,構(gòu)造指定的Servlet對象,創(chuàng)建ServletConfig對象,同時將ServletConfig對象作為參數(shù)來調(diào)用Servlet對象的init方法。

在Servlet容器啟動后:客戶首次向Servlet發(fā)出請求,Servlet容器會判斷內(nèi)存中是否存在指定的Servlet對象,如果沒有則創(chuàng)建它,然后根據(jù)客戶的請求創(chuàng)建HttpRequest、HttpResponse對象,從而調(diào)用Servlet 對象的service方法。

Servlet容器在啟動時自動創(chuàng)建Servlet,這是由在web.xml文件中為Servlet設(shè)置的<load-on-startup>屬性決定的。從中我們也能看到同一個類型的Servlet對象在Servlet容器中以單例的形式存在。

Servlet是由Web服務(wù)器調(diào)用,web服務(wù)器在收到瀏覽器請求之后,會:

image.png

6.4、Mapping問題

  1. 一個Servlet可以指定一個映射路徑
<servlet-mapping>
        <servlet-name>hello</servlet-name>
        <url-pattern>/hello</url-pattern>
    </servlet-mapping>

2.一個Servlet可以指定多個映射路徑

 <servlet-mapping>
        <servlet-name>hello</servlet-name>
        <url-pattern>/hello</url-pattern>
    </servlet-mapping>
    <servlet-mapping>
        <servlet-name>hello</servlet-name>
        <url-pattern>/hello2</url-pattern>
    </servlet-mapping>
    <servlet-mapping>
        <servlet-name>hello</servlet-name>
        <url-pattern>/hello3</url-pattern>
    </servlet-mapping>
    <servlet-mapping>
        <servlet-name>hello</servlet-name>
        <url-pattern>/hello4</url-pattern>
    </servlet-mapping>
    <servlet-mapping>
        <servlet-name>hello</servlet-name>
        <url-pattern>/hello5</url-pattern>
    </servlet-mapping>

3.一個Servlet可以指定通用映射路徑

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

4.默認(rèn)請求路徑(盡量少用)

<!--    默認(rèn)請求HelloServlet類路徑,不會訪問index.jsp-->
    <servlet-mapping>
        <servlet-name>hello</servlet-name>
        <url-pattern>/*</url-pattern>
    </servlet-mapping>

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

<!--    可以自定義后綴實現(xiàn)請求映射
        例如: hello/uihfhfos.tiga
        注意點(diǎn): *前面不能加項目映射的路徑
        例如不能: /hello/xxx.tiga
-->
    <servlet-mapping>
        <servlet-name>hello</servlet-name>
        <url-pattern>*.tiga</url-pattern>
    </servlet-mapping>

6.優(yōu)先級問題
假如精準(zhǔn)匹配,越準(zhǔn)確優(yōu)先級越高
指定了固有的映射路徑優(yōu)先級最高,如果找不到就會走默認(rèn)的處理請求;

<!--    注冊404的Servlet-->
    <servlet>
        <servlet-name>error</servlet-name>
        <servlet-class>com.tiga.servlet.ErrorServlet</servlet-class>
    </servlet>
<!--    映射請求路徑-->
    <servlet-mapping>
        <servlet-name>error</servlet-name>
<!--        映射路徑輸入指定存在的則會返回指定存在的頁面而不會返回404-->
        <url-pattern>/*</url-pattern>
    </servlet-mapping>

6.5、ServletContext

web容器(Tomcat)在啟動的時候,它會為每個web程序都創(chuàng)建一個對應(yīng)的ServletContext對象,它代表了當(dāng)前的web應(yīng)用;

1、共享數(shù)據(jù)

我在這個Servlet中保存的數(shù)據(jù),可以在另外一個servlet中拿到;

image.png
  • 會被request,session替代,這種方法盡量少用
public class HelloServlet extends HttpServlet {
    // 重寫doGet,doPost方法
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        // 當(dāng)前類的對象調(diào)用繼承父類的方法
        // this.getInitParameter();// 初始化參數(shù)
        // this.getServletConfig();// Servlet配置
        // this.getServletContext();// Servlet上下文環(huán)境
        // 每個web工程中只有一個ServletContext對象
        // 代表當(dāng)前web應(yīng)用,是web程序的映射,這個網(wǎng)站由這個對象來管理,公共的,可共享數(shù)據(jù)
        ServletContext context = this.getServletContext();
        String username = "tiga";
        // 先存
        // 將一個數(shù)據(jù)保存在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 {
        // 當(dāng)前類的對象調(diào)用繼承父類的方法
        // this.getInitParameter();// 初始化參數(shù)
        // this.getServletConfig();// Servlet配置
        // this.getServletContext();// Servlet上下文環(huán)境
        // 每個web工程中只有一個ServletContext對象
        // 代表當(dāng)前web應(yīng)用,是web程序的映射,這個網(wǎng)站由這個對象來管理,公共的,可共享數(shù)據(jù)
        ServletContext context = this.getServletContext();
        String username = "tiga";
        // 先存
        // 將一個數(shù)據(jù)保存在ServletContext中,(key值)名字為: username,(value)值為: username對象
        context.setAttribute("username",username);
    }
    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        doGet(req,resp);
    }
}
<!--注冊HelloServlet類的Serlvet-->
    <servlet>
        <servlet-name>hello</servlet-name>
        <servlet-class>com.tiga.servlet.HelloServlet</servlet-class>
    </servlet>
<!--    映射地址-->
    <servlet-mapping>
        <servlet-name>hello</servlet-name>
        <url-pattern>/hello</url-pattern>
    </servlet-mapping>
<!--    注冊GetServlet類的Servlet-->
    <servlet>
        <servlet-name>getc</servlet-name>
        <servlet-class>com.tiga.servlet.GetServlet</servlet-class>
    </servlet>
    <servlet-mapping>
        <servlet-name>getc</servlet-name>
        <url-pattern>/getc</url-pattern>
    </servlet-mapping>

2、獲取初始化參數(shù)

<!--    配置web應(yīng)用初始參數(shù): 在web初始就有了-->
    <context-param>
<!--        key-value形式-->
        <param-name>url</param-name>
<!--        url格式: 接口:數(shù)據(jù)庫類型://主機(jī)名:端口號/數(shù)據(jù)庫-->
<!--        配置一個jdbc連接到數(shù)據(jù)庫的url-->
        <param-value>jdbc:mysql//localhost:3306/mybatis</param-value>
    </context-param>
@Override
    // 從req參數(shù)接收請求處理,再到resp參數(shù)請求響應(yīng)回去
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        // context的應(yīng)用
        ServletContext context = this.getServletContext();
        // 可以在web.xml中配置一些web應(yīng)用初始化參數(shù)
        // 在web.xml中配置完獲取初始化的參數(shù), 形參為web.xml中初始化參數(shù)的參數(shù)名,鍵值對方式
        String url = context.getInitParameter("url");
        resp.getWriter().print(url);
    }

3、請求轉(zhuǎn)發(fā)

  • 轉(zhuǎn)發(fā)路徑不變,頁面改變; 重定向路徑改變,頁面也改變

這里注意重定向時A(客戶端)是發(fā)送了兩次請求,請求轉(zhuǎn)發(fā)時A只發(fā)送了一次請求,而服務(wù)器端對該請求進(jìn)行了轉(zhuǎn)發(fā)

public class ServletDemo02 extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        // 獲取ServletContext對象
        ServletContext context = this.getServletContext();
        // 轉(zhuǎn)發(fā)和重定向的區(qū)別: 轉(zhuǎn)發(fā)像中間商,重定向像媒婆介紹新人
        // 這里只是做了轉(zhuǎn)發(fā)的作用
        // 獲取請求轉(zhuǎn)發(fā),方法形參是要轉(zhuǎn)發(fā)的地址,forward(): 代表轉(zhuǎn)發(fā)的意思
        // RequestDispatcher requestDispatcher = context.getRequestDispatcher("/gp");// 轉(zhuǎn)發(fā)的請求路徑
        // 調(diào)用forward實現(xiàn)請求轉(zhuǎn)發(fā),參數(shù)1: 請求對象 參數(shù)2: 響應(yīng)對象
        // requestDispatcher.forward(req,resp);
        // 直接合并實現(xiàn)請求轉(zhuǎn)發(fā)
        // 記住這個請求轉(zhuǎn)發(fā),這是springmvc dispatchServlet的實現(xiàn)原理
        context.getRequestDispatcher("/gp").forward(req,resp);
    }

轉(zhuǎn)發(fā)和重定向

image.png
1567924457532.png

4、讀取資源文件

Properties

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

Servlet運(yùn)行時,在解壓觸點(diǎn)war包下的classes,目錄下的java下創(chuàng)建的properties文件沒有生成,由于maven中配置pom.xml約定大于配置的原因,資源導(dǎo)出可能存在問題,所以要在pom.xml中添加build配置防止導(dǎo)出失敗

成功導(dǎo)出后,發(fā)現(xiàn):都被打包到了同一個路徑下:classes,我們俗稱這個路徑為classpath:
思路:需要一個文件流;

username=root613516
password=1234ioregroe
public class PropertiesServlet extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        // 獲取Servlet上下文環(huán)境,并把資源變成流
        // 路徑格式: /: 不能省略,代表當(dāng)前web應(yīng)用(項目),或當(dāng)前文件夾,這個項目指的是打包完之后的項目
        // 用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");
        // 修改響應(yīng)編碼格式
        resp.setContentType("text/html;charset=utf-8");
        // 響應(yīng)請求到客戶端
        resp.getWriter().write(user + ":" + pwd);
        // 去webapp下注冊映射地址
    }

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

6.6、HttpServletResponse

web服務(wù)器接收到客戶端的http請求,針對這個請求,分別創(chuàng)建一個代表請求的HttpServletRequest對象,代表響應(yīng)的一個HttpServletResponse;

  • 如果要獲取客戶端請求過來的參數(shù):找HttpServletRequest
  • 如果要給客戶端響應(yīng)一些信息:找HttpServletResponse

1、簡單分類

負(fù)責(zé)向瀏覽器發(fā)送數(shù)據(jù)的方法

ServletOutputStream getOutputStream() throws IOException;
PrintWriter getWriter() throws IOException;

負(fù)責(zé)向瀏覽器發(fā)送響應(yīng)頭的方法

    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);

響應(yīng)的狀態(tài)碼

    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、下載文件

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容