JavaWeb

JavaWeb

Java Web

1.基本概念

1.1前言

Web开发:

  • web,网页的意思,www.baidu.com
  • 静态web
    • html、css
    • 提供给所有人看的数据始终不会发生变化!
  • 动态web
    • 淘宝,几乎是所有的网站
    • 提供给所有人看的数据会发生变化,每个人在不同的事件、不同的地点看到的信息各部相同
    • 技术栈:Servlet/JSP、ASP、PHP

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

1.2web应用程序

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

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

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

1.3静态web

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

1.4动态web

页面会动态展示:web的页面展示的效果因人而异

缺点:

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

优点:

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

2.web服务器

2.1技术讲解

ASP:

  • 微软:国内最早流行的就是ASP
  • 在HTML中嵌入了VB的脚本,ASP+DOM
  • 在ASP开发中,基本一个页面都有大量的业务代码,页面极其混乱
  • 维护成本高
  • C#
  • IIS

PHP:

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

JSP/Servlet:

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

    • B/S:浏览器和服务器
    • C/S:客户端和服务器
  • 基于Java语言的(所有的大公司,或者一些开源的组件,都是用java写的)

  • 可以承载三高的影响(高并发、高可用、高性能)

  • 语法像ASP,ASP–>JSP,加强市场强度

2.2web服务器

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

IIS:微软的;ASP…Windows自带的

Tomcat:

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

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

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

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

下载tomcat:

  1. 安装or解压
  2. 了解配置文件及目录结构
  3. 这个东西的作用

3.Tomcat

3.1安装Tomcat

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

3.2启动Tomcat

startup.bat

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

可能遇到的问题:

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

3.3配置

服务器核心配置文件server.xml

可以配置启动的端口号

  • tomcat的默认多口号:8080
  • mysql:3306
  • http:80
  • https:443

可以配置主机的名称

  • 默认的主机名:localhost->127.0.0.1
  • 默认网站应用存放的位置:webapps

高难度面试题:

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

  1. 输入一个域名,回车
  2. 坚持本机的C:\Windows\System32\drivers\etc\hosts配置文件下有没有这个域名映射
    1. 有:直接返回对应的ip地址,这个地址中,有我们需要访问的web程序,可以直接访问
    2. 没有:去DNS服务器找,找到的话就返回,找不到就返回找不到

可以配置以下环境变量(可选)

3.4发布一个web网站

不会就先模仿!!!

  • 将主机写的网站,放到服务器(Tomcat)中指定的web应用的文件夹(webapps)下,就可以访问了
  • 网站应该有的结构
--webapps:Tomcat服务器的web目录
    --ROOT
    --simpleteen:网站的目录名
        --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:安全的

  • 443

4.2两个时代

  • http1.0
    • HTTP/1.0:客户端可以与web服务器连接后,只能获得一个web资源,断开连接
  • http2.0
    • HTTP/1.1:客户端可以与web服务器连接后,可以获得多个web资源,断开连接

4.3Http请求

  • 客户端–发请求(Request)–服务器

1、请求行

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

2、消息头

Accept:告诉浏览器,它锁支持的数据类型
Accept-Encoding:支持哪种编码格式 GBK UTF-8 GB2312 ISO8859-1
Accept-Language:告诉浏览器,它的语言环境
cache-Control:缓存控制
Connection:告诉浏览器,请求完成时断开还是保存连接

4.4Http响应

  • 服务器–响应(response)–客户端

1.响应体

2.响应状态码

200:请求响应成功

3xx:请求重定向 304

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

4xx:找不到资源 404

  • 资源不存在

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

常见面试题:

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

5.Maven

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

  1. 在Javaweb开发中,需要使用大量的jar包,我们需要手动去导入
  2. 如何能够让一个东西自动帮我们导入核配置这个jar包

由此,Maven诞生了!

5.1Maven项目架构管理工具

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

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

  • 有约束,不要去违反

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

5.2下载安装Maven

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

下在完成后,解压即可!

5.3配置环境变量

在我们的系统环境变量中

配置如下配置

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

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

cmd控制台mvn -version

5.4阿里云镜像

  • 镜像:mirrors
    • 作用:加速我们的下载
  • 国内建议使用阿里云的镜像
<mirror>
<id>nexus-aliyun</id>
<mirrorOf>central</mirrorOf>
<name>Nexus aliyun</name>
<url>http://maven.aliyun.com/nexus/content/groups/public</url>
</mirror>

5.5本地仓库

本地仓库,远程仓库

建立一个本地仓库:localRepository

<localRepository>D:\maven_repository</localRepository>

5.6在IDEA中使用Maven

  1. 启动IDEA

  2. 创建一个MavenWeb项目

  3. 等待项目初始化完毕

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

  5. IDEA中Maven设置

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

    在IDEA中配置Maven
    
    注意:经常在IDEA中会出现先一个问题,就是项目自动创建完成后骂他这个MavenHome会使用IDEA默认,我们如果发现了这个问题,手动改为本地的
    
  6. 到这里Maven在IDEA中的配置和使用就OK了!

解决maven初始化警告

No archetype found in remote catalog. Defaulting to internal catalog
添加参数-DarchetypeCatalog=local
IDEA设置方法:
File=> Other Settings=> Settings for New Projects=> Build,Execution,Deployment=> Build Tools=> Maven=> Runner=>VM Options=> 填写为“-DarchetypeCatalog=local”

5.7创建一个普通的Maven项目

webapp这个只有在web应用下才会有

5.8标记文件夹功能

5.9在IDEA中配置Tomcat

5.10pom文件

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

package项目的打包方式:

  • jar:Java应用
  • war:JavaWeb应用

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

    <!--在build中配置resources,来防止我们资源导出失败的问题-->
    <build>
        <resources>
            <resource>
                <directory>src/main/resources</directory>
                <includes>
                    <include>**/*.properties</include>
                    <include>**/*.xml</include>
                </includes>
                <filtering>true</filtering>
            </resource>
            <resource>
                <directory>src/main/java</directory>
                <includes>
                    <include>**/*.properties</include>
                    <include>**/*.xml</include>
                </includes>
                <filtering>true</filtering>
            </resource>
        </resources>
    </build>

5.11IDEA操作

生成目录树

maven中jar的关联组

5.12解决遇到的问题

1.Maven3.6.2

Unable to import maven project:See logs for details

解决方法:降级为3.6.1

2.Tomcat闪退

解决方法:配置jdk、jre环境变量

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

解决方法:在IDEA中的全局默认设置中去配置

4.Maven项目中Tomcat无法配置

解决方法:添加Application context,防止多个项目重复

5.Maven默认web项目中的web.xml版本问题

解决方法:Tomcat默认项目ROOT拷贝,替换为webapp4.0版本呢和tomcat版本一致

6.Maven仓库的使用

解决方法:https://mvnrepository.com/

6.Servlet

6.1Servlet简介

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

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

6.2HelloServlet

Servlet接口Sun公司有两个默认的实现类HttpServlet和GenericServlet

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

  2. 关于Maven符子工程的理解

    父项目中会有

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

    子项目会有

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

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

  3. Maven环境优化

    1. 修改web.xml为最新的
    2. 将maven的结构搭建完整
  4. 编写一个Servlet程序

    1. 编写一个普通类

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

      package com.simpleteen.servlet;
      
      import javax.servlet.ServletException;
      import javax.servlet.http.HttpServlet;
      import javax.servlet.http.HttpServletRequest;
      import javax.servlet.http.HttpServletResponse;
      import java.io.IOException;
      import java.io.PrintWriter;
      
      public class HelloServlet extends HttpServlet {
          //由于get或者post只是请求实现的不同方式,可以相互调用,业务逻辑都一样
          @Override
          protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
              PrintWriter writer = resp.getWriter();//相应流
              writer.println("Hello,Servlet");
          }
      
          @Override
          protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
              doGet(req, resp);
          }
      }
      
  5. 编写Servlet的映射

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

      <!--web.xml中式配置web的核心应用-->
      <servlet>
        <servlet-name>HelloServlet</servlet-name>
        <servlet-class>com.simpleteen.servlet.HelloServlet</servlet-class>
      </servlet>
      <servlet-mapping>
        <servlet-name>HelloServlet</servlet-name>
        <url-pattern>/hello</url-pattern>
      </servlet-mapping>
    
  6. 配置Tomcat

  7. 启动测试

6.3Servlet原理

Servlet是由Web服务器调用,Web服务器在收到浏览器请求之后,会产生请求(Request)和响应(Response)两个对象。第一次会去请求Servlet,Servlet调用service方法,我们编写的实现类重写这些方法,接收并处理请求,输出相应信息交给Reponse,然后Web服务器读取相应信息返回给客户端

6.4Mapping问题

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

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

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

  4. 默认请求路径/*

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

    (*.do) 注意: *前面不能加项目映射的路径

  6. 优先级问题

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

6.5ServletContext

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

1.共享数据

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

set共享数据

    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        System.out.println("Hello");

        //this.getInitParameter() 初始化参数
        //this.getServletConfig() Servlet配置
        //this.getServletContext() Servlet上下文

        ServletContext servletContext = this.getServletContext();
        String username = "青简工作室";//数据
        servletContext.setAttribute("username",username);//将一个数据保存在ServletContext 键值对 "username" = “青简工作室”
    }

get共享数据

    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        resp.setContentType("text/html");
        resp.setCharacterEncoding("utf-8");

        ServletContext servletContext = this.getServletContext();
        String username = (String) servletContext.getAttribute("username");

        resp.getWriter().print("名字:"+username);
    }

配置web.xml

    <servlet>
        <servlet-name>HelloServlet</servlet-name>
        <servlet-class>com.simpleteen.servlet.HelloServlet</servlet-class>
    </servlet>
    <servlet-mapping>
        <servlet-name>HelloServlet</servlet-name>
        <url-pattern>/hello</url-pattern>
    </servlet-mapping>

    <servlet>
        <servlet-name>GetServlet</servlet-name>
        <servlet-class>com.simpleteen.servlet.GetServlet</servlet-class>
    </servlet>
    <servlet-mapping>
        <servlet-name>GetServlet</servlet-name>
        <url-pattern>/GetServlet</url-pattern>
    </servlet-mapping>

测试访问结果

2.获取初始化参数

配置初始化参数web.xml

    <!--配置一些些web应用的初始化参数-->
    <context-param>
        <param-name>url</param-name>
        <param-value>jdbc:mysql://localhost:3306/school</param-value>
    </context-param>

获取初始化参数

    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        ServletContext servletContext = this.getServletContext();
        String url = servletContext.getInitParameter("url");
        resp.getWriter().print(url);
    }
3.请求转发
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        ServletContext servletContext = this.getServletContext();
        RequestDispatcher requestDispatcher = servletContext.getRequestDispatcher("/Servlet3");//转发的请求路径
        requestDispatcher.forward(req,resp);//调用forward实现请求转发
    }
4.读取资源文件

Properties

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

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

思路:需要一个文件流

    @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 username = prop.getProperty("username");
        String password = prop.getProperty("password");

        resp.getWriter().print(username+":"+password);
    }

db.properties

username=root
password=123456

访问测试即可ok

6.6HttpServletResponse

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 setCharacterEncoding(String var1);
void setContentLength(int var1);
void setContentLengthLong(long var1);
void setContentType(String var1);

响应的状态码

    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.获取OutStream对象

​ 7.将FileOutStream流写入到buffer缓冲区

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

    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        //  1.要获取先下载文件的路径
        String realPath = "D:\\IdeaProjects\\JavaWeb\\javaweb-02-servlet\\response\\src\\main\\resources\\1.jpg";
        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.获取OutStream对象
        ServletOutputStream out = resp.getOutputStream();
        // ​	7.将FileOutStream流写入到buffer缓冲区    //8.使用OutputStream将缓冲区中的数据输出到客户端
        while (in.read(buffer) > 0){
            out.write(buffer,0,len);
        }

        in.close();
        out.close();

    }
3.验证码功能

验证码字母来的?

  • 前端实现
  • 后端实现,需要用到java的图片类,生成一个图片
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
       //如何让浏览器5秒自动刷新一次
        resp.setHeader("refresh","3");
        //在内存中创建一个图片
        BufferedImage bufferedImage = new BufferedImage(80,20,BufferedImage.TYPE_INT_RGB);
        //得到图片
        Graphics2D g = (Graphics2D) bufferedImage.getGraphics();//画笔
        //设置图片的背景颜色
        g.setColor(Color.white);
        g.fillRect(0,0,120,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.write(bufferedImage,"jpg",resp.getOutputStream());
    }

    //生成随机数
    private String makeNum(){
        Random random = new Random();
        String num = random.nextInt(99999999) + "";
        StringBuffer stringBuffer = new StringBuffer();
        for (int i = 0; i < 7-num.length(); i++) {
            stringBuffer.append("0");
        }
        num = stringBuffer.toString()+num;
        return num;
    }
4.实现重定向

一个web资源收到客户端请求后,它会通知客户端去访问另外一个web资源,这个过程叫重定向

常见场景:

  • 用户登录

测试:

    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        /*resp.setHeader("Location","/response_war/ImageServlet");
        resp.setStatus(SC_MOVED_TEMPORARILY);*/
        resp.sendRedirect("/response_war/ImageServlet");//重定向
    }

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

相同点:

  • 页面都会实现跳转

不同点:

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

6.7HttpServletRequest

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

获取前端传递的参数和请求转发

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

        //通过请求转发
        //这里的/代表当前的web应用
        req.getRequestDispatcher("/success.jsp").forward(req,resp);
    }

7.Cookie和Session

7.1会话

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

**有状态会话:**一个人访问过一个网站,下次再访问的时候,网站知道这个人曾经来过,称之为有状态的会话

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

客户端 服务端

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

7.2保存会话的两种技术

cookie

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

session

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

常见的例子:网站登录之后,你下次不用再登录了,下次访问就直接就登录了。

7.3Cookie

  1. 从请求中拿到cookie信息
  2. 服务器响应给客户端cookie
    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        //服务器告诉你,你来的时间,把这个时间封装称为一个新建,你下次来带上新建,我就知道你来了

        //解决中文乱码
        resp.setHeader("content-type","text/html;charset=UTF-8");
        resp.setCharacterEncoding("UTF-8");

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

        //判断Cookie是否存在
        if(cookies!=null){
            //如果存在怎么办
            out.write("您上一次访问的时间是:");
            for (int i = 0; i < cookies.length; i++) {
                Cookie cookie = cookies[i];
                //获取cookie的名字
                if( cookie.getName().equals("LastLoginTime")){
                    //获取cookie中的值
                    long lastLoginTime = Long.parseLong(cookie.getValue());
                    Date date = new Date(lastLoginTime);
                    out.print(date.toLocaleString());
                }
            }
        }else{
            out.write("这是您第一次访问本站");
        }


        //服务器给客户端响应一个cookie
        Cookie cookie = new Cookie("LastLoginTime", System.currentTimeMillis()+"");
        //cookie有效期为一天
        cookie.setMaxAge(24*60*60);
        resp.addCookie(cookie);

    }

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

一个网站cookie是否存在上线!

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

删除Cookie:

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

编码:URLEncoder.encode(“青简工作室”,“utf-8”)

解码:URLDecoder.decode(cookie.getValue(),“utf-8”)

7.4Session(重点)

什么是Session:

  • 服务器会给每一个用户(浏览器)创建一个Session对象

  • 一个Session独占一个浏览器,只要浏览器没有关闭,这个Session就存在

  • 用户登录之后,整个网站它都可以访问–>保存用户的信息;保存购物车的信息…

Session和Cookie的区别

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

Session创建的时候做了什么事情?

Cookie cookie = new Cookie("JSESSIONID", sessionId);
resp.addCookie(cookie);

使用场景:

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

8.JSP

8.1什么是JSP

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

最大的特点:

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

8.2JSP原理

原理:JSP到底是怎么执行的!

  • 代码层面没有任何问题
  • 服务器内部工作
    • tomcat中右一个work目录
    • IDEA使用Tomcat会在IDEA的tomcat中生成一个work目录

发现页面转变成了java程序!

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

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

JSP本质上就是一个Servlet

初始化、销毁、JSPService

  1. 判断请求
  2. 内置一些对象(页面上下文、session、applicationContext、config、out、page当前页、请求、响应)
  3. 输出页面前增加的代码(设置相应的页面类型、对象赋值)
  4. 上面的对象我们可以在JSP中直接使用

在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 < 100; i++) {
      sum+=i;
    }
    out.print("<h1>Sum="+sum+"</h1>");
  %>

脚本片段的再实现

  <%--JSP脚本片段的再实现--%>
  <%
    int x =10;
    out.print(x);
  %>
  <p>这是一个JSP文档</p>
  <%
    int y = 20;
    out.print(x);
    out.print(y);
  %>
  <hr/>

  <%--在代码中嵌入HTML元素--%>
  <%
    for (int i = 0; i < 5; i++) {
  %>
  <h1>Hello,World! <%=i%></h1>
  <%
    }
  %>

JSP声明

  <%--JSP声明<%! %>--%>
  <%!
    static {
      System.out.println("Loading Servlet!");
    }
    private int globalVar = 0;
    public void test(){
      System.out.println("进入test方法");
    }
  %>

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

在JSP,嵌入Java代码即可!

<%脚本片段%>
<%=表达式%>
<%!声明%>
<%--注释--%>

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

8.4JSP指令

<%@%>

<%--定义错误页面--%>
<%@ page errorPage="error/500.jsp" %>
<%--包含页面--%>
<%@ include file="common/header.jsp"%>

jsp:/

    <%--JSP标签--%>
    <jsp:include page="/common/footer.jsp"/>

@ include会将两个页面合二为一

jsp:include 拼接页面,本质还是分开的

8.5九大内置对象

  • PageContext 存东西
  • Request 存东西
  • Response
  • Session 存东西
  • Application【ServletContext】存东西
  • config【ServletConfig】
  • out
  • page
  • exception
<%--内置对象--%>
<%
    pageContext.setAttribute("name1","青简工作室1");//保存的数据只在一个页面中有效
    request.setAttribute("name2","青简工作室2");//保存的数据只在一次请求中有效,请求转发会携带这个数据
    session.setAttribute("name3","青简工作室3");//保存的数据只在一次会话中有效,从打开浏览器到关闭浏览器
    application.setAttribute("name4","青简工作室4");//保存的数据在服务器中有效,,从打开服务器到关闭服务器
%>

<%
    pageContext.setAttribute("hello","hello",PageContext.SESSION_SCOPE);
%>

<%--脚本片段中的代码,会被元丰不懂生成JSP.java
要求:这里面的代码,必须保证Java语法的正确性
--%>
<%
    //通过pageContext取c出我们保存的值,我们通过寻找的方式来
    //从底层到高层(作用域)page-->request-->session-->application
    //JVM: 双亲委派机制
    String name1 = (String) pageContext.getAttribute("name1");
    String name2 = (String) pageContext.getAttribute("name2");
    String name3 = (String) pageContext.getAttribute("name3");
    String name4 = (String) pageContext.getAttribute("name4");
    String name5 = (String) pageContext.getAttribute("name5");//不存在
%>
<%--使用EL表达式输出 ${}--%>
<h1>取除的值为:</h1>
<h3>${name1}</h3>
<h3>${name2}</h3>
<h3>${name3}</h3>
<h3>${name4}</h3>
<h3>${name5}</h3>
<h3><%=name5%></h3>

使用场景

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

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

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

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

EL表达式:${}

  • 获取数据
  • 执行运算
  • 获取web开发的常用对象
  • 调用Java方法(基本不用)

JSP标签

<jsp:include page="/common/header.jsp"/>
<jsp:forward page="/common/header.jsp">
    <jsp:param name="value1" value="value1"/>
</jsp:forward>

JSTL表达式格式化标签SQL标签XML标签

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

核心标签(掌握部分)

JSTL标签库使用步骤

  • 引入对应的taglib
  • 使用其中的方法
  • 在Tomcat页需要引入jstl的包,否则会报错:JSTL解析错误

core-if

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

core-when

<%--定义一个变量score,值为85--%>
<c:set var="score" value="85"></c:set>
<c:choose>
    <c:when test="${score>=90}">
        你的成绩为优秀
    </c:when>
    <c:when test="${score>=80}">
        你的成绩为一般
    </c:when>
    <c:when test="${score>=60}">
        你的成绩为良好
    </c:when>
</c:choose>

core-foreach

<%
    ArrayList<String> people = new ArrayList<>();
    people.add(0,"张三");
    people.add(1,"李四");
    people.add(2,"王五");
    people.add(3,"赵六");
    request.setAttribute("list",people);
%>
<%--
var:每一次遍历出来的变量
items:要遍历谁
begin:开始
end:结束
step:步长
--%>
<c:forEach var="people" items="${list}">
    <c:out value="${people}"></c:out><br>
</c:forEach>

9.JavaBean

实体类

JavaBean有特定的写法:

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

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

ORM:对象关系映射

  • 表–>类
  • 字段–>属性
  • 行记录–>对象

people表

idnameageaddress
1沈金勇1号18江西
2沈金勇2号23江西
3沈金勇3号100江西

people类

class People{
    private int id;
    private String name;
    private int age;
    private String address;
}

javabean.jsp

<%
    //People people = new People();
    //people.setId();
    //people.setName();
    //people.setAge();
    //people.setAddress();
%>
<jsp:useBean id="people" class="com.simpleteen.pojo.People" scope="page"/>
<jsp:setProperty name="people" property="id" value="1"/>
<jsp:setProperty name="people" property="name" value="沈金勇"/>
<jsp:setProperty name="people" property="age" value="23"/>
<jsp:setProperty name="people" property="address" value="江西"/>

ID:<jsp:getProperty name="people" property="id"/>
姓名:<jsp:getProperty name="people" property="name"/>
年龄:<jsp:getProperty name="people" property="age"/>
地址:<jsp:getProperty name="people" property="address"/>

10.MVC三层架构

什么是MVC:Model View Controller 模型、视图、控制器

10.1早些年

Servlet和JSP都可以写Java代码,为了易于维护和使用,Servlet专注于处理请求以及控制视图跳转,JSP专注于显示数据

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

  • 控制器Controller:接收用户的请求、响应客户端内容、重定向或转发

  • 视图View(JSP):展示数据、提供可以供我们操作的请求

  • 模型JavaBean(pojo、entity):通过JDBC连接数据库

servlet-->CRUD-->数据库
弊端:程序十分臃肿,不利于维护
servlet的代码中:处理请求、响应、视图跳转 、处理JDBC、处理业务代码、处理逻辑代码
    
架构:内有什么是加一层解决不了的!

10.2MVC三层架构

  • 控制器Controller(转发或者重定向):接收用户的请求、截屏给业务层去做、视图跳转

  • 视图View(JSP)(操作控制器):展示数据模型、提供用户们操作

  • Madel(Service、Dao、JavaBean):控制业务操作,保存数据,修改数据、删除数据、查询数据

  • 模型JavaBean(pojo、entity):通过JDBC连接数据库

  • Service(login、loginout、查询全部用户)

  • Dao(login、loginout、查询全部用户)

Model

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

View

  • 展示数据
  • 提供连接发起Service请求(a、foem、img…)

Controller(Servlet)

  • 接收用户的请求:(req:请求参数、Session信息…)
  • 交给业务层处理对应的代码
  • 控制视图的跳转
登录--->接收用户的登录请求-->处理用户的请求(获取用户登录的参数,usewrname,password)-->交给业务层处理登录业务(判断用户名密码是否正确:事务)-->Dao层查询用户名和密码是否正确-->数据库

11.过滤器Filter

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

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

Filter开发步骤:

  1. 导包

    import javax.servlet.*;
    
  2. 编写过滤器,实现Filter家口,重写对应的方法即可

package com.simpleteen.filter;

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

public class CharacterEncodingFilter implements Filter {
    //初始化
    @Override
    public void init(FilterConfig filterConfig) throws ServletException {
        System.out.println("CharacterEncodingFilter初始化");
    }

    //过滤
    @Override
    public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
        servletResponse.setContentType("text/html;charset=UTF-8");
        servletResponse.setCharacterEncoding("UTF-8");
        System.out.println("CharacterEncodingFilter执行前...");
        filterChain.doFilter(servletRequest,servletResponse);
        System.out.println("CharacterEncodingFilter执行后...");
    }

    //销毁
    @Override
    public void destroy() {
        System.out.println("CharacterEncodingFilter销毁");
    }
}

3.在web.xml中配置Filter

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

总结:

  1. 过滤中所有代码,在过滤特定请求的时候都会执行
  2. 必须要让过滤器过滤通行,设置允许链式连接

12.监听器

实现一个监听器的接口;(有n种)

1.编写监听器,实现监听器接口

package com.simpleteen.listener;

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

//统计网站在线人数:统计session
public class OnlineCountListener implements HttpSessionListener {
    //创建session监听
    @Override
    public void sessionCreated(HttpSessionEvent httpSessionEvent) {
        ServletContext servletContext = httpSessionEvent.getSession().getServletContext();
        Integer onlineCount = (Integer) servletContext.getAttribute("OnlineCount");

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

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

    //销毁session监听
    @Override
    public void sessionDestroyed(HttpSessionEvent httpSessionEvent) {
        ServletContext servletContext = httpSessionEvent.getSession().getServletContext();
        Integer onlineCount = (Integer) servletContext.getAttribute("OnlineCount");

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

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

2.web.xml种注册监听器

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

13.过滤器和监听器常见应用

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

用户登录之后才能进入主页!用户注销后就不能进入主页!

  1. 用户登录之后,向Session种放入用户数据
  2. 进入主页的时候判断用户是否已经登录;要求在过滤器种实现!
package com.simpleteen.filter;

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

public class SysFilter implements Filter {
    @Override
    public void init(FilterConfig filterConfig) throws ServletException {

    }

    @Override
    public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
        HttpServletRequest request =(HttpServletRequest)servletRequest;
        HttpServletResponse response =(HttpServletResponse) servletResponse;

        if(request.getSession().getAttribute("USER_SESSION")==null){
            response.sendRedirect("/javaweb_filter_war_exploded/error.jsp");
        }

        filterChain.doFilter(servletRequest,servletResponse);
    }

    @Override
    public void destroy() {

    }
}

14.JDBC

14.1JDBC

什么是JDBC:Java连接数据库

需要jar包的支持:

  • java.sql
  • javax.sql
  • mysql-connecter-java

新建数据库

CREATE TABLE users(
id INT PRIMARY KEY,
`name` VARCHAR(40),
`password` VARCHAR(40),
email VARCHAR(60),
birthday DATE
);

导入数据库依赖

        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
            <version>5.1.47</version>
        </dependency>

jdbc测试

package com.simpleteen.test;

import java.sql.*;

public class TestJdbc {
    public static void main(String[] args) throws ClassNotFoundException, SQLException {
        //配置信息
        String url="jdbc:mysql://localhost:3306/school?useUnicode=true&characterEncoding=utf-8&useSSL=false";
        String username = "root";
        String password = "123456";
        //1.加载驱动
        Class.forName("com.mysql.jdbc.Driver");
        //2.连接数据库,代表数据库
        Connection connection = DriverManager.getConnection(url, username, password);
        //3.向数据库发送SQL的对象:CRUD
        Statement statement = connection.createStatement();
        //4.编写SQL
        String sql = "select * from users";
        //5.执行SQL
        ResultSet resultSet = statement.executeQuery(sql);
        //6.查询返回结果
        while (resultSet.next()){
            System.out.println(resultSet.getObject("id"));
            System.out.println(resultSet.getObject("name"));
            System.out.println(resultSet.getObject("password"));
            System.out.println(resultSet.getObject("email"));
            System.out.println(resultSet.getObject("birthday"));
        }
        //7.关闭连接,释放资源
        resultSet.close();
        statement.close();
        connection.close();
    }
}

JDBC固定步骤:

1.加载驱动

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

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

4.编写SQL

5.执行SQL

6.查询返回结果

7.关闭连接,释放资源

14.2事务

要么都成功,要么都失败

ACID原则:保证数据的安全

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

经典案例:转账!!!

Junit单元测试

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

简单使用

@Test注解只有在方法上有效,只要加了诸葛注解的方法,就可以直接执行!

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

搭建一个环境

CREATE TABLE ACCOUNT(
	id INT PRIMARY KEY AUTO_INCREMENT,
	`name` VARCHAR(40),
	money FLOAT
);

INSERT INTO ACCOUNT(`name`,money) VALUES('A',1000);
INSERT INTO ACCOUNT(`name`,money) VALUES('B',1000);
INSERT INTO ACCOUNT(`name`,money) VALUES('C',1000);
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值