web开发:
在java中,动态web资源开发的技术统称为JavaWeb;
什么是程序:按照自己编写的代码去执行的动作
web应用程序:可以提供浏览器访问的程序;
a.html、b.html……多个web资源,这些web资源可以被外部访问,对外界提供服务;
我们能访问的任何一个页面或者资源,都存在于这个世界的某一个角落的计算机上!
URL(资源定位符)
这个统一的web资源会被放在同一个文件夹下,web应用程序,依赖于:Tomcat:服务器
一个web应用有多部分组成(静态web,动态web)
web应用程序编写完毕后,若想提供给外界访问:需要一个服务器来统一管理;
页面会动态展示:“Web的页面展示的效果因人而异”;
缺点:
优点:
新手村: - - 魔鬼训练(分析原理、看源码)- ->PK场
实现web手段,ASP ,JSP,PHP
ASP:
php:
JSP/Servlet:
JSP本质是 Servlet
B/S:浏览和服务器
C/S:客户端和服务器
……
服务器是一种被动的操作,用来处理用户的一些请求和给用户一些响应信息
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最新版本为10.0.5。
……
工作3-5年之后 ,可以尝试手写Tomcat服务器
下载Tomcat:
Tomact官网:https://tomcat.apache.org/
文件夹信息
启动、关闭 Tomact
访问测试:http://localhost:8080/
可能遇到的问题:
可以配置启动的端口号
<Connector port="8080" redirectPort="8443" connectionTimeout="20000" protocol="HTTP/1.1"/>
可以配置主机的名称
-<Host name="localhost" autoDeploy="true" unpackWARs="true" appBase="webapps">
<Valve className="org.apache.catalina.valves.AccessLogValve" pattern="%h %l %u %t "%r" %s %b" suffix=".txt" prefix="localhost_access_log" directory="logs"/>
Host>
请你谈谈网站是如何进行访问的!(阿里四面,真题)
输入一个域名;回车
检查本机的 C:\Windows\System32\drivers\etc**\hosts** 配置文件下有没有这个域名映射;
可以配置一个环境变量(可选)
不会就模仿
-- webapps // Tomcat服务器的web目录
-POOT // 项目的名字
-study // 网站的目录名
- WEB-INF // 放网站程序的
-classes // java程序
-lib // web 应用所依赖的jar包
-web.xml
- index.html // 默认的首页
- static // 可以放资源文件
-css
-style.css
-js
-img
-......
超文本传输协议(Hypertext Transfer Protocol,HTTP)是一个简单的请求-响应协议,它通常运行在TCP之上。
Https:安全的
以百度为例:
常规
Request URL: https://www.baidu.com/ 请求地址
Request Method: GET // 请求方法 get方法/post方法
Status Code: 200 OK // 状态码:200
Remote(远程) Address: 39.156.66.14:443 // 远程地址 ,真实访问走的这个地址
// Referrer Policy: strict-origin-when-cross-origin // 协议 能访问到什么东西
请求 URL: https://www.baidu.com/
请求方法: GET
状态代码: 200 OK
远程地址: 39.156.66.14:443
引用站点策略: strict-origin-when-cross-origin
请求标头
Accept:text/html // 类型
Accept-Encoding: gzip, deflate, br // 类型的编码
Accept-Language: zh-CN,zh;q=0.9 // 类型的语言
Cache-Control: max-age=0 // 缓存控制
Connection: keep-alive // 连接
Accept: 告诉浏览器,他所支持的数据类型
Accept-Encoding: 支持哪种编码格式 GBK UTF-8 GB2312 ISO8859-1
Accept-Language: 告诉浏览器,它的语言环境
Cache-Control: 缓存控制
Connection: 告诉浏览器,请求完成是断开还是保持连接
host: 主机...
百度
响应头
Cache-Control: private 缓存-控制
Connection: keep-alive 连接:保持连接
Content-Encoding: gzip 内容-编码
Content-Type: text/html;charset=utf-8 内容-类型 html
Accept: 告诉浏览器,他所支持的数据类型
Accept-Encoding: 支持哪种编码格式 GBK UTF-8 GB2312 ISO8859-1
Accept-Language: 告诉浏览器,它的语言环境
Cache-Control: 缓存控制
Connection: 告诉浏览器,请求完成是断开还是保持连接
host: 主机...
refresh: 告诉客户端,多久刷新一次
Location: 让网页重新定义;
200:请求响应成功 200
3XX:请求重定向 303
4XX:找不到资源 404
5XX:服务器代码错误 500 505:网关错误
常见面试题:
当你的浏览器中地址栏输入地址并回车的一瞬间到页面能够展示回来,经历了什么?
为什么要学习这个技术?
在Javaweb开发中,需要使用大量的jar包,手动去导入
如何能够让一个东西自动帮我导入和配置这个jar包。
由此,maven诞生了!
目前用它来方便导入jar包的
maven的核心思想:约定大于配置
maven会规定好你如何去编写java代码,必须按照这个来;
官网:http://maven.apache.org/download.cgi
建议:电脑上的所有环境都放在一个文件夹下,方便管理
在体统环境变量中
配置如下:
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-fr5yJeyE-1636270388229)(javaWeb.assets/image-20210609203817053.png)]
测试Maven是否安装成功,保证必须配置完毕!
alimaven
central
aliyun maven
http://maven.aliyun.com/nexus/content/groups/public/
在本地的仓库,远程仓库;
**建立一个本地仓库:**localRepository
D:\学习资料\MarkDown学习\javaWeb\apache-maven-3.8.1-bin\apache-maven-3.8.1\maven-repo
只有在web应用下才会有!
解决警告问题
必须要的配置: 为什么会有这个问题:我们访问一个网站,需要指定一个文件夹名字
pom.xml 是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.0modelVersion>
<groupId>com.kuanggroupId>
<artifactId>javaweb-01-mavenartifactId>
<version>1.0-SNAPSHOTversion>
<packaging>warpackaging>
<name>javaweb-01-maven Maven Webappname>
<url>http://www.example.comurl>
<properties>
<project.build.sourceEncoding>UTF-8project.build.sourceEncoding>
<maven.compiler.source>1.7maven.compiler.source>
<maven.compiler.target>1.7maven.compiler.target>
properties>
<dependencies>
<dependency>
<groupId>junitgroupId>
<artifactId>junitartifactId>
<version>4.11version>
<scope>testscope>
dependency>
dependencies>
<build>
<finalName>javaweb-01-mavenfinalName>
<pluginManagement>
<plugins>
<plugin>
<artifactId>maven-clean-pluginartifactId>
<version>3.1.0version>
plugin>
<plugin>
<artifactId>maven-resources-pluginartifactId>
<version>3.0.2version>
plugin>
<plugin>
<artifactId>maven-compiler-pluginartifactId>
<version>3.8.0version>
plugin>
<plugin>
<artifactId>maven-surefire-pluginartifactId>
<version>2.22.1version>
plugin>
<plugin>
<artifactId>maven-war-pluginartifactId>
<version>3.2.2version>
plugin>
<plugin>
<artifactId>maven-install-pluginartifactId>
<version>2.5.2version>
plugin>
<plugin>
<artifactId>maven-deploy-pluginartifactId>
<version>2.8.2version>
plugin>
plugins>
pluginManagement>
build>
project>
maven由于他的约定大于配置,我们之后可以能遇到我们写的配置文件,无法被导出或者生效的问题,解决方案:
解决方案:更新上一个版本
Tomcat闪退
IDEA中每次都要重复配置Maven
在IDEA中的全局默认配置中去配置
Maven项目中Tomcat没法配置
maven默认web项目的web.xml版本问题
替换为webapp 的版本和Tomcat一致
<web-app xmlns="https://jakarta.ee/xml/ns/jakartaee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="https://jakarta.ee/xml/ns/jakartaee https://jakarta.ee/xml/ns/jakartaee/web-app_5_0.xsd" version="5.0" metadata-complete="true"> <display-name>Welcome to Tomcatdisplay-name> <description> Welcome to Tomcat description>web-app>
Maven仓库的使用
地址 :https://mvnrepository.com/
Servlet就是sun公司开发动态web的一门技术
Sun在这些API中提供一个接口叫做:Servlet,如果你想开发一个Servlet程序,只需要完成两个小步骤:
什么是Servlet:把实现了Servlet接口的Java程序叫做,Servlet
servlet的本质是什么,它是如何工作的?
参考知乎:https://www.zhihu.com/question/21416727/answer/690289895
Serlvet接口在Sun公司有两个默认的实现类:HttpServlet,GenericServlet
构建一个普通的Maven项目,删掉里面的src目录,以后我们学习就在这个项目里面建立Model;这个空的工程就是Maven主工程;
关于Maven父子工程的理解:
父项目中会有
<modules>
<module>servlet-01module>
modules>
子项目中会有
注意:maven3.6.0刚开始有parent后面就没有了,变成了自动继承了,
package com.kuang.servlet;
import javax.servlet.ServletException;
import javax.servlet.ServletOutputStream;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.PrintWriter;
/**
* 第一个Servlet
*
* @author <[email protected]>
* @since 2021/6/13 9:04
*/
public class HelloServlet extends HttpServlet {
// 重写方法
// 由于get或者post只是请求实现的不同的方式,可以相互调用,因为业务逻辑都一样
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// 代码在这里面写
// 输入
// 获得输入流
// req.getInputStream()
// 输出
// IO流分两种:1、resp.getWriter(); 2、resp.getOutputStream()
// ServletOutputStream outputStream = resp.getOutputStream();
PrintWriter writer = resp.getWriter();//响应流
writer.print("HelloServlet");
// 一个类搞定了
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
分析源码:
编写Servlet的映射
为什么需要映射:我们写的是java程序,但是要通过浏览器访问,而浏览器需要连接web服务器,所有我们需要再web服务器中注册我们写的Servlet,还需要给他一个浏览器能够访问的路径;
<!--注册Servlet-->
<servlet>
<servlet-name>Hello</servlet-name>
<servlet-class>com.kuang.servlet.HelloServlet</servlet-class>
</servlet>
<!--Servlet的请求路径-->
<servlet-mapping>
<servlet-name>Hello</servlet-name>
<url-pattern>/hello</url-pattern>
</servlet-mapping>
Servlet是由Web服务器调用,web服务器在收到 浏览器请求之后,会:
Mapping:它是请求映射的一个路径
<servlet-mapping>
<servlet-name>Helloservlet-name>
<url-pattern>/hellourl-pattern>
servlet-mapping>
<servlet-mapping>
<servlet-name>helloservlet-name>
<url-pattern>/hellourl-pattern>
servlet-mapping>
<servlet-mapping>
<servlet-name>helloservlet-name>
<url-pattern>/hello1url-pattern>
servlet-mapping>
<servlet-mapping>
<servlet-name>helloservlet-name>
<url-pattern>/hello2url-pattern>
servlet-mapping>
<servlet-mapping>
<servlet-name>helloservlet-name>
<url-pattern>/hello3url-pattern>
servlet-mapping>
<servlet-mapping>
<servlet-name>helloservlet-name>
<url-pattern>/hello4url-pattern>
servlet-mapping>
<servlet-mapping>
<servlet-name>helloservlet-name>
<url-pattern>/hello/*url-pattern>
servlet-mapping>
<servlet-mapping>
<servlet-name>helloservlet-name>
<url-pattern>/*url-pattern>
servlet-mapping>
[自定义]
<servlet-mapping>
<servlet-name>helloservlet-name>
<url-pattern>*.yinurl-pattern>
servlet-mapping>
优先级问题
指定了固有的映射路径优先级最高,如果找不到就会走默认的处理请求
<servlet>
<servlet-name>errorservlet-name>
<servlet-class>com.kuang.servlet.ErrorServletservlet-class>
servlet>
<servlet-mapping>
<servlet-name>errorservlet-name>
<url-pattern>/*url-pattern>
servlet-mapping>
通过这种方式去处理一些错误的请求
package com.kuang.servlet;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.PrintWriter;
/**
* 请求,专门来处理404页面找不到
*
* @author <[email protected]>
* @since 2021/6/13 11:31
*/
public class ErrorServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// 输出
// 以HTML展示 ,并且编码为 utf-8
resp.setContentType("text/html");
resp.setCharacterEncoding("utf-8");
// 展示完给他个输出对象
PrintWriter writer = resp.getWriter();
writer.print("404
");
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
解决乱码问题:
java.util.logging.ConsoleHandler.encoding = utf-8
改为:java.util.logging.ConsoleHandler.encoding = GBK
web容器在启动的时候,它会为每一web程序都创建一个对应的ServletContext对象,他代表了当前的web应用
我在这个Servlet中保存的数据,可以在另外一个servlet中拿到;
package com.kunag.servlet;
import javax.servlet.ServletContext;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
public class HelloServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// this.getInitParameter() 获得初始化的参数 初始化参数
// this.getServletConfig() 获取服务配置 Servlet配置
// this.getServletContext() 获取服务上下文 Servlet上下文
ServletContext servletContext = this.getServletContext();
String username = "小尹";// 数据
servletContext.setAttribute("username",username);//将一个数据保存在了ServletContext中,名字为:username,值 username
}
}
public class GetServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
ServletContext servletContext = this.getServletContext();
String username = (String) servletContext.getAttribute("username");
resp.setContentType("text/html");
resp.setCharacterEncoding("utf-8");
resp.getWriter().print("名字"+username);
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
<servlet>
<servlet-name>helloservlet-name>
<servlet-class>com.kunag.servlet.HelloServletservlet-class>
servlet>
<servlet-mapping>
<servlet-name>helloservlet-name>
<url-pattern>/hellourl-pattern>
servlet-mapping>
<servlet>
<servlet-name>getcservlet-name>
<servlet-class>com.kunag.servlet.GetServletservlet-class>
servlet>
<servlet-mapping>
<servlet-name>getcservlet-name>
<url-pattern>/getcurl-pattern>
servlet-mapping>
<context-param>
<param-name>urlparam-name>
<param-value>jdbc:mysql://localhost:3306/mabatisparam-value>
context-param>
public class ServletDemo03 extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
ServletContext servletContext = this.getServletContext();
// 获取参数
String url = servletContext.getInitParameter("url");
resp.getWriter().print(url);
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
public class ServletDemo04 extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
ServletContext servletContext = this.getServletContext();
System.out.println("进入了ServletDemo04");
// 请求转发
// /:代表当前项目,转发到当前的gp请求 forward:转发
/*RequestDispatcher requestDispatcher = servletContext.getRequestDispatcher("/gp");// 转发的请求路径
requestDispatcher.forward(req,resp);// 调用 forward 方法实现请求转发*/
// 两句话合并成一句话, 不用去写返回值
servletContext.getRequestDispatcher("/jp").forward(req,resp);
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
<servlet>
<servlet-name>sd4servlet-name>
<servlet-class>com.kunag.servlet.ServletDemo04servlet-class>
servlet>
<servlet-mapping>
<servlet-name>sd4servlet-name>
<url-pattern>/sd4url-pattern>
servlet-mapping>
Properties类
发现:都被打包到了同一个路径下:classes,俗称这个路径为classpath;类路径
响应:
web服务器接收到客户端的http请求,针对这个请求,分别创建一个代表请求的HttpServletRequest对象,代表响应的一个HttpServletResponse;
负责向浏览器发送数据的方法
ServletOutputStream getOutputStream() throws IOException;
// 写中文用 getWriter() 写平常的流用 getOutputStream()
PrintWriter getWriter() throws IOException;
负责向浏览器发送响应头的方法
void setCharacterEncoding(String var1);
void setContentLength(int var1);
void setContentLengthLong(long var1);
void setContentType(String var1);
void setDateHeader(String var1, long var2);
void addDateHeader(String var1, long var2);
void setHeader(String var1, String var2);
void addHeader(String var1, String var2);
void setIntHeader(String var1, int var2);
void addIntHeader(String var1, int var2);
响应的状态码
int SC_CONTINUE = 100;
int SC_SWITCHING_PROTOCOLS = 101;
int SC_OK = 200;
int SC_CREATED = 201;
int SC_ACCEPTED = 202;
int SC_NON_AUTHORITATIVE_INFORMATION = 203;
int SC_NO_CONTENT = 204;
int SC_RESET_CONTENT = 205;
int SC_PARTIAL_CONTENT = 206;
int SC_MULTIPLE_CHOICES = 300;
int SC_MOVED_PERMANENTLY = 301;
int SC_MOVED_TEMPORARILY = 302;
int SC_FOUND = 302;
int SC_SEE_OTHER = 303;
int SC_NOT_MODIFIED = 304;
int SC_USE_PROXY = 305;
int SC_TEMPORARY_REDIRECT = 307;
int SC_BAD_REQUEST = 400;
int SC_UNAUTHORIZED = 401;
int SC_PAYMENT_REQUIRED = 402;
int SC_FORBIDDEN = 403;
int SC_NOT_FOUND = 404;
int SC_METHOD_NOT_ALLOWED = 405;
int SC_NOT_ACCEPTABLE = 406;
int SC_PROXY_AUTHENTICATION_REQUIRED = 407;
int SC_REQUEST_TIMEOUT = 408;
int SC_CONFLICT = 409;
int SC_GONE = 410;
int SC_LENGTH_REQUIRED = 411;
int SC_PRECONDITION_FAILED = 412;
int SC_REQUEST_ENTITY_TOO_LARGE = 413;
int SC_REQUEST_URI_TOO_LONG = 414;
int SC_UNSUPPORTED_MEDIA_TYPE = 415;
int SC_REQUESTED_RANGE_NOT_SATISFIABLE = 416;
int SC_EXPECTATION_FAILED = 417;
int SC_INTERNAL_SERVER_ERROR = 500;
int SC_NOT_IMPLEMENTED = 501;
int SC_BAD_GATEWAY = 502;
int SC_SERVICE_UNAVAILABLE = 503;
int SC_GATEWAY_TIMEOUT = 504;
int SC_HTTP_VERSION_NOT_SUPPORTED = 505;
//ServletInputStream inputStream = req.getInputStream();// 获得输入流
//IO流分为两种
//ServletOutputStream outputStream = resp.getOutputStream(); //有输入一定有这两个东西
PrintWriter writer = resp.getWriter(); // 响应流 向外面写东西
writer.print("Hello,Serlvet");// 往外写东西
下载文件
package com.kuang;
import javax.servlet.ServletException;
import javax.servlet.ServletOutputStream;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.FileInputStream;
import java.io.IOException;
import java.net.URLEncoder;
/**
* 下载文件
* @author <[email protected]>
* @since 2021/6/14 14:41
*/
public class servlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// 1. 要获取下载文件的路径
String realPath = "D:\\IDEA数据\\javaweb\\javaweb-02-01-servlet\\response\\src\\main\\resources\\云.jpg";
// D:\IDEA数据\javaweb\javaweb-02-01-servlet\response\src\main\resources\云.jpg
System.out.println("下载文件的路径:"+realPath);
// 2. 下载的文件名是啥?
// 截取字符串
String fileName = realPath.substring(realPath.lastIndexOf("\\") + 1);
// 3. 设置想办法让浏览器能够支持(Content-disposition)下载我们需要的东西
//设置浏览器行为
/*如果想让网站下载东西,需要搜一个下载文件的头*/
//中文文件名URLEncoder.encode编码,否则有可能乱码
resp.setHeader("Content-disposition","attachment;filename="+ URLEncoder.encode(fileName,"UTF-8"));
// 4. 获取下载文件的输入流
FileInputStream in = new FileInputStream(re alPath);
// 5. 输出:创建缓冲区
int len = 0;
byte[] buffer = new byte[1024];
// 6. 获取OutputStream对象
ServletOutputStream out = resp.getOutputStream();
// 7. 将FileOuputStream流写入到buffer缓冲区
while ((len=in.read(buffer))>0){
out.write(buffer,0,len);
}
// 8. 使用OutputStream将缓冲区中的数据输出到客户端!
in.close();
out.close();
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
验证怎么来的?
新建一个类
package com.kuang;
import javax.imageio.ImageIO;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.awt.*;
import java.awt.image.BufferedImage;
import java.io.BufferedInputStream;
import java.io.IOException;
import java.util.Random;
/**
* 验证码功能
* @author <[email protected]>
* @since 2021/6/14 15:36
*/
public class ImageServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// 如何让浏览器3秒自动刷新一次
// refresh 让当前函数刷新
resp.setHeader("refresh","3");
// 在内存中创建一个图片
BufferedImage image = new BufferedImage(80,20,BufferedImage.TYPE_INT_RGB);
// 得到图片
//在图片上写东西 2D的笔
Graphics2D g = (Graphics2D) image.getGraphics(); // 笔
// 设置图片的背景颜色
//白色的背景
g.setColor(Color.white);
//背景颜色
g.fillRect(0,0,80,20);
// 给图片写数据
g.setColor(Color.blue);
// 字体
g.setFont(new Font(null,Font.BOLD,20));
// 写东西
g.drawString(makeNum(),0,20);
/*告诉浏览器,这个请求用图片的方式打开*/
resp.setContentType("image/jpg");
//网站存在缓存 ,不然浏览器缓存
// 设置它得头类型 让他为-1不缓存
resp.setDateHeader("expires",-1);
//告诉浏览器 Cache-Control:缓存控制 no-cache :不缓存
resp.setHeader("Cache-Control","no-cache");
resp.setHeader("Pragma","no-cache");
//把图片写给浏览器
// ImageIO是图片类,专门写图片的
ImageIO.write(image,"jpg",resp.getOutputStream());
}
// 生成随机数
private String makeNum(){
// 创建随机类
Random random = new Random();
// 生成一个随机数
String num = random.nextInt(9999999) + "";
// 随机数往外边写 StringBuffer可变长字符串
StringBuffer sb = new StringBuffer();
// 遍历
for (int i = 0; i < 7-num.length(); i++) {
// 追加: append
sb.append("0");
}
num = sb.toString() + num;
return num;
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
在xml下配置
<servlet>
<servlet-name>ImageServletservlet-name>
<servlet-class>com.kuang.ImageServletservlet-class>
servlet>
<servlet-mapping>
<servlet-name>ImageServletservlet-name>
<url-pattern>/imgurl-pattern>
servlet-mapping>
一个web资源(B)收到客户端(A)请求后,它(B)会通知客户端(A)去访问另外一个web资源(C),这个过程叫做重点向
常见场景:
方法
void sendRedirect(String var1) throws IOException;
测试
package com.kuang;
public class RedirectServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// 发送一个请求
/*
原理
resp.setHeader("Location","/r/img");
resp.setStatus(302);
* */
resp.sendRedirect("/r/img"); // 重定向
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
<servlet>
<servlet-name>RedirectServletservlet-name>
<servlet-class>com.kuang.RedirectServletservlet-class>
servlet>
<servlet-mapping>
<servlet-name>RedirectServletservlet-name>
<url-pattern>/redurl-pattern>
面试题:重定向和转发的区别?
相同点
不同点
重定向跳转
类
package com.kuang;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
/**
* @author <[email protected]>
* @since 2021/6/14 16:58
*/
public class RequestTest extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
System.out.println("进入这个请求");
// 处理请求 从请求里面获取一个参数
String username = req.getParameter("username");
String password = req.getParameter("password");
System.out.println(username+":"+password);
// 重定向 跳转 注意路径问题,否则404没找到页面 500 代表代码错了
resp.sendRedirect("/r/success.jsp");
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
配置xml
<servlet>
<servlet-name>requestservlet-name>
<servlet-class>com.kuang.RequestTestservlet-class>
servlet>
<servlet-mapping>
<servlet-name>requestservlet-name>
<url-pattern>/loginurl-pattern>
servlet-mapping>
jsp跳转前
Hello World!
<%--这里提交的路径,需要寻找到项目的路径--%>
<%--${pageContext.request.contextPath}代表 :当前项目--%>
jsp先带入jar包
jsp跳转后
<%--
Created by IntelliJ IDEA.
User: Lenovo
Date: 2021/6/14
Time: 17:52
To change this template use File | Settings | File Templates.
--%>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
Title
Success
重点在request中有两个应用场景
类
package com.kuang.servlet;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.util.Arrays;
/**
* @author <[email protected]>
* @since 2021/6/14 20:05
*/
public class LoginServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
req.setCharacterEncoding("utf-8");
resp.setCharacterEncoding("utf-8");
String username = req.getParameter("username");
String password = req.getParameter("password");
String[] hobbys = req.getParameterValues("hobbys");
System.out.println("=============");
System.out.println(username);
System.out.println(password);
System.out.println(Arrays.toString(hobbys));
System.out.println("=====================");
//方式一、resp.sendRedirect("/r/img"); // 重定向
//方式二、this.getServletContext() 转发
//方式三、通过请求转发
//通过请求转发
//说明这里的 / 代表当前的web应用
req.getRequestDispatcher(req.getContextPath()+"/success.jsp").forward(req,resp);
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
xml文件
<web-app xmlns="https://jakarta.ee/xml/ns/jakartaee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="https://jakarta.ee/xml/ns/jakartaee
https://jakarta.ee/xml/ns/jakartaee/web-app_5_0.xsd"
version="5.0"
metadata-complete="true">
<servlet>
<servlet-name>LoginServletservlet-name>
<servlet-class>com.kuang.servlet.LoginServletservlet-class>
servlet>
<servlet-mapping>
<servlet-name>LoginServletservlet-name>
<url-pattern>/loginurl-pattern>
servlet-mapping>
web-app>
转发前的jsp代码
<%--
Created by IntelliJ IDEA.
User: Lenovo
Date: 2021/6/14
Time: 20:03
To change this template use File | Settings | File Templates.
--%>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
登录
登录
<%--action 请求的地址 ${pageContext.request.contextPath}/login :代表外部项目,下面有了login --%>
<%--表单的意思:以post方式提交表单,提交到我们的login请求--%>
转发后的jsp代码
<%--
Created by IntelliJ IDEA.
User: Lenovo
Date: 2021/6/14
Time: 20:11
To change this template use File | Settings | File Templates.
--%>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
Title
成功
请求转发和重定向的区别
相同点
不同点
**会话:**用户打开一个浏览器,点击了很多超链接,访问多个web资源,关闭浏览器,这个过程可以称之为会话。
**有状态会话:**一个同学来过教室,下次再来教室,我们会知道这个同学,曾经来过,称之为有状态会话;(记录了某些东西)
现实生活中:你能怎么证明你是学生?
你 学校
网络中:一个网站,怎么证明你来过?
客户端 服务端
cookie
session
常见场景:网站登录之后,下次不用再登录了,第二次访问直接就进去了
一个网站,怎么证明你来过?
Cookie方法
Cookie[] cookies = req.getCookies();//获得cookie
cookie.getName();//获得cookie中的key
cookie.getValue();//获得cookie中的值
new Cookie("lastLoginTime", System.currentTimeMillis()+"");//新建一个cookie
cookie.setMaxAge(24*60*60);//设置cookie的有效期
resp.addCookie(cookie);//响应给客户端一个cookie
cookie:一般会保存在本地的 用户目录下 appdata;
问题:一个网站cookie是否存在上限!细节问题
删除cookie
编码解码:
URLEncoder.encode("小明","utf-8") //编码
URLDecoder.decode(cookie.getValue(),"UTF-8"); // 解码
代码
保存用户上一次访问的时间
package com.kuang.servlet;
import javafx.scene.chart.PieChart;
import javax.servlet.ServletException;
import javax.servlet.http.Cookie;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.PrintWriter;
import java.util.Date;
/**
* 保存用户上一次访问的时间
* @author <[email protected]>
* @since 2021/6/15 16:57
*/
public class CookieDemo01 extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//服务器,告诉你,你来的时间,把这个时间封装成为一个 信件,你下次带来,我就知道你来了
//解决中文乱码问题
req.setCharacterEncoding("utf-8");
resp.setCharacterEncoding("utf-8");
//客户端返回一些字符串
// 需要输出对象 响应
PrintWriter out = resp.getWriter();
//Cookie ,服务器端从客户端获取
Cookie[] cookies = req.getCookies();//这里返回数组,说明Cookie可能存在多个
//判断,Cookie是否存在
if (cookies!=null){
//如果存在怎么办? 遍历数组
out.write("你上一次访问的时间是:");
/*for (Cookie cookie : cookies) {
}*/
//两种区别 ,下边的这个更灵活,可以取到每一个的下标
for (int i = 0; i < cookies.length; i++) {
Cookie cookie = cookies[i];
//获取cookie名字
if (cookie.getName().equals("lastLoginTime")){
//如果拿到的名字跟他一样
//获取cookie中的值 判断是否一样
/*cookie.getValue();*/
//解析为长整型,在解析为字符串 把字符串变成一个时间戳了
long lastLoginTime = Long.parseLong(cookie.getValue());
//把时间戳变成一个对象 常用类知识
Date date = new Date(lastLoginTime);
//输出 本地时间格式化
out.write(date.toLocaleString());
}
}
}else {
out.write("这是您第一次访问本站");
}
//服务器给客户端响应(发送)创建一个cookie //获取时间
Cookie cookie = new Cookie("lastLoginTime", System.currentTimeMillis()+"");
//设置cookie的有效期 1天
cookie.setMaxAge(24*60*60);
//添加一个cookie, 响应给客户端
resp.addCookie(cookie);
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
删除上一次访问时间
package com.kuang.servlet;
import javax.servlet.ServletException;
import javax.servlet.http.Cookie;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.PrintWriter;
import java.util.Date;
/**
* 删除上一次访问时间
* @author <[email protected]>
* @since 2021/6/15 16:57
*/
public class CookieDemo02 extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//创建一个cookie
Cookie cookie = new Cookie("lastLoginTime", System.currentTimeMillis()+"");
//设置有效期 ,立马过期
cookie.setMaxAge(0);
//响应给客户端
resp.addCookie(cookie);
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
中文数据传递
package com.kuang.servlet;
import javax.servlet.ServletException;
import javax.servlet.http.Cookie;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.PrintWriter;
import java.net.URLDecoder;
import java.net.URLEncoder;
import java.util.Date;
/**
* 中文数据传递
*
* @author <[email protected]>
* @since 2021/6/15 19:51
*/
public class CookieDemo03 extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//解决请求和响应的中文乱码问题
req.setCharacterEncoding("utf-8");
resp.setCharacterEncoding("utf-8");
//输出
// 需要输出对象 响应
PrintWriter out = resp.getWriter();
//Cookie ,服务器端从客户端获取
Cookie[] cookies = req.getCookies();//这里返回数组,说明Cookie可能存在多个
//判断,Cookie是否存在
if (cookies!=null){
//如果存在怎么办? 遍历数组 取出数据
out.write("你上一次访问的时间是:");
/*for (Cookie cookie : cookies) {
}*/
//两种区别 ,下边的这个更灵活,可以取到每一个的下标
for (int i = 0; i < cookies.length; i++) {
Cookie cookie = cookies[i];
//获取cookie名字
if (cookie.getName().equals("lastLoginTime")){
/*解码*/
out.write(URLDecoder.decode(cookie.getValue(),"utf-8"));
}
}
}else {
out.write("这是您第一次访问本站");
}
// 需要编码和解码 最好用URLEncoder.encode() 编码
Cookie cookie = new Cookie("name", URLEncoder.encode("小明","utf-8"));
// 取出数据
out.write(cookie.getValue());
// 响应给客户端
resp.addCookie(cookie);
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
一个网站,怎么证明你来过?
什么是Session:
常用的Session方法
Session 和 Cookie 区别:
使用场景:
使用Session:
package com.kuang.servlet;
import com.kuang.pojo.Person;
import javax.servlet.ServletException;
import javax.servlet.http.*;
import java.io.IOException;
/**
* Session
* @author <[email protected]>
* @since 2021/6/16 14:48
*/
public class SessionDemo01 extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
/* 1、解决乱码问题*/
req.setCharacterEncoding("utf-8");
resp.setCharacterEncoding("utf-8");
// 响应回去是个HTML页面 设置浏览器响应的格式
resp.setContentType("text/html;charset=utf-8");
/*2、得到Session*/
// 从请求里面拿!
HttpSession session = req.getSession();
/*3、 给Session中存东西 ~ */
session.setAttribute("name",new Person("小尹",1));
/*方法*/
// 获取Session的ID
String Sessionid = session.getId();
// 判断Session是不是新创建的
if (session.isNew()){
// write: 往外写出去 print:打印
resp.getWriter().write("Session创建成功,ID:"+Sessionid);
}else {
resp.getWriter().write("Session已经在服务器中存在了,ID:"+Sessionid);
}
/*Session创建的时候做了什么事情?*/
// 键:JSESSIONID 值:Sessionid
// Cookie cookie = new Cookie("JSESSIONID",Sessionid);
// // Session在创建的时候做了这样一个cookie,把cookie响应回去了,不然不可能在cookie中看到这个东西
// resp.addCookie(cookie);
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
得到Session
package com.kuang.servlet;
import com.kuang.pojo.Person;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession;
import java.io.IOException;
/**
* 取Session
* @author <[email protected]>
* @since 2021/6/16 14:48
*/
public class SessionDemo02 extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
/* 1、解决乱码问题*/
req.setCharacterEncoding("utf-8");
resp.setCharacterEncoding("utf-8");
// 响应回去是个HTML页面 设置浏览器响应的格式
resp.setContentType("text/html;charset=utf-8");
/*2、得到Session*/
// 从请求里面拿!
HttpSession session = req.getSession();
/*3、获取Session*/
Person name = (Person) session.getAttribute("name");
System.out.println(name.toString());
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
注销
package com.kuang.servlet;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession;
import java.io.IOException;
/**
* 注销 Session
*
* @author <[email protected]>
* @since 2021/6/16 15:53
*/
public class SessionDemo03 extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// 获得Session
HttpSession session = req.getSession();
// 取消
session.removeAttribute("name");
// 手动注销 Session
session.invalidate();
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
会话自动过期:web.xml中配置
<session-config>
<session-timeout>1session-timeout>
session-config>
理解!
Java Server Pages :Java服务器端页面,也和Servlet一样,用于开发动态Web技术!
最大的特点:
思路:JSP到底怎么执行的!
地址:C:\Users\Lenovo\AppData\Local\JetBrains\IntelliJIdea2021.1\tomcat
电脑的地址
C:\Users\Lenovo\AppData\Local\JetBrains\IntelliJIdea2021.1
\tomcat\52743147-623a-422a-935b-
ca4218af9099\work\Catalina\localhost\ROOT\org\apache\jsp
页面转变成了 java 程序
浏览器向服务器发送请求,不管访问什么资源,其实都是在访问Servlet!
JSP最终也会被转换成为一个Java类!
JSP本质上就是一个Servlet (继承关系)
源码剖析
// 初始化
public void _jspInit() {
}
// 销毁
public void _jspDestroy() {
}
// JSPService (JSP的服务)
public void _jspService(HttpServletRequest request, HttpServletResponse response)
// Tomact每个它都要走这个请求服务,会去处理请求和响应,请求和响应可以做事情
请求和响应做的事情
if (!javax.servlet.DispatcherType.ERROR.equals(request.getDispatcherType())) {
final java.lang.String _jspx_method = request.getMethod();
if ("OPTIONS".equals(_jspx_method)) {
response.setHeader("Allow","GET, HEAD, POST, OPTIONS");
return;
}
if (!"GET".equals(_jspx_method) && !"POST".equals(_jspx_method) && !"HEAD".equals(_jspx_method)) {
response.setHeader("Allow","GET, HEAD, POST, OPTIONS");
response.sendError(HttpServletResponse.SC_METHOD_NOT_ALLOWED, "JSP 只允许 GET、POST 或 HEAD。Jasper 还允许 OPTIONS");
return;
}
}
final javax.servlet.jsp.PageContext pageContext; // pageContext 页面上下文
javax.servlet.http.HttpSession session = null; // session
final javax.servlet.ServletContext application; // applicationContext 它本质:ServletContext 它的作用域非常高
final javax.servlet.ServletConfig config; // config 配置
javax.servlet.jsp.JspWriter out = null; // out 输出
final java.lang.Object page = this; // page 当前页
HttpServletRequest request // request 请求
HttpServletResponse response // response 响应
response.setContentType("text/html"); // 设置它得页面响应类型: text/html
pageContext = _jspxFactory.getPageContext(this, request, response,
null, true, 8192, true);// getPageContext 初始化创造对象
_jspx_page_context = pageContext; // pageContext 给父类一个值
application = pageContext.getServletContext(); //
config = pageContext.getServletConfig(); // 获取了一个配置对象
session = pageContext.getSession();
out = pageContext.getOut();
_jspx_out = out;
jsp本质就是servlet,只不过简化了一个东西,只需要写页面部分就可以
JSP原理流程图:
JSP代码
<%--
Created by IntelliJ IDEA.
User: Lenovo
Date: 2021/6/16
Time: 17:49
To change this template use File | Settings | File Templates.
--%>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
Title
<%
// java代码
String name = "小尹";
%>
<%--使用java代码
输出
--%>
<%--name输出到页面上--%>
name:<%=name%>
源码:
结论:
在JSP页面中;
只要是Java代码就会原封不动的输出;
如果是HTML代码,就会被转换为
out.write("\r\n");
这样的格式,输出到前端
jsp帮我们简化页面这个块的编写
1、新建一个普通项目
2、操作
<dependencies>
<dependency>
<groupId>javax.servletgroupId>
<artifactId>servlet-apiartifactId>
<version>2.5version>
dependency>
<dependency>
<groupId>javax.servlet.jspgroupId>
<artifactId>javax.servlet.jsp-apiartifactId>
<version>2.3.1version>
dependency>
<dependency>
<groupId>javax.servlet.jsp.jstlgroupId>
<artifactId>jstl-apiartifactId>
<version>1.2version>
dependency>
<dependency>
<groupId>taglibsgroupId>
<artifactId>standardartifactId>
<version>1.1.2version>
dependency>
dependencies>
3、新建jsp文件
任何语言都有自己的语法,Java中有。JSP作为java技术的一种应用,它拥有一些自己扩充的语法(了解,知道即可!),java所有语法都支持!
<%--JSP表达式
作用:用来将程序的输出,输出到客户端
公式:<%= 变量或者表达式%>
--%>
<%-- 时间--%>
<%= new java.util.Date()%>
<%--jsp脚本片段--%>
<%
int sun = 0;
for (int i = 1; i <100 ; i++) {
sun+=i;
}
//输出一句话
out.println("Sum="+sun+"
");
%>
脚本片段的再实现
<%--脚本片段的再实现--%>
<%
int x =10;
out.println(x);
%>
这是JSP文档
<%
int y=20;
out.println(y);
%>
<%--进阶:在代码嵌入HTML元素--%>
<%
for (int i = 0; i <5 ; i++) {
%>
Hello,word <%= i%>
<%
}
%>
<%--jsp声明--%>
<%!
//static 静态代码块
static {
System.out.println("Loading Servlet");
}
// 全局变量
private int globalVar = 0;
public void kuang(){
System.out.println("进入了方法Kuang!");
}
%>
作用域更高了
区别:
JSP声明:会被编译到JSP生成Java的类中!其他的,就会被生成到_jspService方法中!
方法里面的可以调用方法外边的
在JSP中,嵌入Java代码即可!
优化
<%--进阶:在代码嵌入HTML元素--%>
<%--EL表达式--%>
<% for (int i = 0; i <5 ; i++) { %>
Hello,word <%= i%>
<% } %>
总结
<%%> 片段
<%= %> 表达式,输出一个值
<%! %> 定义全局的
<%--注释--%>
JSP的注释,不会再客户端显示,HTML的注释会! JSP安全性高
<%@page pageEncoding="utf-8" %>
<%@page page... %>
<%--提取公共页面--%>
<%@include file=""%> include包含一个文件
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<%--主要页面--%>
Title
<%--脚本语言--%>
<%--file :文件--%>
<%--@include会将两个页面合二为一--%>
<%@include file="common/header.jsp"%>
网页主体
<%-- <%int i =10; %>
页面报错 栈错误 重名了 --%>
<%@include file="common/footer.jsp"%>
<%--JSP标签--%>
<%--page:页面--%>
<%--
jsp:include 拼接页面 本质还是三个
--%>
网页主体
<%-- <%
int i =10; // 直接报错!
%>--%>
<%–JSP标签–%>
<%–脚本语言–%>
跳转到500或404页面
新建一个包存图片
新建一个404或500.jsp文件
<error-page>
<error-code>404error-code>
<location>/error/404.jsplocation>
error-page>
<error-page>
<error-code>500error-code>
<location>/error/500.jsplocation>
error-page>
//存东西
pageContext.setAttribute("name1", "小尹1号");//保存的数据只在一个页面中有效
request.setAttribute("name2","小尹2号"); //保存的数据只在一次请求中有效,请求转发会携带这个数据
session.setAttribute("name3","小尹3号"); //保存的数据只在一次会话中有效,从打开浏览器到关闭浏览器
application.setAttribute("name4","小尹4号"); //保存的数据只在服务器中有效,从打开服务器到关闭服务器
request:客户端向服务器发送请求,产生的数据,用户看完了就没用了,比如:新闻,用户看完就没用的了
session:客户端向服务器发送请求,产生的数据,用户看完了还会用,比如:购物车;
application:客户端向服务器发送请求,产生的数据,一个用户用完了,其他用户还可以 使用,比如:聊天数据;
作用域的概念 重点!
JSTL标签 需要导包
<dependency>
<groupId>javax.servlet.jsp.jstlgroupId>
<artifactId>jstl-apiartifactId>
<version>1.2version>
dependency>
<dependency>
<groupId>taglibsgroupId>
<artifactId>standardartifactId>
<version>1.1.2version>
dependency>
LE表达式:${ }
jsp标签
<%--标签--%>
<%--包含上下文--%>
<%--
<%--
相当于转发的时候携带了两个参数
http://localhost:8080/jstag.jsp?name=xiaoyin&age=12
--%>
<%--跳转到那个页面--%>
<%--参数--%>
<%--转发的时候是可以写东西的--%>
JSTL表达式
JSTL标签库的使用就是为了弥补HTML标签的不足;它自定义许多标签,可以供我们使用,标签的功能和java代码一样!
格式化标签
SQL标签
XML 标签
核心标签(掌握部分)
JSTL标签库使用步骤
c:if 重点!
if测试
<%--判断如果提交的用户名是管理员,则登录成功--%>
<%-- test:必须要写的 var :接收他的返回值的 scope:作用域 --%>
<%--自闭和标签--%>
c:choose c:when
<%--用c:set 保存一些数据--%>
<%--定义一个变量score,值为85--%>
<%--判断--%>
<%--choose:选择--%>
你的成绩为优秀
你的成绩为良好
你的成绩为一般
你的成绩为不及格
c:forEach 数据库大量用到它!
<%--建一个对象--%>
<%
ArrayList people = new ArrayList<>();
// people :人 add:添加
people.add(0,"张三");
people.add(1,"李四");
people.add(2,"王五");
people.add(3,"田七");
// 放的request里 当前这个页面 有个list集合
request.setAttribute("list",people);
%>
<%--遍历出来--%>
<%--
var:每一次遍历出来的变量
items: 要遍历的对象
begin: 哪里开始 默认从0开始
end: 到哪里 默认最后一个
step:步长 默认是1
--%>
Bean:豆子
实体类
JavaBean
一般用来和数据库的字段做映射 ORM;
ORM:对象关系映射
数据库的一张表
people表
id | name | age | address |
---|---|---|---|
1 | 小尹1号 | 12 | 山东 |
2 | 小尹2号 | 24 | 山东 |
3 | 小尹3号 | 30 | 山东 |
拆成一个java类
class people{
private int id;
private String name;
private int id;
private String address;
}
class A{
new people(1,"小尹1号",12,"山东");
new people(2,"小尹2号",24,"山东");
new people(3,"小尹3号",30,"山东");
}
什么是MVC:Model view Controller 模型、试图、控制器
用户直接访问控制层,控制层就可以直接操作数据库
servlet --> CRUD(增删改查) -->数据库
弊端:程序十分臃肿,不利于维护
servlet的代码中:处理请求、响应、试图跳转、处理JDBC、处理业务代码、处理逻辑代码
在中间在加一层
架构思想中:没有什么是加一层解决不了的! 如果一层解决不了,再加一层
加一层解决不了的方案:
jdbc
程序员调用jdbc
jdbc
|
|
MySQL Oracle sqlservlet……
有三层
Model
View 视图
Controller :(Servlet) 控制层
接收用户的请求:(req :请求参数、Session信息……)
交给业务层处理对应的代码
控制试图的跳转
登录-->接收用户的登录请求 --> 处理用户的请求(获取用户登录的参数,username,password)--->交给业务层处理登录业务(判断用户密码是否正确:事务) --> Dao层查询用户名和密码是否正确-->数据库
Filter:过滤器,用来过滤网站的数据
Filter开发步骤:
导包
在pom.xml下
<dependencies>
<dependency>
<groupId>javax.servletgroupId>
<artifactId>servlet-apiartifactId>
<version>2.5version>
dependency>
<dependency>
<groupId>javax.servlet.jspgroupId>
<artifactId>javax.servlet.jsp-apiartifactId>
<version>2.3.1version>
dependency>
<dependency>
<groupId>javax.servlet.jsp.jstlgroupId>
<artifactId>jstl-apiartifactId>
<version>1.2version>
dependency>
<dependency>
<groupId>taglibsgroupId>
<artifactId>standardartifactId>
<version>1.1.2version>
dependency>
<dependency>
<groupId>mysqlgroupId>
<artifactId>mysql-connector-javaartifactId>
<version>8.0.24version>
dependency>
dependencies>
package com.kuang.filter;
import javax.servlet.*;
import java.io.IOException;
/**
* 字符集编码过滤器
* @author <[email protected]>
* @since 2021/6/21 18:33
*
* 过滤器 总结
* 1.过滤器中的所有代码,在过滤特点请求的时候都会执行
* 2.必须要让过滤器 继续同行 :把这个请求往下转交 转交的代码
* filterChain.doFilter(servletRequest,servletResponse); 固定的死代码
*
*/
public class CharacterEncodingFilter implements Filter {
@Override
// 初始化 :web服务器启动,就已经初始化了,随时等待过滤器对象出现! 他和服务器一同启动的
public void init(FilterConfig filterConfig) throws ServletException {
}
@Override
// FilterChain Chain :链
public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
// 过来的请求等于utf-8编码
servletRequest.setCharacterEncoding("utf-8");
// 过去的请求等于utf-8编码
servletResponse.setCharacterEncoding("utf-8");
// 上下文
servletResponse.setContentType("text/html;charset=UTF-8");
System.out.println("CharacterEncodingFilter执行前");
// 让我们的请求继续走,如果不写,程序到这里就拦截停止了 如:两个过滤器
filterChain.doFilter(servletRequest,servletResponse);
System.out.println("CharacterEncodingFilter执行前后");
}
@Override
// 销毁 web服务器关闭的时候,过滤器会销毁
public void destroy() {
System.out.println("CharacterEncodingFilter销毁");
}
}
<servlet>
<servlet-name>ShowServletservlet-name>
<servlet-class>com.kuang.servlet.ShowServletservlet-class>
servlet>
<servlet-mapping>
<servlet-name>ShowServletservlet-name>
<url-pattern>/servlet/showurl-pattern>
servlet-mapping>
<servlet-mapping>
<servlet-name>ShowServletservlet-name>
<url-pattern>/showurl-pattern>
servlet-mapping>
<filter>
<filter-name>CharacterEncodingFilterfilter-name>
<filter-class>com.kuang.filter.CharacterEncodingFilterfilter-class>
filter>
<filter-mapping>
<filter-name>CharacterEncodingFilterfilter-name>
<url-pattern>/servlet/*url-pattern>
filter-mapping>
package com.kuang.servlet;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
/**
* 展示Servlet
* @author <[email protected]>
* @since 2021/6/21 18:35
*/
public class ShowServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//以前的处理方式
// resp.setCharacterEncoding("utf-8");
// 乱码
resp.getWriter().write("你好,世界");
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
结论:
当路径是 localhost:8080/servlet/show的时候 显示 你好,世界 过滤器启动了
当路径是localhost:8080/show的时候,显示 ??? 乱码 过滤器未启动
过滤器 总结
1.过滤器中的所有代码,在过滤特点请求的时候都会执行
2.必须要让过滤器 继续同行 :把这个请求往下转交 转交的代码
filterChain.doFilter(servletRequest,servletResponse); 固定的死代码
用代码来讲是:实现一个监听器的接口 (有N种)
统计网上在线人数: 监听 统计 session
编写一个监听器
实现监听器的接口…
package com.kuang.listener;
import javax.servlet.ServletContext;
import javax.servlet.http.HttpSessionEvent;
import javax.servlet.http.HttpSessionListener;
/**
* 统计网上在线人数: 监听 统计 session
* listener :听众
* @author <[email protected]>
* @since 2021/6/21 19:26
*/
public class OnlineCountListener implements HttpSessionListener {
@Override
// 创建session 监听 :看你的一举一动
// 一旦创建一个session 就会触发一次这个事件
public void sessionCreated(HttpSessionEvent httpSessionEvent) {
// getSession() 拿到Session getServletContext() 拿到网站上下文
ServletContext ctx = httpSessionEvent.getSession().getServletContext();
// 输出一下session的ID
System.out.println(httpSessionEvent.getSession().getId());
// 取东西
Integer onlineCountListener = (Integer) ctx.getAttribute("OnlineCountListener");
if (onlineCountListener==null){
onlineCountListener = new Integer(1);
}else {
int count = onlineCountListener.intValue();
onlineCountListener = new Integer(count+1);
}
ctx.setAttribute("OnlineCountListener",onlineCountListener);
}
@Override
// 销毁 session 监听
// 一旦销毁一个session 就会触发一次这个事件
public void sessionDestroyed(HttpSessionEvent httpSessionEvent) {
// getSession() 拿到Session getServletContext() 拿到网站上下文
ServletContext ctx = httpSessionEvent.getSession().getServletContext();
// 1.手动销毁
// httpSessionEvent.getSession().invalidate();
// 取东西
Integer onlineCountListener = (Integer) ctx.getAttribute("OnlineCountListener");
if (onlineCountListener==null){
onlineCountListener = new Integer(0);
}else {
int count = onlineCountListener.intValue();
onlineCountListener = new Integer(count-1);
}
ctx.setAttribute("OnlineCountListener",onlineCountListener);
}
/*
* Session销毁的两种情况
* 1.手动销毁 httpSessionEvent.getSession().invalidate();
* 2.自动销毁在 web.xml中配置 session 过期时间
* */
}
<listener>
<listener-class>com.kuang.listener.OnlineCountListenerlistener-class>
listener>
监听器:GUI 编程中经常使用
GUI :图形界面编程
package com.kuang.listener;
import java.awt.*;
import java.awt.event.WindowAdapter;
import java.awt.event.WindowEvent;
import java.awt.event.WindowListener;
/**
* GUL 监听器
*
* @author <[email protected]>
* @since 2021/6/21 20:31
*/
public class TestPanel {
public static void main(String[] args) {
// 新建一个界面 窗口的标题
Frame frame = new Frame("快乐");
// 面板
Panel panel = new Panel(null);
// 布局 设置窗体的布局
frame.setLayout(null);
// 给窗体设置一些坐标
frame.setBounds(300,300,300,300);
// 背景颜色
frame.setBackground(new Color( 0,0,255));
// 面板 坐标
panel.setBounds(50,50,50,50);
panel.setBackground(new Color( 255,0,0));
// 内嵌的效果
frame.add(panel);
// 设置可见性
frame.setVisible(true);
// 给关闭加事件
// 监听关闭事件
frame.addWindowListener(new WindowListener() {
@Override
public void windowOpened(WindowEvent e) {
// 窗口打开的事件
System.out.println("打开");
}
@Override //窗口关闭中
public void windowClosing(WindowEvent e) {
System.out.println("关闭中");
// 把程序结束进程
// 正常终止 0 ;非正常终止 1
System.exit(0);
}
@Override //窗口关闭
public void windowClosed(WindowEvent e) {
System.out.println("关闭");
}
@Override // 窗体图标的事件
public void windowIconified(WindowEvent e) {
}
@Override //是否被激活的事件
public void windowDeiconified(WindowEvent e) {
System.out.println("激活");
}
@Override // 没有被激活的事件
public void windowActivated(WindowEvent e) {
System.out.println("未激活");
}
@Override
public void windowDeactivated(WindowEvent e) {
}
});
// 重写它的子类 有选择型!
frame.addWindowListener(new WindowAdapter() {
@Override
public void windowDeiconified(WindowEvent e) {
super.windowDeiconified(e);
}
});
}
}
----
输出:
未激活
未激活
未激活
用户登录之后才能进入主页!用户注销后就不能进入主页了!(用过滤器来实现)
用户登录之后,向Session中放入用户的数据
进入主页的时候要判断用户是否已经登录;要求在过滤器中实现!
public void doFilter(ServletRequest req, ServletResponse resp, FilterChain filterChain) throws IOException, ServletException {
// ServletRequest HttpServletRequest 父子关系
// 获取这个 存不存在
HttpServletRequest request = (HttpServletRequest) req;
HttpServletResponse response = (HttpServletResponse) resp;
// Object user_session = request.getSession().getAttribute("USER_SESSION");
// 在过滤器阶段没有他
if (request.getSession().getAttribute("USER_SESSION")==null){
// 重定向回去
response.sendRedirect("/error.jsp");
}
// filterChain继续往下走
filterChain.doFilter(req,resp);
}
<%--登录页面 login 登录--%>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>Titletitle>
head>
<body>
<%--表单--%>
<%--action:提交的地址--%>
<h1>登录h1>
<form action="/servlet/login" method="post">
<%--user:用户--%>
<input type="text" name="username">
<input type="submit">
form>
body>
html>
package com.kuang.servlet;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
/**
* @author <[email protected]>
* @since 2021/6/22 16:27
*/
public class LoginServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// 获取前端请求的参数
String username = req.getParameter("username");
// 登录成功
if (username.equals("admin")){
// 把用户信息放到某个地方 getSession 可以在多个页面中取到 setAttribute :放 键(常量) 和 值
req.getSession().setAttribute("USER_SESSION",req.getSession().getId());
// 重定向
resp.sendRedirect("/sys/success.jsp");
// 登录失败
}else {
resp.sendRedirect("/error.jsp");
}
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
会有两种情况,要么成功,要么失败,
假设成功就进入成功的页面,
<%--主页 success :成功--%>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
Title
主页
<%--注销--%>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
Title
错误
没有权限,不是管理员
用户名错误
返回登录页面
成功了就让他注销
失败了就让他返回主页
加一个权限验证
加一个过滤器:做一些用户Servlet判断,重点是放servlet和存servlet
package com.kuang.filter;
import javax.servlet.*;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
/**
* 过滤器
*
* @author <[email protected]>
* @since 2021/6/22 18:30
*/
public class SysFilter implements Filter {
@Override
public void init(FilterConfig filterConfig) throws ServletException {
}
@Override
public void doFilter(ServletRequest req, ServletResponse resp, FilterChain filterChain) throws IOException, ServletException {
// ServletRequest HttpServletRequest 父子关系
// 获取这个 存不存在
HttpServletRequest request = (HttpServletRequest) req;
HttpServletResponse response = (HttpServletResponse) resp;
// Object user_session = request.getSession().getAttribute("USER_SESSION");
// 在过滤器阶段没有他
if (request.getSession().getAttribute("USER_SESSION")==null){
// 重定向回去
response.sendRedirect("/error.jsp");
}
// filterChain继续往下走
filterChain.doFilter(req,resp);
}
@Override
public void destroy() {
}
}
public class Constant {
// 修改统一
public static String USER_SESSION = "USER_SESSION";
}
<servlet>
<servlet-name>LoginServletservlet-name>
<servlet-class>com.kuang.servlet.LoginServletservlet-class>
servlet>
<servlet-mapping>
<servlet-name>LoginServletservlet-name>
<url-pattern>/servlet/loginurl-pattern>
servlet-mapping>
<servlet>
<servlet-name>LogoutServletservlet-name>
<servlet-class>com.kuang.servlet.LogoutServletservlet-class>
servlet>
<servlet-mapping>
<servlet-name>LogoutServletservlet-name>
<url-pattern>/servlet/logouturl-pattern>
servlet-mapping>
<filter>
<filter-name>SysFilterfilter-name>
<filter-class>com.kuang.filter.SysFilterfilter-class>
filter>
<filter-mapping>
<filter-name>SysFilterfilter-name>
<url-pattern>/sys/*url-pattern>
filter-mapping>
J:java
DB:数据库
C:Connection 连接
Java连接数据库!
连接数据库需要两个jar包
-- 创建 users 表
create table users(
id int primary key,
`name` varchar (40),
`password` varchar (40),
email varchar (60),
birthady DATE
);
-- 插入一个数据
insert into users (id,`name`,`password`,email,birthady) values
(1,'张三','123456','[email protected]' ,'2000-01-01');
insert into users (id,`name`,`password`,email,birthady) values
(2,'李四','123456','[email protected]' ,'2000-01-01');
insert into users (id,`name`,`password`,email,birthady) values
(3,'王五','123456','[email protected]' ,'2000-01-01');
-- 查询数据
select *from users
导入数据库依赖 pom.xml下
<dependencies>
<dependency>
<groupId>mysqlgroupId>
<artifactId>mysql-connector-javaartifactId>
<version>8.0.24version>
dependency>
dependencies>
IDEA中连接数据库
idea 连接 需要改时区:Asia/Shanghai
idea 连接 需要改时区:UTC
普通的
package com.kuang.test;
import com.mysql.jdbc.Driver;
import java.sql.*;
// 查询
public class TestJdbc {
public static void main(String[] args) throws ClassNotFoundException, SQLException {
// 配置信息
// 解决中文乱码 :useUnicode=true&characterEncoding=utf-8
String url = "jdbc:mysql://localhost:3306/jdbc?useUnicode=true&characterEncoding=utf-8&serverTimezone=UTC ";
String username = "root";
String password = "123456";
// 1.加载驱动
Class.forName("com.mysql.jdbc.Driver");
// 2.连接数据库,connection:代表数据库 , DriverManager:连接数据库驱动
Connection connection = DriverManager.getConnection(url, username, password);
// 3.向数据库发送SQL的对象 Statement(普通的),PreparedStatement (安全的) :CRUD,增删改查
Statement statement = connection.createStatement();
// PreparedStatement 安全的
// 4.编写SQL语句
String sql = "select * from users";
// 5.执行查询SQL statement:用来执行的对象 返回一个结果集 ResultSet
ResultSet rs = statement.executeQuery(sql);
// 判断他还有没有数据,有的话就往下遍历,遍历完就得到具体的对象
while (rs.next()){
// 从结果集里面.get某个对象,如果都不知道,get object
System.out.println("id"+rs.getObject("id"));
System.out.println("name"+rs.getObject("name"));
System.out.println("password"+rs.getObject("password"));
System.out.println("email"+rs.getObject("email"));
System.out.println("birthady"+rs.getObject("birthady"));
}
// 6.关闭连接,释放资源(一定要做)先开后关
rs.close();
statement.close();
connection.close();
}
}
安全的
防止SQL注入
package com.kuang.test;
import java.sql.*;
/**
* PreparedStatement对象
* PreparedStatement 安全的
* @author <[email protected]>
* @since 2021/6/23 20:07
*/
public class TestJdbc2 {
public static void main(String[] args) throws ClassNotFoundException, SQLException {
// 配置信息
// 解决中文乱码 :useUnicode=true&characterEncoding=utf-8
String url = "jdbc:mysql://localhost:3306/jdbc?useUnicode=true&characterEncoding=utf-8&serverTimezone=UTC ";
String username = "root";
String password = "123456";
// 1.加载驱动
Class.forName("com.mysql.jdbc.Driver");
// 2.连接数据库,connection:代表数据库 , DriverManager:连接数据库驱动
Connection connection = DriverManager.getConnection(url, username, password);
// 3.编写SQL
String sql = " insert into users(id, name, password, email, birthady) values (?,?,?,?,?); ";
// 4.预编译
PreparedStatement preparedStatement = connection.prepareStatement(sql);
// 把具体的值赋进去
// 第一个占位符? 的值赋值为1;
preparedStatement.setInt(1,4);
// 第二个占位符? 的值赋值为小明;
preparedStatement.setString(2,"小明");
// 第三个占位符? 的值赋值为123456;
preparedStatement.setString(3,"123456");
// 第四个占位符? 的值赋值为[email protected];
preparedStatement.setString(4,"[email protected]");
// 第五个占位符? 的值赋值为new Date(new java.util.Date().getTime());
// 外面的Date是sql.Date ,里面是util.Date 加一个时间戳
preparedStatement.setDate(5,new Date(new java.util.Date().getTime()));
// 5.执行SQL
int i = preparedStatement.executeUpdate();
if (i>0){
System.out.println("插入成功");
}
// 6.关闭连接,释放资源(一定要做)先开后关
preparedStatement.close();
connection.close();
}
}
事务
要么都成功,要么都失败!
特性:
ACID原则:保证数据的安全
原子性,一致性,隔离性,持久性
状态
开启事务
事务提交 commit()
事务回滚 rollback()
关闭事务
转账:
A:1000元
B:1000元
A(900) --转100--> B(1100) 在一组事务中
Junit单元测试
依赖 jar包
<dependency>
<groupId>junitgroupId>
<artifactId>junitartifactId>
<version>4.11version>
dependency>
简单使用
@Test注解只有在方法上有效,只要加了这个注解的方法,就可以直接运行!
@Test
public void test(){
System.out.println("Hello");
}
失败的时候是红色
搭建一个环境,测试一波事务
模拟转账
# 开启事务
start transaction ;
#模拟转账 修改这张表
update account set money = money-100 where name ='A';
update account set money = money+100 where name ='B';
# 回滚 回到开始事务之前的状态
rollback ;
# 提交 提交完之后,转账才会执行
commit ;
用IDEA 模拟转账
package com.kuang.test;
import org.junit.Test;
import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.SQLException;
/**
* 事务
* 模拟转账
* @author <[email protected]>
* @since 2021/6/24 17:34
*/
public class TestJdbc3 {
@Test
public void test() {
// 配置信息
// 解决中文乱码 :useUnicode=true&characterEncoding=utf-8
String url = "jdbc:mysql://localhost:3306/jdbc?useUnicode=true&characterEncoding=utf-8&serverTimezone=UTC ";
String username = "root";
String password = "123456";
// 提对象
Connection connection = null;
// 1.加载驱动
try {
Class.forName("com.mysql.jdbc.Driver");
// 2.连接数据库,connection:代表数据库 , DriverManager:连接数据库驱动
connection = DriverManager.getConnection(url, username, password);
// 3.通知数据库开启 事务 ,false(开启)
// 下边的这句话等价于 start transaction (开启事务);
connection.setAutoCommit(false);
String sql1 = "update account set money = money-100 where name ='A';";
// 执行一下
connection.prepareStatement(sql1).executeUpdate();
// 制造错误
// int i = 1/0;
String sql2 = "update account set money = money-100 where name ='B';";
connection.prepareStatement(sql2).executeUpdate();
// 提交事务
connection.commit();// 以上两条SQL都执行成功了,就提交事务!
System.out.println("提交成功");
System.out.println("success");
} catch ( Exception e) {
// 如果出现异常,就通知数据库回滚事务 connection.rollback(); :回滚!
try {
connection.rollback();
} catch (SQLException throwables) {
throwables.printStackTrace();
}
e.printStackTrace();
}finally {
// 把连接关闭掉 connection.close();
try {
connection.close();
} catch (SQLException throwables) {
throwables.printStackTrace();
}
}
}
}