Javaweb学习

文章目录

Javaweb学习

1.基本概念

1.1前言

web开发:

  • web, 网页的意思,www.baidu.com

  • 静态web

    • html, css
    • 提供给所有人看的数据始终不会变化!
  • 动态web

    • 淘宝,几乎所有的网站。

    • 提供给所有人看的数据始终会变化!每个人在不同的时间,在不同的地点看到的信息各不相同!

    • 技术栈:servlet,jsp,ASP,PHP。

在java中,动态web开发的技术统称为javaWeb:

1.2web应用程序

什么是程序:可以提供浏览器访问的程序;

  • a.html b.html…多个web资源,这些资源可以被外界访问,对外界提供服务;
  • 你们能访问到的任何一个资源,都存在在世界的某一个角落的计算机上。
  • URL
  • 这些统一的web资源会被放在同一个文件夹,web应用程序->Tomcat:服务器
  • 一个web应用由多部分组成(静态web,动态web)
    • html,css,js
    • jsp,servlet
    • java程序
    • jar包
    • 配置文件(properties)

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

1.3静态web
  • .html,.html这些都是网页的后缀,如果服务器上一直存在这些东西,我们就可以直接进行读取。通络:
image-20201006103151298
  • 静态web存在的缺点
    • web页面无法动态更新,所有用户看到的是同一个页面
      • 轮播图,点击特效:伪动态
      • JavaScript[实际开发中,他用的最多]
      • VBScript
    • 无法和数据库交互(数据无法持久化,用户无法交互)
1.4动态web

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

image-20201006103846117

缺点:

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

有点:

  • web页面无法动态更新,所有用户看到的不是同一个页面
  • 他可以和数据库交互(数据持久化:注册,商品信息,用户信息。。。)
image-20201006104206453

新手村:–魔鬼训练(分析原理,看源码)–>PK厂

2.Web服务器


2.1技术讲解

ASP

  • 微软:国内最早流行的就是ASP;
  • 在HTML中嵌入了VB脚本,ASP+COM;
  • 在ASP开发中,基本一个页面都有几千行的业务代码,页面机器混乱
  • 维护成本高!
  • C#
  • IIS服务器

PHP

  • PHP开发速度很快,功能很强大,跨平台,代码简单(70%,WP)

  • 无法承载大访问量的情况(局限性)

JSP/Servlet:

B/S:浏览器和服务器

C/S:客户端和服务器

  • sun公司主推的B/S架构

  • 基于java语言的(所有的大公司,或者一些开源的组件,都是用java写的)

  • 可以承载三高问题带来的影响;

  • 语法像ASP,ASP->jsp,加强市场强度;

2.2Web服务器

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

IIS

  • 微软的:ASP…Windows中自带的

Tomcat

image-20201006110453774

面向百度编程:

Tomcat是Apache 软件基金会(Apache Software Foundation)的Jakarta 项目中的一个核心项目,由Apache、Sun 和其他一些公司及个人共同开发而成。由于有了Sun 的参与和支持,最新的Servlet 和JSP 规范总是能在Tomcat 中得到体现,Tomcat 5支持最新的Servlet 2.4 和JSP 2.0 规范。因为Tomcat 技术先进、性能稳定,而且免费,因而深受Java 爱好者的喜爱并得到了部分软件开发商的认可,成为目前比较流行的Web 应用服务器。

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

诀窍是,当配置正确时,Apache 为HTML页面服务,而Tomcat 实际上运行JSP 页面和Servlet。目前Tomcat最新版本为9.0.37**。**

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

下载tomcat:

1.安装 or 解压;

2.了解配置文件及目录文件

3.这个东西的作用

3.Tomcat

3.1安装Tomcat

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

image-20201006111847110image-20201006111911045

image-20201006111924512
3.2Tomcat使用和配置

文件夹的作用:

image-20201006112639289

启动,关闭Tomcat

image-20201006112807654

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

可能遇到的问题:

​ 1.java环境变量没有配置

​ 2.闪退问题:需要配置兼容性

​ 3.乱码问题:修改配置文件

3.3配置
image-20201006113958098

可以配置主机的端口号

  • 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="localhost"  appBase="webapps"
            unpackWARs="true" autoDeploy="true">

高难度面试题

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

1.输入一个域名:回车

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

​ 1.有 ,直接返回对应的IP地址。这个地址中有我们需要访问的Web程序, 可以直接访问

		127.0.0.1   www.gxl.com

​ 2.没有:去DNS服务器找

image-20201006115631006

3.配置一下环境变量(可选性)

3.4发布一个web网站

不会就先模仿

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

网站应该有的结构

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

HTTP协议:面试

Maven:构建工具

  • Maven安装包

Servlet入门

  • HelloWorld!
  • Servlet配置
  • 原理

4.Http


4.1什么是Http

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

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

Https(安全的):默认端口:443

4.2两个时代
  • http1.0
    • Http/1.0:客户端可以与web服务器连接后,只能获得一个web资源,断开连接
  • http2.0
    • HTTP/1.1:客户端可以与web服务器连接后,只能获得多个web资源
4.3Http请求
  • 客户端—发请求到(Request)—服务器

百度:

请求 URL: https://www.baidu.com/		请求地址
请求方法: GET						get方法/post方法
状态代码: 200 OK					状态码:200
远程地址: 39.156.66.18:443			
Accept-Encoding: gzip, deflate, br
Accept-Language: zh-CN,zh;q=0.9,en;q=0.8,en-GB;q=0.7,en-US;q=0.6
Cache-Control: max-age=0
Connection: keep-alive
1.请求行
  • 请求行中的请求方式:GET
  • 请求方式:Get,Post…
    • get:请求能够携带的参数比较小,大小有限制,会在浏览器的URL地址栏显示数据内容,不安全,但高效。
    • post:请求能够携带的参数没有限制,大小没有限制,不会在浏览器的URL地址栏显示数据内容,安全,但不高效。
2.消息头
Accept:告诉浏览器,它所支持的类型
Accept-Encoding: 支持哪种编码格式  GBK  UTF-8   GB2312   ISO8859-1
Accept-Language: 告诉浏览器,他的语言环境
Cache-Control: 缓存控制
Connection: 高速浏览器,请求完成是断开还是保持连接
Host:主机..../
4.4Http响应
  • 服务器—响应—客户端

百度:

Cache-Control: private			缓存控制
Connection: keep-alive			连接:保持活跃
Content-Encoding: gzip			编码
Content-Type: text/html;		类型
1.响应体
Accept:告诉浏览器,它所支持的类型
Accept-Encoding: 支持哪种编码格式  GBK  UTF-8   GB2312   ISO8859-1
Accept-Language: 告诉浏览器,他的语言环境
Cache-Control: 缓存控制
Connection: 高速浏览器,请求完成是断开还是保持连接
Host:主机..../
Reflush:告诉客户端,多久刷新一次
Location:让网页重新定位
2.响应状态码

200:请求响应成功

3**:请求重定向

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

4xx:找不到资源 404

  • 资源不存在;

5xx:服务器代码错误 500 502:网关错误

常见面试题:

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

5Maven

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

1.在javaweb开发中,与要使用大量的jar包,我们手动去导入。

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

​ 由此,Maven诞生了

5.1Maven项目架构管理工具

我们目前就是用它来导入jar包的!

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

  • 有约束不要去违反。

Maven会规定好你如何去编写好我们的java代码,必须要按照这个规范来。

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

image-20201007142931846
5.2下载安装Maven

下载完成后,解压即可:

5.3配置环境变量

在我们的系统环境变量中

配置如下配置:

  • M2_HOME maven目录下的bin目录
  • MAVEN_HOME maven的目录
  • 在系统的path中配置
image-20201007145742761

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

5.4阿里云镜像
  • 镜像:mirrors

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

<mirror>
      <id>alimaven</id>
      <name>aliyun maven</name>
      <url>http://maven.aliyun.com/nexus/content/groups/public/</url>
      <mirrorOf>central</mirrorOf>       
</mirror> 
5.5本地仓库

建立一个仓库,远程仓库

建立一个本地仓库:

<localRepository>D:\environment\apache-maven-3.5.0\maven_repository</localRepository>
5.6在IDEA中使用Maven

1.启动IDEA

2.创建一个MavenWeb项目

image-20201007151600466 image-20201007151731207 image-20201007151925305 image-20201007152054809

3.等待项目初始化完毕

image-20201007152530148 image-20201007153554689

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

5.IDEA中的Maven设置

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

image-20201007153711540 image-20201007153909531

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

5.7创建一个普通的Maven项目
image-20201007154516872 image-20201007154640791 image-20201007154746370
5.8标记文件夹功能

image-20201007155124556image-20201007155326325

image-20201007155508922 image-20201007155605651
5.9在IDEA中配置tomcat
image-20201007160035178 image-20201007160407436

解决警告问题

为什么会有这个问题:我们访问一个网站,需要制定一个文件夹的名字;

image-20201007160519102 image-20201007160750386 image-20201007160932266 image-20201007161731491
5.10pom文件

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

image-20201007162112816
<?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>

<!--  这是我们刚才配置的GAV-->
  <groupId>org.long</groupId>
  <artifactId>javaweb01-Maven</artifactId>
  <version>1.0-SNAPSHOT</version>
  
<!--   Package:项目的打包方式
      jar:java应用
      war;javaweb应用  
-->
  <packaging>war</packaging>

  <name>javaweb01-Maven Maven Webapp</name>
  <!-- FIXME change it to the project's website -->
  <url>http://www.example.com</url>

<!--  配置-->
  <properties>
<!--    项目的默认编码-->
    <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
<!--    编码版本-->
    <maven.compiler.source>1.7</maven.compiler.source>
    <maven.compiler.target>1.7</maven.compiler.target>
  </properties>

<!--  项目依赖-->
  <dependencies>
<!--    具体依赖的jar包的配置文件-->
    <dependency>
      <groupId>junit</groupId>
      <artifactId>junit</artifactId>
      <version>4.11</version>
      <scope>test</scope>
    </dependency>
  </dependencies>

<!--  项目构建用的东西-->
  <build>
    <finalName>javaweb01-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>

Maven的高级之处在于它不仅可以导入jar包,他还可以帮你导入这个jar包所依赖的其他jar包
image-20201007163730156
Maven由于他的约定大于配置,我们之后可能遇到我们写的文件无法被导出,或者生效的问题
解决方案:
<!--    在build中配置resources,来防止我们资源导出失败的问题-->
    <build>
        <resources>
            <resource>
                <directory>src/main/resources</directory>
                <excludes>
                    <exclude>**/*.properties</exclude>
                    <exclude>**/*.xml</exclude>
                </excludes>
                <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.12IDEA操作
image-20201007164306986
5.13解决遇到的问题
1.Maven 3.6.2

解决方法:降级为3.6.1

image-20201008160259153

2.Tomcat闪退
3.IDEA中每次都要重复配置Maven
  • 在IDEA中的全局默认配置中去配置
  • image-20201008161206082
image-20201008161447247

4.Maven项目中Tomcat无法配置

5.Maven默认项目中的web.xml版本问题
image-20201008162228694
6.替换为webapp4.0版本和tomcat一致
7.Maven仓库的使用

地址:https://mvnrepository.com/

image-20201008163339976

image-20201008163518040

image-20201008163645267

image-20201008163724903

6Servlet


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

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

6.2HelloServlet

Servlet在sun公司有两个默认的实现类:HttpServlet,

image-20220906165547014

1,构建一个普通的Maven项目,删掉里边的sec目录,以后我们的学习就在这个项目里边建立module;
  • 这个空的工程就是Maven的主工程;
2,关于Maven父子工程的理解;
  • 在父项目中会有
 <modules>
        <module>Servlet-01</module>
 </modules>

子项目中有时候会有parents;

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

son extends father
3.Maven环境优化

​ 1.修改web.xml为最新的

​ 2.将Maven的结构搭建完整

4.编写一个Servlet程序

​ 1.编写一个普通类

​ 2.实现Servlet接口,这里我们直接继承HttpServlet

public class HelloServlet extends HttpServlet {
    //由于get或者post知识请求实现的不同的方式,可以互相调用,业务逻辑都一样
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
      PrintWriter writer=resp.getWriter();//响应流
        writer.print("hello,servlet");
    }

    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        doGet(req, resp);
    }
}
5.编写Servlet映射

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

 <servlet>
    <servlet-name>hello</servlet-name>
    <servlet-class>com.HelloServlet</servlet-class>
  </servlet>
  <servlet-mapping>
    <servlet-name>hello</servlet-name>
    <url-pattern>/hello</url-pattern>
  </servlet-mapping>
6.配置Tomcat

注意:配置项目发布的路径就可以了

7.启动测试
6.3Servlet原理

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

image-20201008202033590

6.4Mapping
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>/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.默认请求路径
 <servlet-mapping>
    <servlet-name>hello</servlet-name>
    <url-pattern>/*</url-pattern>
  </servlet-mapping>
5.指定一些后缀或者前缀等
 <!--可以自定义后缀实现请求映射
	注意点,*前边不能加项目映射的路径
	hello/sajdkajda.guoxiulong,只要结尾匹配即可-->
 <servlet-mapping>
    <servlet-name>hello</servlet-name>
    <url-pattern>*.guoxiulong</url-pattern>
  </servlet-mapping>
6.优先级问题

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

  <servlet>
    <servlet-name>error</servlet-name>
    <servlet-class>com.ErrorServlet</servlet-class>
  </servlet>
  <servlet-mapping>
    <servlet-name>error</servlet-name>
    <url-pattern>/*</url-pattern>
  </servlet-mapping>
6.5ServletContext

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

1.共享数据

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

    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//        this.getInitParameter()   初始化参数
//          this.getServletConfig()   Servlet配置
//        this.getServletContext()    servlet上下文
        ServletContext context= this.getServletContext();

        String username="龙";//数据
        context.setAttribute("username",username);//将一个数据保存在了servletcontext中,名字为:username

        System.out.println("hello");
    }
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//        this.getInitParameter()   初始化参数
//          this.getServletConfig()   Servlet配置
//        this.getServletContext()    servlet上下文
        ServletContext context= this.getServletContext();

        String username="龙";//数据
        context.setAttribute("username",username);//将一个数据保存在了servletcontext中,名字为:username

        System.out.println("hello");
    }

测试访问结界:应当先存后取

2.获取初始化参数
<!--配置一些web应用的初始化参数-->
<context-param>
  <param-name>url</param-name>
  <param-value>jbdc:mysql://localhost:3306/mybatis</param-value>
</context-param>
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        ServletContext context = this.getServletContext();

        String url = context.getInitParameter("url");
        resp.getWriter().print(url);
    }
3.请求转发
   protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        ServletContext context = this.getServletContext();

        System.out.println("进入了demo04");
        //RequestDispatcher requestDispatcher = context.getRequestDispatcher("/demo3");//转发的请求路径
        //requestDispatcher.forward(req,resp);//调用forward实现请求转发;
        context.getRequestDispatcher("/demo3").forward(req,resp);
    }

image-20201008214350572*

  • 上图中,第一个图是请求转发,第二个图是重定向
4.读取资源文件

Piroperties

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

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

思路:需要一个文件流;

username=root
password=123456
package com;

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.InputStream;
import java.util.Properties;
import java.util.PropertyPermission;

/**
 * user: Xiu long
 * dote: 2020/10/8
 * time: 21:58
 */
public class ServletDemo5 extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        InputStream is = this.getServletContext().getResourceAsStream("/WEB-INF/classes/db.properties");

        Properties prop = new Properties();
        prop.load(is);
        String user= prop.getProperty("username");
        String pwd=prop.getProperty("password");

        resp.getWriter().print(user+":"+pwd);

    }

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

访问测试即可ok;

6.6HttpServletReponse

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.常见应用

1.向浏览器输出消息(一直在讲,就不说了)

2.下载文件

​ 1.要获取下载文件的路径

​ 2.下载的文件名是啥

​ 3.设置想办法让浏览器能够支持下载我们需要的东西

· 4.获取下载文件的输入流

​ 5.创建缓冲区

​ 6.获取Outputstream对象

​ 7,将FileOutputStream流写入到缓冲区

​ 8.使用OutputStream将缓冲区中的数据流输出到客户端!

    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//  1.要获取下载文件的路径
        String realPath = "D:\\IDEA\\workplace\\javaStudy\\Servlet\\javaweb-02-Servlet\\responses\\target\\responses\\WEB-INF\\classes\\1.png";
        System.out.println("下载文件的路径:"+realPath);
//​	2.下载的文件名是啥
        String filename = realPath.substring(realPath.lastIndexOf("\\") + 1);
//​	3.设置想办法让浏览器能够支持下载我们需要的东西,中文文件名用URLEncoder.encode去编码,否则可能乱码
        resp.setHeader("Content-disposition","attachment;filename="+ URLEncoder.encode(filename,"utf-8"));
//·	4.获取下载文件的输入流
        FileInputStream in = new FileInputStream(realPath);
//​	5.创建缓冲区
        int len=0;
        byte[] buffer=new byte[1024];
//​	6.获取Outputstream对象
        ServletOutputStream out = resp.getOutputStream();
//​	7,将FileOutputStream流写入到缓冲区
        while ((len=in.read(buffer))!=-1)
        {
            out.write(buffer,0,len);
        }

        in.close();
        out.close();
//​	8.使用OutputStream将缓冲区中的数据流输出到客户端!
    }

    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        doGet(req, resp);
    }
3.验证码功能

验证码怎么来的?

  • 前端实现
  • 后端实现,需要用到java的图片类,生成一个图片
4.实现重定向(重点)

image-20201009194323319

一个web资源受到客户端请求后,他会通知客户端区访问另一个web资源,这个过程叫做重定向;

常见场景:

  • 用户登录
void sendRedirect(String var1) throws IOException;

测试:

//        resp.setHeader("Location","/r/image");
//        resp.setStatus(302);
        
       resp.sendRedirect("/r/image");//重定向

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

相同点:

  • 页面都会实现跳转

不同点:

  • 请求转发的时候,URL不会发生跳转
  • 重定向的时候,URL地址栏会发生变化
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
    //处理请求
    String username = req.getParameter("username");
    String password = req.getParameter("password");
    System.out.println(username+":"+password);

    //重定向一定要注意,路径问题,否则404
    resp.sendRedirect("/r/success.jsp");
}

@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
    doGet(req, resp);
}
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>Title</title>
</head>
<body>
<h1>success</h1>
</body>
</html>
6.7HttpServletRequest

HttpServletRequest代表客户端的请求,用户通过Http访问服务器,HTTP请求中的所有信息会被封装到HttpServletRequest。通过这个HttpServletRequest的方法,获得客户端的所有信息。

image-20201009203239650

1.获取前端传递的参数,请求转发

image-20201009203547598

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

        req.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+":"+password);
        System.out.println(Arrays.toString(hobbys));
        System.out.println("*******************************");


        System.out.println(req.getContextPath());
//        resp.sendRedirect("/r/success.jsp");
        //通过请求转发
        //这里的‘/’代表当前的web路径
        req.getRequestDispatcher("/success.jsp").forward(req,resp);
    }
面试题:请你聊聊重定向和转发的区别?

相同点:

  • 页面都会实现跳转

不同点:

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

7Cookie,Session


7.1会话

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

有状态会话:一个同学来过教室,下次再来教室,我们就知道这个同学曾经来过,称之为有状态会话;

你怎么证明你是西开的学生?

​ 你 西开

​ 1.发票 西开给你发票

​ 2.学校登记 西开给你标记了你来过

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

客户端 服务端

1.服务端给客户端一个信件,客户端下次访问服务器带上信件就可以了;cookie

2.服务器标记你来过了,下次你来的时候我匹配你;session

7.2保存会话的两种技术

cookie

  • 一种客户端技术(响应,请求)

session

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

常见常见:网站登录之后,你下次就不用在登陆了,第二次就直接登陆了

7.3cookie

image-20201010174405338

1.从请求中拿到cookie信息

2.服务器形影给客户端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:一般会保存在本地的用户目录下appdata

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

  • 一个cookie只能保存一个信息

  • 一个web站点可以给浏览器发送多个cookie,最多存放20个cookie

  • cookie大小有限制4KB

  • 300个cookie浏览器上限

删除cookie:

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

编码解码:

URLDecoder.decode(cookie.getValue(),"utf-8")//解码
URLEncoder.encode("罗诗涵","UTF-8")//编码
7.4Session(重点)

image-20201010180937771

什么是session:

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

image-20201010165156218

Session和Cookie的区别:

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

使用场景:

  • 保存一个登陆用户的信息;
  • 购物车信息;
  • 在整个网站中经常会使用的数据,我们经常保存在Session中;

使用session:

package com.servlet;

import com.poji.person;

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

/**
 * user: Xiu long
 * dote: 2020/10/10
 * time: 16:46
 */
public class SessionDemo01 extends HttpServlet {
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        //解决乱码问题
        resp.setCharacterEncoding("utf-8");
        req.setCharacterEncoding("utf-8");
        resp.setContentType("text/html;charset=utf-8");


        //获得session
        HttpSession session = req.getSession();

        //给session存东西
        session.setAttribute("name",new person("罗诗涵",1));

        //获取Session的ID
        String id = session.getId();

        //判断Session是不是新创建的
        if (session.isNew())
        {
            resp.getWriter().write("session创建成功,session的id为:"+id);
        }else
        {
            resp.getWriter().write("session已经创建过了"+id);
        }

//        //session创建的时候做了什么
//        Cookie cookie = new Cookie("JSESSIONID", id);
//        resp.addCookie(cookie);
    }

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



   //获得session
        HttpSession session = req.getSession();

        person person = (person) session.getAttribute("name");

        System.out.println(person.toString());

  HttpSession session = req.getSession();

        session.removeAttribute("name");
        //手动注销session
        session.invalidate();

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

<!--    设置默认的session生效时间-->
    <session-config>
<!--       15分钟后,session自动失效,以分钟为单位-->
        <session-timeout>1</session-timeout>
    </session-config>

image-20201010181312553

8JSP

8.1什么事JSP

java Server Pages:java服务页端面,也和servlet一样,用于开发动态web技术!

最大的特点:

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

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

  • 代码层面没有任何问题

  • 服务器内部工作

    tomcat中有一个work目录;

    IDEA中使用tomcat的会在IDEA的tomcat中生产一个work目录

image-20201010182319506

​ 发现页面变成了java程序

我的电脑的位置:

C:\Users\long\AppData\Local\JetBrains\IntelliJIdea2020.1\tomcat\Unnamed_javaweb-session-cookie\work\Catalina\localhost\r\org\apache\jsp

image-20201010182809581

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

JSP最终会被转换成一个java类!

JSP本质上就是一个servlet!

//初始化 
public void _jspInit() {
  }
//销毁
  public void _jspDestroy() {
  }
//JSPservice
  public void _jspService(.HttpServletRequest request,.HttpServletResponse response)

1.判断请求

2.内置了一些对象

final javax.servlet.jsp.PageContext pageContext;	//页面上下文
javax.servlet.http.HttpSession session = null;		//session
final javax.servlet.ServletContext application;		//applicationcontext
final javax.servlet.ServletConfig config;			//config
javax.servlet.jsp.JspWriter out = null;				//out
final java.lang.Object page = this;					//page:当前页
HttpServletRequest request							//请求
HttpServletResponse response						//访问

3.输出页面前增加的代码

  response.setContentType("text/html");				//设置相应的页面类型
      pageContext = _jspxFactory.getPageContext(this, request, response,
      			null, true, 8192, true);
      _jspx_page_context = pageContext;
      application = pageContext.getServletContext();
      config = pageContext.getServletConfig();
      session = pageContext.getSession();
      out = pageContext.getOut();
      _jspx_out = out;

4.以上的这些个对象我们可以在jsp页面中直接使用!

image-20201010190609421

在JSP页面中:

只要是java代码就会不动的输出;

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

    out.write("<html>\r\n");

这样的格式;

8.3JSP的基础语法、

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

JSP表达式
<%--  jsp表达式
  作用:用来将程序的输出,输出到客户端
    <%= 变量或表达式%>
--%>
<%=new java.util.Date()%>
jsp脚本片段

<%--  jsp脚本片段--%>
<%
    int sum=0;
    for (int i=0;i<10;i++)
    {
      sum+=i;
    }
    out.print("<h1>sum="+sum+"</h1>");
%>

    <%
        int x=10;
        out.print(x);
    %>
      <p>这是一个jsp文档</p>
    <%
        int y=10;
        out.print(y);
    %>

<%--  在代码中嵌入HTML元素--%>
    <%
      for (int i = 0; i < 5; i++) {
    %>
    <h1>htllo long  <%=i%> </h1>
    <%
      }
    %>
JSP声明
<%!
   static {
     System.out.println("Loading Service");
   }

   private int globalVar=0;

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

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

在jsp中嵌入java代码即可!

<%%>
<%=%>
<%!%>

<%-- --%>

JSP的注释,不会再客户端显示,HTML就会!

8.4JSP指令
<%@ page args.... %>

<%--两种指令的区别--%>

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

<hr>
<%--jsp:include相当于拼接页面--%>
<%--jsp标签--%>
<jsp:include page="common/header.jsp"/>
    <h1>网页主体</h1>
<jsp:include page="common/footer.jsp"/>
8.5.9大内置对象
  • PageContext 存东西

  • Request 存东西

  • Response

  • Session 存东西

  • Application [ServletContext] 用来存东西

  • config [ServletConfig]

  • page,不用了解

  • out

  • exception

 pageContext.setAttribute("name1","郭修龙1");//保存的数据只在一个页面中有效
    request.setAttribute("name2","郭修龙2");//保存的数据只在依次请求中有效,请求转发会携带这个数据
    session.setAttribute("name3","郭修龙3");//保存的数据在依次会话中有效,从打开浏览器到关闭浏览器
    application.setAttribute("name4","郭修龙4");//保存的数据只在服务器中有效,从打开服务器到关闭服务器

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

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

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

8.6JSP标签,JSTL标签,EL表达式
<!--        JSTL表达式依赖-->
        <!-- https://mvnrepository.com/artifact/javax.servlet.jsp.jstl/jstl-api -->
        <dependency>
            <groupId>javax.servlet.jsp.jstl</groupId>
            <artifactId>jstl-api</artifactId>
            <version>1.2</version>
        </dependency>
<!--        standard标签库-->
        <!-- https://mvnrepository.com/artifact/taglibs/standard -->
        <dependency>
            <groupId>taglibs</groupId>
            <artifactId>standard</artifactId>
            <version>1.1.2</version>
        </dependency>

EL表达式: ${}

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

JSP标签

<%--jsp:include--%>
<%--
http://localhost:8080/jsptag.jsp?name=kuangshen&age=12
--%>
<h1>1</h1>
<jsp:forward page="/jsptag2.jsp">
    <jsp:param name="name1" value="name1"/>
    <jsp:param name="name2" value="name2"/>
</jsp:forward>

JSTL表达式

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

核心标签(掌握部分即可)

格式化标签

SQL标签

XML标签

image-20201011101719797

JSTL标签的使用步骤

  • 引入对应的taglib

  • 使用其中的方法

  • 在Tomcat也需要引入jstl的包,否则就会报错:jstl解析错误

c: if

<body>

<h1>if测试</h1>
<form action="coreif.jsp" method="get">
<%--    EL表达式获取表中的数据
        ${param.参数名}
--%>
    <input type="text" name="username" value="${param.username}">
    <input type="submit" value="登录">
</form>
<%--判断如果登录的用户是管理员就登陆成功--%>
<c:if test="${param.username=='admin'}" var="isAdmin">
    <c:out value="管理员欢迎你"></c:out>
</c:if>
<c:out value="${isAdmin}"></c:out>

<%--<%--%>
<%--    if(request.getParameter("username").equals("admin"))--%>
<%--    {--%>
<%--        out.print("登陆成功");--%>
<%--    }--%>
<%--%>--%>
</body>

c: which

<body>
<%--定义一个变量score,值为85--%>
<c:set var="成绩" value="90"></c:set>

<c:choose>
    <c:when test="${成绩>=90}">
        <c:out value="你的成绩为优秀"></c:out>
    </c:when>
    <c:when test="${成绩<90}">
        <c:out value="难道你是罗诗涵吗?"></c:out>
    </c:when>
</c:choose>
</body>

c: foreach

<body>

<%
    ArrayList<String> people = new ArrayList<>();
    people.add(0,"张三");
    people.add(1,"张三");
    people.add(2,"李四");
    people.add(3,"王五");
    people.add(4,"罗诗涵");
    people.add(5,"郭大爷");
    request.setAttribute("list",people);
%>

<%--
var   每一次遍历出来的变量
items   要遍历的对象
begin   开始
end     结束
step    步长
--%>
<c:forEach var="people" items="${list}">
    <c:out value="${people}"></c:out><br>
</c:forEach>

<c:forEach var="people" items="${list}" begin="1" end="3" step="2">
    <c:out value="${people}"></c:out><br>
</c:forEach>

</body>

9.javaBean

实体类

Javabean有特定的写法:

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

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

ORM:对象关系映射

  • 表–>类
  • 字段–>属性
  • 行记录–>对象
idnameageadderss
1郭修龙13青岛
2郭修龙24青岛
3郭修龙35青岛
class Paople{
    private int id;
    private String name;
    private int age;
    private String address;
}
class A{
    new People(1,"郭修龙1",3,"青岛");
    new People(1,"郭修龙2",4,"青岛");
    new People(1,"郭修龙3",5,"青岛");
`	
}

10三层架构


什么是MVC:Model view Controller 模型,视图,控制器

10.1早些年

image-20201011145824785

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

servlet---CRUD---数据库
数据:程序十分臃肿,数据库不易维护     servlet的代码中:处理请求,响应,视图跳转,处理JDBC,处理业务代码,处理逻辑代码
    
    架构:没有什么是加一层是解决不了的!
    程序员
    |
    JDBC
    |
	MySQLOracleSQLserver....
10.2MVC三层架构

image-20201011150822440

Model:

  • 业务处理:业务逻辑(service)
  • 数据持久层:CRUD (Dao)

View:

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

Controller:(servlet)

  • 接收用户的请求:(req:请求参数,session信息)

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

  • 控制视图的跳转

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

11过滤器(Filter)(重点)


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

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

image-20201011152148973

Filter开发步骤:

1.导包

2.编写过滤器

​ 1.导包不要错

image-20201011153111778

实现filter接口,重写对应的方法即可:

//web服务器启东时他就初始化了,随时等待对象出现
//初始化
@Override
public void init(FilterConfig filterConfig) throws ServletException {
    System.out.println("初始化");
}

/**
 * @param chain
 *
 1.过滤中的所有代码,在过滤特定请求时都会执行
 2.必须要让过滤器据需同行
        chain.doFilter(request, response);

 */

@Override
public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
    request.setCharacterEncoding("utf-8");
    response.setCharacterEncoding("utf-8");
    response.setContentType("text/heml;charset=utf-8");


    System.out.println("执行前....");
    chain.doFilter(request, response);//让我们的请求据需走,如果不写,程序到这里拦截停止
    System.out.println("执行后....");
}
//销毁:web服务器关闭的时候,过滤会销毁
@Override
public void destroy() {
    System.out.println("销毁");
}

3.在web.xml中配置Filter过滤器

    <filter>
        <filter-name>CharacterEncodingFilter</filter-name>
        <filter-class>com.filter.CharacterEncodingFilter</filter-class>
    </filter>
    <filter-mapping>
        <filter-name>CharacterEncodingFilter</filter-name>
<!--        只要是/servlet的任何请求都会经过过滤器-->
        <url-pattern>/servlet/*</url-pattern>
    </filter-mapping>

12监听器


实现一个监听器的接口:(有N种)

1.编写一个监听器

​ 实现监听器的接口

//创建session监听:看你的一举一动
    //一旦创建一个session,就会触发一次这个事件
    @Override
    public void sessionCreated(HttpSessionEvent se) {
        ServletContext ctx = se.getSession().getServletContext();

        System.out.println("当前"+se.getSession().getId());

        Integer onlineCount = (Integer)ctx.getAttribute("OnlineCount");

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

            onlineCount=new Integer(count+1);
        }

        ctx.setAttribute("OnlineCount",onlineCount);
    }
//销毁session监听
//一旦销毁一个session,就会触发一次这个事件
    @Override
    public void sessionDestroyed(HttpSessionEvent se) {
        ServletContext ctx = se.getSession().getServletContext();
        Integer onlineCount = (Integer)ctx.getAttribute("OnlineCount");

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

            onlineCount=new Integer(count-1);
        }

        ctx.setAttribute("OnlineCount",onlineCount);
    }

/*
session销毁:
1.手动销毁   getsession().invalidate()
2.自动销毁
*/

2.web.xml中注册监听器


<!--    注册监听器-->
    <listener>
        <listener-class>com.listener.OnlineCounterLisrener</listener-class>
    </listener>

3.看情况是否使用

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

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

public class TestPanel {
    public static void main(String[] args) {
        Frame frame = new Frame("中秋节快乐");

        Panel panel = new Panel(null);
        frame.setLayout(null);

        frame.setBounds(200,200,300,300);
        frame.setBackground(new Color(0,0,255));

        panel.setBounds(50,50,300,300);
        panel.setBackground(new Color(0,255,0));

        frame.add(panel);

        frame.setVisible(true);


        //监听事件
        frame.addWindowListener(new WindowAdapter() {
            @Override
            public void windowClosing(WindowEvent e) {
                super.windowClosing(e);
            }
        });

    }
}

用户登录后才能进入首页!用户注销后不能进入主页了!

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

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

@Override
public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {



    HttpServletRequest req = (HttpServletRequest) request;
    HttpServletResponse resp = (HttpServletResponse) response;
    if((req.getSession().getAttribute("USER_SESSION"))==null)
    {
        System.out.println(12345);
        resp.sendRedirect("/error.jsp");
    }

    //System.out.println("guolv");
    chain.doFilter(request,response);
}

14JDBC复习


什么是JDBC:java连接数据库。

image-20201011192125673

需要jar包的支持:

  • java.sql
  • javax.sql
  • mysql.conneter-java…连接驱动必须要的

实验环境搭建

  • 创建数据库

  • 导入数据库依赖

<!--        mysql驱动-->
        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
            <version>8.0.20</version>
        </dependency>
  • IDEA中连接数据库

JDBC固定步骤:

1.加载驱动

2.连接数据库,代表数据库

3.向数据库发送SQL的对象statement:CRUD

4.编写SQL(根据业务,不同的SQL)

5.执行查询SQL,返回一个resultset : 结果集

6.关闭连接,释放资源(一定要做)先开后关

public class jdbcTest {
    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";
        String username= "root";
        String password= "1111";

        //1.加载驱动
        Class.forName("com.mysql.jdbc.Driver");
        //2.获取数据库连接,代表数据库
        Connection connection = DriverManager.getConnection(url, username, password);

        //3.向数据库发送SQL的对象statement
        Statement statement = connection.createStatement();

        //4.编写SQL
        String sql="select * from people";

        //5.执行查询SQL,返回一个resultset : 结果集
        ResultSet rs = statement.executeQuery(sql);

        while (rs.next())
        {
            System.out.println("id"+rs.getObject("id"));
        }

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

预编译SQL

    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";
        String username= "root";
        String password= "1111";

        //1.加载驱动
        Class.forName("com.mysql.jdbc.Driver");
        //2.获取数据库连接,代表数据库
        Connection connection = DriverManager.getConnection(url, username, password);

        //3.编写SQL
        String sql="insert into people(id) values(?)";

        //4.预编译
        PreparedStatement preparedStatement = connection.prepareStatement(sql);

        preparedStatement.setInt(1,12);

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

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

        preparedStatement.close();
        connection.close();
    }

事务:

要么都成功,要么都失败!

ACID原则:保证数据安全。

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

转账:
A:
B:1000
C:1000

A(900)---100-->B(1100)

Junit单元测试

 <dependency>
            <groupId>junit</groupId>
            <artifactId>junit</artifactId>
            <version>4.12</version>
        </dependency>

简单使用

@Test注解只有在方法上有用,只要在方法上加了这个注解,方法就可以运行

import org.junit.Test;

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

正确示范:

image-20201012165327353

错误示范:

image-20201012165429619

搭建一个环境

数据库事务回滚:(提示:一定要开启事务)

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

        Connection connection=null;

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

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

        //3.通知数据库开启事务,false是开启
        connection.setAutoCommit(false);

        String sql1="update people set age=age+100 where id=1";
        connection.prepareStatement(sql1).executeUpdate();

        //4.制造错误
//        int i=1/0;

        String sql2="update people set age=age+100 where id=1";
        connection.prepareStatement(sql1).executeUpdate();

        connection.commit();//以上两条SQL都执行成功了就提交事务
        System.out.println("提交成功");
        } catch (Exception e) {

            try {

                //如果出现异常,就通知数据库处理事务
                connection.rollback();
            } catch (SQLException throwables) {
                throwables.printStackTrace();
            }

            e.printStackTrace();
        }finally {
            try {
                connection.close();
            } catch (SQLException throwables) {
                throwables.printStackTrace();
            }
        }
    }
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值