💻 JavaWeb:
1. 基本概念:
1.1、前言:
web开发:
-
web,网页的意思,www.baidu.com
-
静态web
- html,css
- 提供给所有人看的数据,始终不会发生变化!
-
动态web
-
几乎所有的网站
-
每个人在不同的时间,不同的地点,看到的信息各不相同!
-
技术栈:Servlet / JSP,ASP,PHP
-
在java中,动态web资源开发的技术统称为javaWeb;
1.2、web应用程序:
web应用程序:可以提供浏览器访问的程序。
- a.html、b.html…多个web资源,这些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
- 它无法和数据库交互(数据无法持久化,用户无法交互)
- web页面无法动态更新,所有用户看到的都是同一个页面
1.4、动态web:
页面会动态展示:“Web的页面展示的效果因人而异”。
缺点:
- 假如服务器的动态web资源出现了错误,我们需要重新编写我们的后台程序,重新发布。
- 停机维护
优点:
- web页面可以动态更新,不同的用户看到不同的页面
- 可以与数据库交互 (数据持久化:用户信息,商品信息…)
2. web服务器:
2.1、技术讲解:
ASP
- 微软:国内最早流行的就是ASP;
- 在HTML中嵌入了VB的脚本,ASP+COM;
- 在ASP开发中,基本一个页面都有几千行的业务代码,页面及其混乱
- 维护成本高
- C#
- IIS
<h1>
<%
System.out.println("hello");
%>
</h1>
PHP
- PHP开发速度很快,功能很强大,跨平台,代码很简单
- 无法承载大访问量的情况下(局限性);
JSP/Servlet:
B/S:浏览器和服务器
C/S:客户端和服务器
- SUN公司主推的B/S架构
- 基于java语言(所有的大公司,或者一些开源的组件,都是用java写的)
- 可以承载三高问题带来的影响;
- 语法像ASP,ASP—>JSP,加强市场的强度;
2.2、web服务器:
服务器是一种被动的操作,用来处理用户的一些请求和给用户一些响应信息。
IIS
微软的:ASP… window中自带的
Tomcat
Tomcat 服务器是一个免费的开放源代码的Web 应用服务器,属于轻量级应用服务器,在中小型系统和并发访问用户不是很多的场合下被普遍使用,是开发和调试JSP 程序的首选。
下载tomcat:
- 安装 or 解压
- 了解配置文件及目录结构
- 这个东西的作用
3. Tomcat:
3.1、安装Tomcat:
官网:https://tomcat.apache.org/
3.2、Tomcat启动和配置:
Tomcat启动
bin文件夹中的 startup.bat
Tomcat关闭
bin文件夹中的 shutdown.bat
访问测试
http://localhost:8080/
肯能会遇到的问题:
- java环境变量没有配置:闪退
- 需要配置兼容性
- 乱码问题:配置文件中设置
服务器核心配置文件
可以配置启动的端口号 port:
- tomcat的默认端口号:8080
- mysql:3306
- http:80
- https:443
<Connector port="8080" protocol="HTTP/1.1"
connectionTimeout="20000"
redirectPort="8443" />
可以配置主机的名称 localhost:
<Host name="localhost" appBase="webapps"
unpackWARs="true" autoDeploy="true">
主机名称配置完后,需要到:C:\Windows\System32\drivers\etc\hosts 文件中设置映射关系
127.0.0.1 mtc.com
Tomcat在IDEA中乱码问题:
修改logging.properties文件
3.3、面试题:
请你谈谈网站是如何进行访问的:
-
输入一个域名:回车
-
检查本机的 C:\Windows\System32\drivers\etc\hosts 配置文件下有没有这个域名的映射
- 有:直接返回对应的ip地址,这个地址中,有我们需要访问的web程序,可以直接访问
127.0.0.1 mtc.com
- 没有:去DNS服务器上找,找到就返回,找不到就返回找不到;
3.4、发布一个web网站:
- 将自己写的网站,放到服务器(Tomcat)中指定的web应用的文件夹(webapps)下,就可以访问了
网站应该有的结构:
--webapps : Tomcat 服务器的web目录
--ROOT
--mtc : 网站的目录名
- 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、两个时代:
- http 1.0
- HTTP/1.0:客户端与web服务器连接后,只能获取一个web资源,断开连接
- http 2.0
- HTTP/1.1:客户端与web服务器连接后,可以获取多个web资源。
4.3、Http请求:
- 客户端—>发请求—>服务器
百度:
Request URL: https://www.baidu.com/ 请求地址
Request Method: GET get方法/post 方法
Status Code: 200 OK 状态码
Referrer Policy: no-referrer-when-downgrade
Accept:text/html
Accept-Encoding: gzip, deflate, br
Accept-Language: zh-CN,zh;q=0.9
Connection: keep-alive
1. 请求行
- 请求行中的请求方式:GET
- 请求方式:GET/POST
- GET:请求能够携带的参数比较少,大小有限制,会在浏览器的URL地址栏显示数据内容,不安全,但高效。
- POST: 请求能够携带的参数没有限制,大小没有限制,不会在浏览器的URL地址栏显示数据内容,安全,但不高效。
2. 请求头
Accept:告诉浏览器,他所支持的数据类型
Accept-Encoding: 支持哪种编码格式 GBK UTF-8 GB2312
Accept-Language: 告诉浏览器,他的语言环境
Cache-Control: 缓存控制
Connection: 告诉浏览器,请求完成是断开,还是保持连接
HOST: 主机
4.4、Http响应:
- 服务器—>响应—>客户端
百度:
Cache-Control: private // 缓存控制
Connection: keep-alive // 连接
Content-Encoding: gzip // 编码
Content-Type: text/html;charset=utf-8 // 类型
1.响应体:
Accept:告诉浏览器,他所支持的数据类型
Accept-Encoding: 支持哪种编码格式 GBK UTF-8 GB2312
Accept-Language: 告诉浏览器,他的语言环境
Cache-Control: 缓存控制
Connection: 告诉浏览器,请求完成是断开,还是保持连接
HOST: 主机
Refresh: 告诉客户端,多久刷新一次
Location: 让网页重新定位
2. 响应状态码:
200:请求响应成功
3xx:请求重定向
- 重定向:你重新到我给你的新位置去
4xx:找不到资源 404
5xx:服务器代码错误 500 502:网关错误
4.5、常见面试题
当你的浏览器中地址栏输入地址,并回车的一瞬间到页面能够展示回来,经历了什么?
5. Maven:
我为什么要学习这个技术?
-
在javaweb开发中,需要使用大量的jar包,我们需要手动导入;
-
如何能够让一个东西自动帮我们导入和配置这些jar包。
由此,Maven诞生了!
5.1、Maven项目架构管理工具:
我们目前用它就是方便导入jar包的。
Maven的核心思想:约定大于配置
- 有约束,不要违反
Maven会规定好我们该如何去编写java代码,必须要按照这个规范来。
5.2、下载安装Maven:
官网:https://maven.apache.org/
下载完成后,解压即可;
5.3、配置环境变量:
在我们的系统环境变量中:
配置如下:
- MAVEN_HOME maven的目录
- 在系统的path中配置 %MAVEN_HOME%\bin
测试Maven是否安装成功,保证必须配置完毕!
5.4、修改一下配置文件:
阿里云镜像:mirrors
- 作用:加速我们的下载
- 国内建议使用阿里云的镜像
- 在maven配置文件settings.xml文件中 标签内加上:
<mirror>
<id>nexus-aliyun</id>
<mirrorOf>*,!jeecg,!jeecg-snapshots</mirrorOf>
<name>Nexus aliyun</name>
<url>http://maven.aliyun.com/nexus/content/groups/public</url>
</mirror>
5.5、本地仓库:
在本地的仓库,远程仓库;
建立一个本地仓库 localRepository
- 新建一个maven-repo文件夹
- 在settings.xml文件中加上
<localRepository>D:\java后端\javaweb\apache-maven-3.6.3\maven-repo</localRepository>
5.6、在IDEA中使用Maven:
- 启动IDEA
- 创建一个MavenWeb项目
这个目录只有在web应用下才有
到这边maven项目就创建完成了。
5.7、创建一个普通的Maven项目:
选择Maven ,直接点下一步
5.8、在IDEA中标记文件夹功能:
5.9、在IDEA中配置Tomcat:
解决警告问题:
为什么会有这个问题,我们访问一个网站,需要指定一个文件夹的名字;
启动Tomcat
5.10、pom文件:
pom.xml是Maven的核心配置文件
maven由于约定大于配置,我们之后可能会遇到我们写的配置文件,无法被导出或者生效的问题
解决方案
<!--在build中配置resource,来防止我们资源导出失败问题-->
<build>
<resources>
<resource>
<directory>src/main/java</directory>
<includes>
<include>**/*.properties</include>
<include>**/*.xml</include>
</includes>
<filtering>false</filtering>
</resource>
<resource>
<directory>src/main/resources</directory>
<includes>
<include>**/*.properties</include>
<include>**/*.xml</include>
</includes>
<filtering>false</filtering>
</resource>
</resources>
</build>
5.11、IDEA操作:
5.12、解决遇到的问题:
-
Maven 3.6.2报错Unable to import maven project See for details
解决方法:降级为3.6.1
-
Tomcat闪退:
-
IDEA中每次都要重复配置Maven
在IDEA中的全局默认配置中去配置
- IDEA中maven默认web项目中的web.xml版本问题:
解决:替换为webapp4.0版本和tomcat一致:
<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee
http://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsd"
version="4.0"
metadata-complete="true">
</web-app>
6. Servlet:
6.1、Servlet简介:
- Servlet 就是sun公司开发动态web的一门技术
- Sun在这些API中提供了一个接口叫做:Servlet,如果你想开发一个Servlet程序,只需要完成两个小步骤
- 编写一个类,实现Servlet接口
- 把开发好的java类部署到web服务器中。
把实现了Servlet接口的java程序叫做,Servlet
6.2、HelloServlet:
Servlet接口在SUN公司有两个默认的实现类:HttpServlet GenericServlet
- 构建一个普通Maven项目,删掉里面的src目录,在这个项目里面建立Moudel,这个空的工程就是Maven的主工程
- 导入servlet和jsp的jar包
- 在项目中新建一个maven-webapp的Moudel
- 关于Maven父子工程的理解:
父项目中会有:
<modules>
<module>servlet-01</module>
</modules>
-
Maven环境优化:
-
修改web.xml为最新的
-
将Maven的结构搭建完整
-
-
编写一个Servlet程序:
- 创建一个jave普通类
- 实现Servlet接口,这里我们直接继承HttpServlet。
public class HelloServlet extends HttpServlet {
// 由于get或者post只是请求实现的不同的方式,可以相互调用
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
resp.setContentType("text/html");
resp.setCharacterEncoding("utf8");
//响应流
PrintWriter writer = resp.getWriter();
writer.print("hello Servlet");
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req,resp);
}
}
-
在web.xml中,编写Servlet的映射:
为什么需要映射:我们写的是java程序,需要通过浏览器访问,而浏览器需要连接web服务器,所有我们需要在web服务中注册我们写的Servlet,还需要给他一个浏览器能够访问的路径。
<!--注册servlet-->
<servlet>
<servlet-name>hello</servlet-name>
<servlet-class>com.ma.servlet.HelloServlet</servlet-class>
</servlet>
<!--Servlet的请求路径-->
<servlet-mapping>
<servlet-name>hello</servlet-name>
<url-pattern>/hello</url-pattern>
</servlet-mapping>
-
配置Tomcat:
注意:配置项目发布的路径。
-
启动测试:成功!
6.3、Servlet原理:
Servlet是有Web服务器调用,web服务器在收到浏览器请求之后,会
6.4、Mapping:
- 一个Servlet可以指定一个映射路径
<servlet-mapping>
<servlet-name>hello</servlet-name>
<url-pattern>/hello</url-pattern>
</servlet-mapping>
- 一个Servlet可以指定多个映射路径
<servlet-mapping>
<servlet-name>hello</servlet-name>
<url-pattern>/hello</url-pattern>
</servlet-mapping>
<servlet-mapping>
<servlet-name>hello</servlet-name>
<url-pattern>/hello1</url-pattern>
</servlet-mapping>
- 一个Servlet可以指定通用映射路径
<servlet-mapping>
<servlet-name>hello</servlet-name>
<url-pattern>/hello/*</url-pattern>
</servlet-mapping>
- 默认请求路径:
<servlet-mapping>
<servlet-name>hello</servlet-name>
<url-pattern>/*</url-pattern>
</servlet-mapping>
- 指定一些后缀或前缀:
<!--
可以自定义后缀实现请求映射
注意点:*前面不能加映射的路径
-->
<servlet-mapping>
<servlet-name>hello</servlet-name>
<url-pattern>*.ma</url-pattern>
</servlet-mapping>
-
优先级问题:
指定了固有的映射路径,优先级最高,如果找不到就会走默认的处理请求路径(/*)
6.5、ServletContext:
web容器在启动的时候,它会为每一个web程序都创建一个对应的ServletContext对象,它代表了当前的web应用;
1. 共享数据
我在这个Servlet中保存的数据,可以在另一个Servlet中拿到:
this.getServletContext().setAttribute(key,value); // 设置属性
this.getServletContext().getAttribute(key); // 获取属性
public class HelloServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//this.getInitParameter(); 初始化参数
//this.getServletConfig(); Servlet配置
//this.getServletContext(); Servlet上下文
resp.setContentType("text/html");
resp.setCharacterEncoding("utf-8");
ServletContext servletContext = this.getServletContext();
String username="小马"; //数据
// 将一个数据保存到ServletContext中 名字为:username,值:username
servletContext.setAttribute("username",username);
System.out.println("hello ma");
}
}
public class GetServlet extends HttpServlet {
@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);
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee
http://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsd"
version="4.0"
metadata-complete="true">
<servlet>
<servlet-name>hello</servlet-name>
<servlet-class>com.ma.servlet.HelloServlet</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>hello</servlet-name>
<url-pattern>/hello</url-pattern>
</servlet-mapping>
<servlet>
<servlet-name>getc</servlet-name>
<servlet-class>com.ma.servlet.GetServlet</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>getc</servlet-name>
<url-pattern>/getc</url-pattern>
</servlet-mapping>
</web-app>
2. 获取初始化参数:
this.getServletContext().getInitParameter(str);
<!--配置一些web应用初始化参数-->
<context-param>
<param-name>url</param-name>
<param-value>jdbc:mysql://localhost:3306/mybatis</param-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);
}
}
<servlet>
<servlet-name>getc</servlet-name>
<servlet-class>com.ma.servlet.GetServlet</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>getc</servlet-name>
<url-pattern>/getc</url-pattern>
</servlet-mapping>
3. 请求转发(307):
访问http://localhost:8080/s2/sd4,获得http://localhost:8080/s2/getc中信息
注意:/代表当前的web应用
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
ServletContext servletContext = this.getServletContext();
System.out.println("进入servletdemo04");
// RequestDispatcher requestDispatcher = servletContext.getRequestDispatcher("/getc"); //转发的请求路径
//requestDispatcher.forward(req,resp); // 调用forword实现请求转发
servletContext.getRequestDispatcher("/getc").forward(req,resp);
}
<servlet>
<servlet-name>sd4</servlet-name>
<servlet-class>com.ma.servlet.ServletDemo04</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>sd4</servlet-name>
<url-pattern>/sd4</url-pattern>
</servlet-mapping>
4. 读取资源文件:
Properties类
- 在java目录下新建properties
- 在resource目录下新建properties
发现:都被打包到同一个路径下:classes,我们俗称这个路径为classpath
思路:需要一个文件流:
public class ServletDemo05 extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
resp.setContentType("text/html");
resp.setCharacterEncoding("utf-8");
InputStream resourceAsStream = this.getServletContext().getResourceAsStream("/WEB-INF/classes/db.properties");
Properties properties = new Properties();
properties.load(resourceAsStream);
String user=properties.getProperty("username");
String pwd= properties.getProperty("password");
resp.getWriter().print(user+":"+pwd);
if(resourceAsStream!=null){
resourceAsStream.close();
}
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
<servlet>
<servlet-name>sd5</servlet-name>
<servlet-class>com.ma.servlet.ServletDemo05</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>sd5</servlet-name>
<url-pattern>/sd5</url-pattern>
</servlet-mapping>
访问测试ok
6.6. HttpServletRespone:
web服务器接收到客户端的http请求,针对这个请求,分别会创建一个代表请求的HttpServletRequest对象和代表响应的HttpServletRespone对象:
- 如果要获取客户端请求过来的参数:找HttpServletRequest
- 如果要给客户端响应一些信息,找HttpServletRespone
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 setBufferSize(int 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);
void setStatus(int var1);
2. 下载文件:
- 向浏览器输出消息
- 下载文件
- 要获取下载文件的路径
- 下载的文件是什么
- 设置想办法让浏览器能够支持下载我们需要的东西
- 获取下载文件的输入流
- 创建缓冲区
- 获取OutputStream对象
- 将FileOutputStream流写入到buffer缓冲区
- 使用OutputStream将缓冲区中的数据输出到客户端
public class FileServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// 1 要获取下载文件的路径
String realPath ="D:\\IDEA项目\\javaweb--02-servlet\\response\\src\\main\\resources\\小汐.jpg";
System.out.println("下载文件的路径:"+realPath);
// 2 下载的文件是什么(文件名)
String fileName=realPath.substring(realPath.lastIndexOf("\\")+1);
// 3 设置想办法让浏览器能够支持下载我们需要的东西 中文文件名URLEncoder.encode(fileName,"UTF-8")编码,否则有可能乱码
resp.setHeader("Content-disposition","attachment;filename="+ URLEncoder.encode(fileName,"UTF-8"));
// 4 获取下载文件的输入流
FileInputStream fileInputStream = new FileInputStream(realPath);
// 5 创建缓冲区
int i=0;
byte[]buffer=new byte[1024];
// 6 获取OutputStream对象
ServletOutputStream outputStream = resp.getOutputStream();
// 7 将FileOutputStream流写入到buffer缓冲区
while ((i=fileInputStream.read(buffer))!=-1){
outputStream.write(buffer,0,i);
}
outputStream.flush();
outputStream.close();
fileInputStream.close();
// 8 使用OutputStream将缓冲区中的数据输出到客户端
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
<servlet>
<servlet-name>filedown</servlet-name>
<servlet-class>com.ma.servlet.FileServlet</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>filedown</servlet-name>
<url-pattern>/filedown</url-pattern>
</servlet-mapping>
3. 验证码实现:
验证码怎么来的:
- 前端实现:
- 后端实现,需要用到java的图片类,生成一个图片
ublic class imageServlet extends HttpServlet {
@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 graphics = (Graphics2D) bufferedImage.getGraphics(); //笔
//设置图片的背景颜色
graphics.setColor(Color.white);
graphics.fillRect(0,0,80,20);
//给图片写数据
graphics.setColor(Color.BLUE);
graphics.setFont(new Font(null,Font.BOLD,20));
graphics.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(9999999)+"";
StringBuffer sb = new StringBuffer();
for (int i = 0; i < 7-num.length(); i++) {
sb.append("0");
}
// 若不是7位 ,用0 填充
num=sb.toString()+num;
return num;
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
<servlet>
<servlet-name>imageServlet</servlet-name>
<servlet-class>com.ma.servlet.imageServlet</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>imageServlet</servlet-name>
<url-pattern>/image</url-pattern>
</servlet-mapping>
4. Response重定向 (302):
一个web资源收到客户端请求后,他会通知客户端去访问另一个web资源,这个过程叫重定向。
常见场景:
- 用户登录
void sendRedirect(String varl) throws IOException
测试:
public class RedirectServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
resp.sendRedirect("/response_war/image");
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
面试题:请你聊聊重定向和请求转发的区别:
相同点:
- 页面都会实现跳转
不同点:
- 请求转发的时候,url不会发生变化 307
- 重定向时候,url地址栏会发生变化 302
5. jsp和servlet使用:
- index页面:
<%@page language="java" contentType="text/html;charset=UTF-8" pageEncoding="UTF-8" %>
<html>
<body>
<h2>Hello World!</h2>
<%--这里提交的路径,需要寻找到项目的类路径--%>
<%--${pageContext.request.contextPath} 代表当前的项目--%>
<form action="${pageContext.request.contextPath}/login" method="get">
用户名:<input type="text" name="username"><br>
密码: <input type="password" name="password"> <br>
<input type="submit">
</form>
</body>
</html>
- 登录后,重定向的页面:
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>Title</title>
</head>
<body>
<h1>success</h1>
</body>
</html>
- java代码:
public class RequestTest extends HttpServlet {
@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("/response_war/success.jsp");
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
- 映射:
<servlet>
<servlet-name>request</servlet-name>
<servlet-class>com.ma.servlet.RequestTest</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>request</servlet-name>
<url-pattern>/login</url-pattern>
</servlet-mapping>
6.7. HttpServletRequest:
HttpServletRequest代表客户端的请求,用户通过Http协议访问服务器,Http请求中的所有信息会被封装到HttpServletRequest,通过这个HttpServletRequest的方法,可以获得客户端的所有信息。
1. 获取前端传递的参数,请求转发:
重点方法:
req.getParameter(String s);
req.getParameterValues(String s);
演示:
- index.jsp
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>登录</title>
</head>
<body>
<h1>登录</h1>
<div style="text-align: center">
<%--这里表单表示的信息: 以post方式提交表单,提交到我们的login请求--%>
<form action="${pageContext.request.contextPath}/login" method="get">
用户名:<input type="text" name="username"> <br>
密码:<input type="password" name="password"> <br>
爱好:
<input type="checkbox" name="hobbies" value="代码">代码
<input type="checkbox" name="hobbies" value="跑步">跑步
<input type="checkbox" name="hobbies" value="游戏">游戏
<input type="checkbox" name="hobbies" value="女孩" >女孩
<br>
<input type="submit">
</form>
</div>
</body>
</html>
- success.jsp:
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>登录成功</title>
</head>
<body>
<h1>登录成功</h1>
</body>
</html>
- LoginServlet.java:
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[] hobbies = req.getParameterValues("hobbies");
System.out.println("=========");
System.out.println(username+":"+password);
System.out.println(Arrays.toString(hobbies));
System.out.println("==========");
// 通过请求转发
// 这里的 / 代表当前的web应用。
req.getRequestDispatcher("/success.jsp").forward(req,resp);
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
- web.xml:
<servlet>
<servlet-name>LoginServlet</servlet-name>
<servlet-class>com.ma.servlet.LoginServlet</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>LoginServlet</servlet-name>
<url-pattern>/login</url-pattern>
</servlet-mapping>
7. Cookie和Seeion:
7.1、会话:
会话:用户打开一个浏览器,点击了很多超链接,访问多个web资源,关闭浏览器,这个过程,可以称之为会话。
有状态会话:如:一个人来过南京,下次再来,我知道他曾经来过,称之为有状态的会话;
一个网站,怎么证明你来过?
客户端 ,服务端
- 服务端给客户端一个信件,客户端下次访问服务端带上信件就可以了;cookie
- 服务器登记你来过了,下次你来的时候我来匹配你;
7.2、保存会话的两种技术:
cookie
- 客户端技术(响应,请求)
session
- 服务器技术,利用这个技术,可以保存用户的会话信息,我们可以把信息或数据放在Session中!
常见场景:网站登录之后,你下次不用再登录了,第二次访问直接就上去了!
7.3、Cookie:
- 从请求中拿到cookie信息
- 服务器响应给客户端cookie
Cookie[] cookies = req.getCookies(); //获得Cookie
cookie.getName(); // 获得cookie中的key
cookie.getValue(); // 获取cookie中的值
new Cookie(key,value); // 新建一个cookie
cookie.setMaxAge(24*60*60); // 设置cookie的有效期
resp.addCookie(cookie); // 响应给客户端一个cookie
cookie:一般会保存在本地的 用户目录下 appdata;
一个网站的cookie是否存在上限!
- 一个Cookie只能保存一个信息
- 一个web站点可以给浏览器发送多个cookie,每个站点最多存放20 个cookie
- Cookie的大小有限制4kb
- 浏览器上限300个cookie
删除Cookie:
- 不设置有效期,关闭浏览器,自动失效;
- 设置有效期时间为0,如:
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// 创建一个cookie 名字必须要和要删除的名字一致
Cookie cookie = new Cookie("lastLoginTime",System.currentTimeMillis()+"");
// 将cookie 有效期设置为0 立马过期
cookie.setMaxAge(0);
resp.addCookie(cookie);
}
具体代码实现:
// 保存用户上一次访问的时间
public class CookieDemo01 extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// 服务器告诉你来的时间,把这个时间封装成一个信件,你下次再来我就知道你来了
// 解决中文乱码
req.setCharacterEncoding("UTF-16");
resp.setCharacterEncoding("UTF-16");
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.write(date.toLocaleString());
}
}
}else{
out.write("这是你第一次访问");
}
// 服务器给客户端响应一个cookie
Cookie cookie = new Cookie("lastLoginTime",System.currentTimeMillis()+"");
//cookie 有效期为一天
cookie.setMaxAge(24*60*60);
resp.addCookie(cookie);
}
@Override
protected void doPut(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
<servlet>
<servlet-name>CookieDemo01</servlet-name>
<servlet-class>com.ma.servlet.CookieDemo01</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>CookieDemo01</servlet-name>
<url-pattern>/login</url-pattern>
</servlet-mapping>
1. 关于Cookie中的中文乱码问题:
// 编码
Cookie cookie = new Cookie("name", URLEncoder.encode("小马同学","utf-8"));
// 解码
out.write( URLDecoder.decode(cookie.getValue(),"utf-8"));
7.4、Session(重点):
1. 什么是Session:
- 服务器会给每一个用户(浏览器)创建一个Session对象
- 一个Session独占一个浏览器,只要浏览器没有关闭,这个Session就存在;
- 用户登录之后,整个网站他都可以访问!—>保存用户的信息,保存购物车的信息
2. Session和Cookie的区别:
- Cookie是把用户的数据写给用户的浏览器,浏览器保存 (可以保存多个)。
- Session把用户的数据写到用户独占的Session中,服务器端保存(保存重要的信息,减少服务器资源的浪费)
- Session 对象由服务器创建
3. 使用场景:
- 保存一个登陆用户的信息
- 购物车信息
- 在整个网站中,经常会使用的数据,我们将他保存在Session中
4. 使用Session:
- 设置Session:
public class SessionDemo01 extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//解决乱码问题
req.setCharacterEncoding("UTF-16");
resp.setCharacterEncoding("UTF-16");
resp.setContentType("text/html;charset=utf-8");
//得到Session
HttpSession session = req.getSession();
// 给Session 中存放东西
session.setAttribute("name",new Person("小马",1));
// 获取Session的ID
String sessionId = session.getId();
// 判断Session是不是新创建的
if (session.isNew()){
resp.getWriter().write("Session创建成功,ID:"+sessionId);
}else{
resp.getWriter().write("Session已经在服务器中存在了,ID:"+sessionId);
}
// Session创建的时候做了什么事情
// Cookie cookie = new Cookie("JSESSIONID","sessionId");
// resp.addCookie(cookie);
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
- 获取Session:
public class SessionDemo02 extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//解决乱码问题
req.setCharacterEncoding("UTF-16");
resp.setCharacterEncoding("UTF-16");
resp.setContentType("text/html;charset=utf-8");
//得到Session
HttpSession session = req.getSession();
Person person = (Person) session.getAttribute("name");
System.out.println(person.toString());
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
- 删除Session:
- 方法一:手动注销,主要使用在用户的注销功能:
public class SessionDemo03 extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
HttpSession session = req.getSession();
// 删除Session
session.removeAttribute("name");
// 手动注销Session
session.invalidate();
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
- 方法二:会话自动过期,在web.xml中设置一个注销时间,当用户这段时间没来访问,就会删除:
<!--设置Session默认的失效时间-->
<session-config>
<!--15 分钟后,Session自动失效,以分钟为单位-->
<session-timeout>1</session-timeout>
</session-config>
8. JSP:
8.1、什么是JSP:
Java Server Pages:Java服务器端页面,也和Servlet一样,用于动态Web技术。
最大的特点:
- 写JSP就像在写HTML
- 区别:
- HTML只给用户提供静态的数据
- JSP页面中可以嵌入Java代码,为用户提供动态数据;
8.2、JSP原理:
JSP怎么执行的:
-
代码层面没有任何问题
-
服务器内部工作
tomcat中有一个work目录
IDEA中使用Tomcat的话,会在IDEA的Tomcat中生成一个work目录
电脑地址:
C:\Users\user\.IntelliJIdea2019.2\system\tomcat\Unnamed_javaweb-session-cookie\work\Catalina\localhost\ROOT\org\apache\jsp
发现页面转化为java程序:
浏览器向服务器发送请求,不管是访问什么资源,其实都是在访问Servlet!
JSP最终也会被转化成一个java类
JSP本质上就是一个Servlet
// 初始化
public void _jspInit() {
}
// 销毁
public void _jspDestroy() {
}
// JSPService
public void _jspService(final javax.servlet.http.HttpServletRequest request, final javax.servlet.http.HttpServletResponse response)
throws java.io.IOException, javax.servlet.ServletException{}
- 判断请求
- 内置一些对象
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; // 响应
- 输出页面前增加的代码:
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;
- 以上的这些对象我们可以在JSP页面中直接使用!
在JSP页面中:
只要是Java代码,就会原封不动的输出,
如果是HTML代码,就会被转化为:
out.write("<html>\r\n")
这样的格式输出到前端
8.3、JSP的基础语法:
需要的依赖:
<!--Servlet依赖-->
<dependency>
<groupId>javax.servlet</groupId>
<artifactId>servlet-api</artifactId>
<version>2.5</version>
</dependency>
<!--JSP依赖-->
<dependency>
<groupId>javax.servlet.jsp</groupId>
<artifactId>javax.servlet.jsp-api</artifactId>
<version>2.3.3</version>
</dependency>
<!--JSTL依赖-->
<dependency>
<groupId>javax.servlet.jsp.jstl</groupId>
<artifactId>jstl-api</artifactId>
<version>1.2</version>
</dependency>
<!--standard标签库-->
<dependency>
<groupId>taglibs</groupId>
<artifactId>standard</artifactId>
<version>1.1.2</version>
</dependency>
任何语言都有自己的语法,Java中有,JSP作为java技术的一种应用,他拥有自己扩充的语法(了解,知道即可),java所有语法都支持
1. JSP表达式:
<%--JSP表达式
作用:用来将程序输出到客户端
<%= 变量或表达式%>
--%>
<%= new java.util.Date()%>
2. JSP脚本片段:
<%--jsp 脚本片段--%>
<%
int sum=0;
for (int i = 0; i <=100 ; i++) {
sum+=i;
}
out.println("<h1>Sum="+sum+"</h1>");
%>
3. 脚本片段的再实现:
<%
int x=0;
out.println(x);
%>
<p>这是一个jsp文档</p>
<%
int y=20;
out.println(y);
%>
<%--在代码嵌入HTML元素--%>
<%
for (int i = 0; i <5 ; i++) {
%>
<h1>hello 小马 <%=i%></h1>
<%
}
%>
4. JSP声明:
<%!
static {
System.out.println("Loading Servlet");
}
private int globalVar=0;
public void ma(){
System.out.println("进入了方法ma");
}
%>
区别:
JSP声明:会被编译到JSP生成的Java的类中,其他的,就会被生成到_jspService中
在JSP中,嵌入代码。
<%%> <%--片段--%>
<%=%> <%--表达式输出值--%>
<%!%> <%--全局--%>
JSP的注释不会在客户端显示,HTML就会。
8.4、 JSP指令:
- 500错误页面显示:
- 方法一:
- 编写一个500错误页面显示的内容:500.jsp。
- 再向需要访问的页面,申明发送500错误时转到的页面
<%--500错误页面--%>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>Title</title>
</head>
<body>
<img src="../img/500.png" alt="">
</body>
</html>
<%--导入错误页面--%>
<%--<%@page errorPage="error/500.jsp" %>--%>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>Title</title>
</head>
<body>
<%
int x=1/0;
%>
</body>
</html>
- 方法二,在web.xml里申明:
<error-page>
<error-code>500</error-code>
<location>/error/500.jsp</location>
</error-page>
- 提取公共页面:
- 新建一个公共的头部 header.jsp
- 在页面中导入公共的header.jsp
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>Title</title>
</head>
<body>
<h1>我是header</h1>
</body>
</html>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>Title</title>
</head>
<body>
<%--@include 会将两个页面合二为一--%>
<%@include file="common/header.jsp"%>
<h1>网页主体</h1>
<%@include file="common/footer.jsp"%>
<hr>
<%--jsp标签--%>
<%--
jsp:include :拼接页面,本质还是三个
--%>
<jsp:include page="common/header.jsp"/>
<h1>网页主体</h1>
<jsp:include page="common/footer.jsp"/>
</body>
</html>
8.5、9大内置对象:
- PageContext 存东西
- Request 存东西
- Response
- Session 存东西
- Application 【ServletContext】存东西
- config 【ServletConfig】
- out
- page 不用了解
- exception
1. 作用域范围:
从底层到高层(作用域) page–>request–>session–>application
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>Title</title>
</head>
<body>
<%--内置对象--%>
<%
pageContext.setAttribute("name1","小马1号"); // 保存的数据只在一个页面中有效
request.setAttribute("name2","小马2号"); // 保存的数据在一次请求中有效,请求转发会携带这个参数
session.setAttribute("name3","小马3号");// 保存的数据在一次会话中有效,从打开浏览器到关闭浏览器
application.setAttribute("name4","小马4号");// 保存的数据在服务器中有效,从打开服务器到关闭服务器
%>
<%--通过pageContext取出我们保存的值--%>
<%
// 从pageContext取出,我们通过寻找的方式来
// 从底层到高层(作用域) page-->request-->session-->application
String name1 = (String)pageContext.findAttribute("name1");
String name2= (String) pageContext.findAttribute("name2");
String name3= (String) pageContext.findAttribute("name3");
String name4= (String) pageContext.findAttribute("name4");
String name5= (String) pageContext.findAttribute("name5"); // 不存在
%>
<%--使用EL表达式输出--%>
<h1>取出的值为:</h1>
<h3>${name1}</h3>
<h3>${name2}</h3>
<h3>${name3}</h3>
<h3>${name4}</h3>
<h3>${name5}</h3> <%--不显示--%>
<h3><%=name5%></h3> <%--null--%>
</body>
</html>
request:客户端向服务器发送请求,产生的数据,用户看完就没用了,比如:新闻,用户看完没用
session:客户端向服务器发送请求,产生的数据,用户看完一会还有用,比如:购物车
application:客户端向服务器发送请求,产生的数据,一个用户用完了,其他用户还可能使用,比如:聊天数据,统计人数。
2. 两种请求转发方式:
<%
// 两种转发方式
pageContext.forward("/index.jsp");
request.getRequestDispatcher("/index.jsp").forward(request,response);
%>
8.6、JSP标签、JSTL标签、EL标签:
EL表达式:${}
需要的jar包:
<!--JSTL依赖-->
<dependency>
<groupId>javax.servlet.jsp.jstl</groupId>
<artifactId>jstl-api</artifactId>
<version>1.2</version>
</dependency>
<!--standard标签库-->
<dependency>
<groupId>taglibs</groupId>
<artifactId>standard</artifactId>
<version>1.1.2</version>
</dependency>
- 获取数据
- 执行运算
- 获取web开发的常用对象
- 调用java方法
- JSP标签:
<%--<jsp:include page=""/>--%>
<%--
http://localhost:8080/jsptag.jsp?name=xiaoma&age=23
--%>
<jsp:forward page="/jsptag2.jsp">
<jsp:param name="name" value="xiaoma"/>
<jsp:param name="age" value="23"/>
</jsp:forward>
-
JSTL表达式:
JSTL标签库的使用就是为了弥补HTML标签的不足;它自定义了许多标签,可以供我们使用,标签的功能和java代码一样
- 核心标签(掌握部分)
- 引入核心标签库:
<%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %>
- 核心标签(掌握部分)
主要在Tomcat也需要引入jstl的包(jstl,standard),否则会报:JSTL解析错误
- c:if标签:
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<%--引入JSTL核心标签库,我们才能使用JSTL标签 core--%>
<%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %>
<html>
<head>
<title>Title</title>
</head>
<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:if>
<c:out value="${isAdmin}"></c:out>
</body>
</html>
- c:choose标签
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<%--引入JSTL核心标签库,我们才能使用JSTL标签 core--%>
<%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %>
<html>
<head>
<title>Title</title>
</head>
<body>
<%--定义一个变量score 它的值为85--%>
<c:set var="score" value="85"/>
<c:choose>
<c:when test="${score>=90}">
你的成绩为优秀
</c:when>
<c:when test="${score>=80}">
你的成绩为良好
</c:when>
<c:when test="${score>=70}">
你的成绩为一般
</c:when>
</c:choose>
</body>
</html>
- c:foreach标签:
<%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %>
<%@ page import="java.util.ArrayList" %>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<%--引入JSTL核心标签库,我们才能使用JSTL标签 core--%>
<%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %>
<html>
<head>
<title>Title</title>
</head>
<body>
<%
ArrayList<String> people=new ArrayList<String>();
people.add(0,"张三");
people.add(1,"李四");
people.add(2,"王五");
people.add(3,"赵六");
people.add(4,"田七");
request.setAttribute("list",people);
%>
<%--
var 每一次遍历出来的变量
items 要变量的对象
begin 哪里开始
end 到哪里
step 步长
--%>
<c:forEach var="people" items="${list}">
<c:out value="${people}"/> <br>
</c:forEach>
<hr>
<c:forEach begin="1" end="3" step="2" var="people" items="${list}">
<c:out value="${people}"/> <br>
</c:forEach>
</body>
</html>
- 格式化标签
- SQL标签
- XML标签
9. JavaBean:
实体类
JavaBean有特殊的写法:
- 必须要有一个无参构造
- 属性必须私有化
- 必须有对应的get和set方法
一般用来和数据库的字段做映射 ORM
ORM:对象关系映射
- 表–>类
- 字段–>类的属性
- 行记录–>对象
如:people表
id | name | age | address |
---|---|---|---|
1 | 小马1号 | 3 | 江苏 |
2 | 小马2号 | 18 | 南京 |
3 | 小马3号 | 100 | 浦口 |
class People{
private int id;
private String name;
private int age;
private String address;
}
class A{
new People(1,"小马1号",3,"江苏");
new People(2,"小马2号",18,"南京");
new People(3,"小马3号",100,"浦口");
}
jsp设置获取信息:
jsp:useBean id="people" class="com.ma.pojo.People" scope="page"/>
<jsp:setProperty name="people" property="id" value="1"/>
<jsp:setProperty name="people" property="name" value="小马1号"/>
<jsp:setProperty name="people" property="age" value="1"/>
<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--->CRUD--->数据库
弊端:程序十分臃肿,不利于维护 servlet的代码中:处理请求,响应,视图跳转,处理JDBC、处理业务代码,处理逻辑代码
架构:没有什么是加一层解决不了的
10.2、MVC三层架构:
Model:
- 业务处理:业务逻辑(Service)
- 数据库持久层:CRUD(Dao)
View:
- 展示数据
- 提供链接,发起Servlet请求(a,form,img)
Controller:
- 接收用户请求:(req:请求参数,Session信息…)
- 交给业务层处理对应的代码
- 控制视图的跳转
登录--->接收用户的请求--->处理用户的请求(获取用户登录的参数,username,password)--->交给业务层处理登录业务(判断用户名密码是否正确:事务)--->Dao层查询用户名和密码是否正确--->数据库
11、Filter(重点):
Filter:过滤器,用来过滤网站的数据;
- 处理中文乱码
- 登录验证
Filter开发步骤:
-
导包
-
编写过滤器:
- 导包不要错:
- 编写java代码:实现Filter接口,重写对应的方法
public class CharacterEncodingFilter implements Filter {
// 初始化
public void init(FilterConfig filterConfig) throws ServletException {
// 初始化:web服务器启动,就已经初始化了,随时等待过滤对象出现
System.out.println("CharacterEncodingFilter 初始化");
}
// chain :链
/*
1. 过滤中的所有代码,在过滤特定请求的时候都会执行
2. 必须要让过滤器继续同行
* */
public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
request.setCharacterEncoding("utf-16");
response.setCharacterEncoding("utf-16");
response.setContentType("text/html;charset=UTF-16");
System.out.println("CharacterEncodingFilter执行前");
chain.doFilter(request,response); // 让我们的请求继续走,如果不写,程序到这里就被拦截停止了
System.out.println("CharacterEncodingFilter执行后");
}
// 销毁
public void destroy() {
// 销毁:web服务器关闭的时候,过滤器会销毁
System.out.println("CharacterEncodingFilter 销毁");
}
}
- 设置web.xml:
<filter>
<filter-name>CharacterEncodingFilter</filter-name>
<filter-class>com.ma.filter.CharacterEncodingFilter</filter-class>
</filter>
<filter-mapping>
<filter-name>CharacterEncodingFilter</filter-name>
<!--只要是/servlet下的任何请求,都会经过这个过滤器-->
<url-pattern>/servlet/*</url-pattern>
</filter-mapping>
12. 监听器:
实现一个监听器的接口(N种)
- 编写一个监听器,实现监听器的接口:
/*统计网站在线人数 , 统计Session*/
public class OnlineCountListener implements HttpSessionListener {
// 创建session监听 看你的一举一动
// 一旦创建Session就会触发一次这个时间
public void sessionCreated(HttpSessionEvent se) {
ServletContext servletContext = se.getSession().getServletContext();
System.out.println(se.getSession().getId());
Integer onlineCount = (Integer) servletContext.getAttribute("onlineCount");
if(onlineCount==null){
onlineCount=new Integer(1);
}else {
int count=onlineCount.intValue();
onlineCount=new Integer(count+1);
}
servletContext.setAttribute("onlineCount",onlineCount);
}
// 销毁session监听
public void sessionDestroyed(HttpSessionEvent se) {
ServletContext servletContext = se.getSession().getServletContext();
Integer onlineCount = (Integer) servletContext.getAttribute("onlineCount");
if(onlineCount==null){
onlineCount=new Integer(0);
}else {
int count=onlineCount.intValue();
onlineCount=new Integer(count-1);
}
servletContext.setAttribute("onlineCount",onlineCount);
}
/*
Session 销毁:
1. 手动销毁: se.getSession().invalidate();
2. 自动销毁,在web.xml中配置过期时间
* */
}
- web.xml中注册监听器:
<!--注册监听器-->
<listener>
<listener-class>com.ma.listener.OnlineCountListener</listener-class>
</listener>
<!--销毁Session-->
<session-config>
<session-timeout>1</session-timeout>
</session-config>
- 页面展示:index.jsp:
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>$Title$</title>
</head>
<body>
<h1>当前有 <span><%=this.getServletConfig().getServletContext().getAttribute("onlineCount")%></span></h1>
</body>
</html>
13. 过滤器、监听器的常见应用:
监听器:GUI编程中经常使用;
public class TestListener {
public static void main(String[] args) {
Frame frame = new Frame("小马加油"); // 新建一个窗体
Panel panel=new Panel(null); // 面板
frame.setLayout(null); // 设置窗体的布局
frame.setBounds(300,300,500,500);
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) {
System.exit(0);
}
});
}
}
用户登录后才能进入主页,用户注销后就不能进入主页了!
- 用户登录后,向Session中放入用户的数据
- 进入主页的时候要判断用户是否已经登录
- 用户登录
public class LoginServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// 获取前端请求的参数
String username=req.getParameter("username");
System.out.println(username);
if (username.equals("admin")){
//登录成功
req.getSession().setAttribute(Constant.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);
}
}
- 用户注销
public class LoginOutServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
Object user_session = req.getSession().getAttribute(Constant.USER_SESSION);
if(user_session!=null){
req.getSession().removeAttribute(Constant.USER_SESSION);
resp.sendRedirect("/login.jsp");
}else {
resp.sendRedirect("/login.jsp");
}
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
- 过滤器:
public class SysFilter implements Filter {
public void init(FilterConfig filterConfig) throws ServletException {
}
public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
HttpServletRequest request1 = (HttpServletRequest) request;
HttpServletResponse response1 = (HttpServletResponse) response;
Object userSession = request1.getSession().getAttribute(Constant.USER_SESSION);
if(userSession==null){
response1.sendRedirect("/error.jsp");
}
chain.doFilter(request,response);
}
public void destroy() {
}
}
14. JDBC:
什么是JDBC:java 连接数据库!
需要jar包:
- java.sql
- javax.sql
- mysql-connector-java 连接驱动(必须导入)
1、模拟JDBC:
- 创建数据库
CREATE TABLE users(
id INT PRIMARY KEY,
`name` VARCHAR(40),
`password` VARCHAR(40),
`email` VARCHAR(60),
`birthday` DATE
);
INSERT INTO users(`id`,`name`,`password`,`email`,`birthday`)
VALUES(1,'张三','123456','zs@qq.com','2000-01-01'),(2,'李四','123456','ls@qq.com','2000-01-01'),
(3,'王五','123456','ww@qq.com','2000-01-01');
- 在pom.xml中导入依赖:
<!-- mysql 的驱动-->
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
<version>5.1.47</version>
</dependency>
-
IDEA中连接数据库
-
编写程序:
public class TestJdbc {
public static void main(String[] args) throws ClassNotFoundException, SQLException {
String url="jdbc:mysql://localhost:3306/jdbc?useUnicode=true&characterEncoding=utf-8&useSSL=true";
String username="root";
String password="087415157";
// 1. 加载驱动
Class.forName("com.mysql.jdbc.Driver");
// 2. 连接数据库 代表数据库
Connection connection = DriverManager.getConnection(url, username, password);
// 3. 向数据库发送SQL的对象Statement :CRUD
Statement statement = connection.createStatement();
// 4. 编写SQL
String sql="select * from users";
// 5. 执行查询的SQL。返回一个resultSet :结果集
ResultSet resultSet = statement.executeQuery(sql);
while (resultSet.next()){
System.out.println("id:"+resultSet.getInt("id"));
System.out.println("name:"+resultSet.getString("name"));
System.out.println("password:"+resultSet.getString("password"));
System.out.println("email:"+resultSet.getString("email"));
System.out.println("birthday:"+resultSet.getInt("birthday"));
}
// 6. 关闭连接,释放资源 先开后关
resultSet.close();
statement.close();
connection.close();
}
}
预编译程序(preparedStatement):
public class TestJdbc02 {
public static void main(String[] args) throws ClassNotFoundException, SQLException {
String url="jdbc:mysql://localhost:3306/jdbc?useUnicode=true&characterEncoding=utf-8&useSSL=true";
String username="root";
String password="087415157";
// 1. 加载驱动
Class.forName("com.mysql.jdbc.Driver");
// 2. 连接数据库 代表数据库
Connection connection = DriverManager.getConnection(url, username, password);
String sql="insert into users (id, name, password, email, birthday) VALUES (?,?,?,?,?)";
// 3. 向数据库发送SQL的对象Statement :CRUD
PreparedStatement preparedStatement = connection.prepareStatement(sql);
preparedStatement.setInt(1,4);
preparedStatement.setString(2,"小马");
preparedStatement.setString(3,"123456");
preparedStatement.setString(4,"xm@qq.com");
preparedStatement.setDate(5,new Date(new java.util.Date().getTime()));
int i = preparedStatement.executeUpdate();
if(i>0){
System.out.println("插入成功");
}
// 6. 关闭连接,释放资源 先开后关
preparedStatement.close();
connection.close();
}
}
3、Junit单元测试:
依赖
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>4.12</version>
<scope>test</scope>
</dependency>
简单使用
@Test注解只有在方法上有效,只要加这个注解的方法,就可以直接运行。
成功的时候是绿色,失败的时候是红色
public class TestJdbc03 {
@Test
public void test(){
System.out.println("hello");
}
}
2、事务:
要么都成功,要么都失败
ACID原则:保证数据的安全
开启事务
事务提交
事务回滚
关闭事务
15. SMBMS项目:
1、项目搭建准备工作:
-
搭建一个maven web项目
-
配置Tomcat
-
测试项目是否能启动
-
导入需要的jar包
jsp,Servlet,mysql驱动,jstl,standard
-
创建项目包结构
-
编写实体类:
ORM映射:表-类映射
-
编写基础公共类
-
数据库配置文件:db.properties
driver=com.mysql.jdbc.Driver url=jdbc:mysql://localhost:3306/smbms?useUnicode=true&characterEncoding=utf-8&useSSL=true username=root password=087415157
-
编写数据库的公共类BaseDao:
// 操作数据库的公共类 public class BaseDao { private static String dirver; private static String url; private static String username; private static String password; // 启动后就加载 static { // 通过类加载器读取对应的资源 InputStream resourceAsStream = BaseDao.class.getClassLoader().getResourceAsStream("db.properties"); Properties properties = new Properties(); try { properties.load(resourceAsStream); } catch (IOException e) { e.printStackTrace(); } dirver=properties.getProperty("driver"); url =properties.getProperty("url"); username=properties.getProperty("username"); password=properties.getProperty("password"); } // 获取数据库的连接 public static Connection getConnection(){ Connection connection=null; try { Class.forName(dirver); connection = DriverManager.getConnection(url, username, password); } catch (Exception e) { e.printStackTrace(); } return connection; } // 编写查询公共类 public static ResultSet execute(Connection connection,String sql,Object[]params,ResultSet resultSet){ try { PreparedStatement preparedStatement = connection.prepareStatement(sql); for (int i =0;i<params.length;i++) { // setObject,占位符从1开始,但是我们的数组是从0开始的 preparedStatement.setObject(i+1,params[i]); } resultSet= preparedStatement.executeQuery(); } catch (SQLException e) { e.printStackTrace(); } return resultSet; } // 编写增删改的公共方法 public static int execute(Connection connection,String sql,Object[]params){ int updateRows=0; try { PreparedStatement preparedStatement = connection.prepareStatement(sql); for (int i =0;i<params.length;i++) { // setObject,占位符从1开始,但是我们的数组是从0开始的 preparedStatement.setObject(i+1,params[i]); } updateRows= preparedStatement.executeUpdate(); } catch (SQLException e) { e.printStackTrace(); } return updateRows; } // 释放资源 public static boolean closeResource(Connection connection, PreparedStatement preparedStatement,ResultSet resultSet){ boolean flag=true; if (resultSet!=null) { try { resultSet.close(); //GC回收 resultSet = null; } catch (SQLException e) { e.printStackTrace(); flag = false; } } if (preparedStatement!=null) { try { preparedStatement.close(); //GC回收 preparedStatement = null; } catch (SQLException e) { e.printStackTrace(); flag = false; } } if (connection!=null){ try { connection.close(); //GC回收 connection=null; } catch (SQLException e) { e.printStackTrace(); flag=false; } } return flag; } }
-
编写字符编码过滤器CharacterEncodingFilter:
public class CharacterEncodingFilter implements Filter { public void init(FilterConfig filterConfig) throws ServletException { } public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException { servletRequest.setCharacterEncoding("utf-16"); servletResponse.setCharacterEncoding("utf-16"); filterChain.doFilter(servletRequest,servletResponse); } public void destroy() { } }
-
-
导入静态资源:
2、登录功能实现:
-
编写前端页面:
-
设置首页web.xml:
<!--设置欢迎页面--> <welcome-file-list> <welcome-file>login.jsp</welcome-file> </welcome-file-list>
-
编写dao层登录用户登录的接口UserDao:
public interface UserDao { // 得到要登录的用户 public User getLoginUser(Connection connection,String userCode); }
-
编写dao层实现类UserDaoImpl:
blic class UserDaoImpl implements UserDao { public User getLoginUser(Connection connection, String userCode) { PreparedStatement pstm=null; ResultSet rs=null; User user=null; if (connection!=null){ String sql="select * from smbms_user where userCode=?"; Object[]params={userCode}; try { rs= BaseDao.execute(connection,pstm,rs,sql,params); if (rs.next()){ user=new User(); user.setId(rs.getInt("id")); user.setUserCode(rs.getString("userCode")); user.setUserName(rs.getString("userName")); user.setUserPassword(rs.getString("userPassword")); user.setGender(rs.getInt("gender")); user.setBirthday(rs.getDate("birthday")); user.setPhone(rs.getString("phone")); user.setAddress(rs.getString("address")); user.setUserRole(rs.getInt("userRole")); user.setCreatedBy(rs.getInt("createBy")); user.setCreationDate(rs.getTimestamp("creationDate")); user.setModifyBy(rs.getInt("modifyBy")); user.setModifyDate(rs.getTimestamp("modifyDate")); } BaseDao.closeResource(null,pstm,rs); } catch (SQLException e) { e.printStackTrace(); } } return user; } }
-
业务层接口UserService:
public interface UserService { // 用户登录 public User login(String userCode,String password); }
-
业务实现类UserServiceImpl:
public class UserServiceImpl implements UserService { // 业务层都会调用dao层,所以我们要引入Dao层 private UserDao userDao; public UserServiceImpl(){ userDao=new UserDaoImpl(); } public User login(String userCode, String password) { Connection connection=null; User user=null; connection = BaseDao.getConnection(); // 通过业务层调用对应的具体的数据库操作 user= userDao.getLoginUser(connection,userCode); BaseDao.closeResource(connection,null,null); return user; } }
-
编写Servlet类LoginServlet:
public class LoginServlet extends HttpServlet { // Servlet:控制层调用业务层 @Override protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { System.out.println("LoginServlet-->start..."); // 获取用户名和密码 String userCode= req.getParameter("userCode"); String userPassword=req.getParameter("userPassword"); // 和数据库中的用户名和密码对比,调用业务层 UserServiceImpl userService = new UserServiceImpl(); // 这里已经把登录的人给查出来了 User user = userService.login(userCode, userPassword); if(user!=null){ //查有此人,可以登录 // 将用户的信息放到Session 中 req.getSession().setAttribute(Constants.USER_SESSION,user); // 跳转到内部主页 resp.sendRedirect("jsp/frame.jsp"); } else { //查无此人,无法登录 // 转发回登录页面,顺带提示他,用户名,密码错误: req.setAttribute("error","用户或密码不正确"); req.getRequestDispatcher("login.jsp").forward(req,resp); } } @Override protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { doGet(req, resp); } }
3、退出及权限过滤:
- 退出功能:
public class LoginoutServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// 移除用户的session
req.getSession().removeAttribute(Constants.USER_SESSION);
resp.sendRedirect(req.getContextPath()+"/login.jsp"); // 返回登录页面
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
- 注册xml:
<servlet>
<servlet-name>LogoutServlet</servlet-name>
<servlet-class>com.ma.servlet.user.LoginoutServlet</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>LogoutServlet</servlet-name>
<url-pattern>/jsp/logout.html</url-pattern>
</servlet-mapping>
- 登录拦截优化
public class SysFilter implements Filter {
public void init(FilterConfig filterConfig) throws ServletException {
}
public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
HttpServletRequest request= (HttpServletRequest) servletRequest;
HttpServletResponse response= (HttpServletResponse) servletResponse;
// 过滤器,从Session中获取用户
User user = (User) request.getSession().getAttribute(Constants.USER_SESSION);
if(user==null){ // 已经被移除或注销,或者未登录
response.sendRedirect("/smbms/error.jsp");
}else {
filterChain.doFilter(servletRequest, servletResponse);
}
}
public void destroy() {
}
}
- 拦截器注册:
<!--用户登录过滤器-->
<filter>
<filter-name>SysFilter</filter-name>
<filter-class>com.ma.filter.SysFilter</filter-class>
</filter>
<filter-mapping>
<filter-name>SysFilter</filter-name>
<url-pattern>/jsp/*</url-pattern>
</filter-mapping>
4、密码修改:
-
导入前端素材:
<li><a href="${pageContext.request.contextPath }/jsp/pwdmodify.html">密码修改</a></li>
-
写项目,建议从底层向上写
-
UserDao接口:
// 修改当前用户密码 public int updatePwd(Connection connection,int id,int password) throws SQLException;
-
UserDaoimpl:
public int updatePwd(Connection connection, int id, int password) throws SQLException { PreparedStatement pstm=null; int execute=0; if(connection!=null) { String sql = "update smbms_user set userPassword=? where id=?"; Object[] params = {password, id}; execute = BaseDao.execute(connection, pstm, sql, params); BaseDao.closeResource(null, pstm, null); } return execute; }
-
UserService接口:
// 根据用户ID修改当前用户密码 public boolean updatePwd( int id, int password) ;
-
UserService实现类:
public boolean updatePwd(int id, int password) { Connection connection=null; boolean flag=false; connection = BaseDao.getConnection(); // 修改密码 try { if (userDao.updatePwd(connection,id,password)>0){ flag=true; } } catch (SQLException e) { e.printStackTrace(); }finally { BaseDao.closeResource(connection,null,null); } return flag; }
-
记得实现复用,需要提取方法
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { String method = req.getParameter("method"); if (method.equals("savepwd")&& method!=null){ this.updatePwd(req,resp); } }
5、优化密码修改使用Ajax:
-
阿里巴巴的fastjson:
<dependency> <groupId>com.alibaba</groupId> <artifactId>fastjson</artifactId> <version>1.2.61</version> </dependency>
-
后台代码UserServlet中:
// 验证旧密码,session中有用户密码 public void pwdModify(HttpServletRequest req, HttpServletResponse resp){ User o = (User) req.getSession().getAttribute(Constants.USER_SESSION); String oldpassword=req.getParameter("oldpassword"); // 万能的Map: 结果集 HashMap<String, String> resultMap = new HashMap<String, String>(); if(o==null){ // Session失效,session过期 resultMap.put("result","sessionerror"); }else if(StringUtils.isNullOrEmpty(oldpassword)){ // 输入密码为空 resultMap.put("result","error"); }else{ String userPassword=o.getUserPassword(); //Session中用户的密码 if(oldpassword.equals(userPassword)){ resultMap.put("result","true"); }else{ resultMap.put("result","false"); } } try { resp.setContentType("application/json"); PrintWriter writer = resp.getWriter(); // JSONArray 阿里巴巴JSON工具类 转换格式的 writer.write(JSONArray.toJSONString(resultMap)); writer.flush(); writer.close(); } catch (IOException e) { e.printStackTrace(); } }
6、用户管理实现:
-
导入分页的工具类
-
用户列表页面导入
userlist.jsp
1.获取用户数量:
-
UserDao
// 查询用户总数 public int getUserCount(Connection connection,String username,int userRole)throws SQLException;
-
UserDaoImpl
// 根据用户名或者角色查询用户总数 public int getUserCount(Connection connection, String username, int userRole) throws SQLException { PreparedStatement pstm=null; ResultSet rs=null; int count=0; if (connection!=null){ StringBuffer sql=new StringBuffer(); sql.append("select count(1) as count from smbms_user u,smbms_role r where u.userRole=r.id"); ArrayList<Object> list = new ArrayList<Object>();// 存放我们的参数 if(!StringUtils.isNullOrEmpty(username)){ sql.append("and u.userName like ?"); list.add("%"+username+"%"); // 9 } if(userRole>0){ sql.append(" and u.userRole=?"); list.add(userRole); } // 把list转换成数组 Object[] objects = list.toArray(); System.out.println("UserDaoImpl-->getUserCount"+" "+sql.toString());// 输出最后的sql rs= BaseDao.execute(connection, pstm, rs, sql.toString(), objects); if(rs.next()){ count = rs.getInt("count");//从结果集中获取最终的数量 } BaseDao.closeResource(null,pstm,rs); } return count; }
-
UserService
// 查询记录数 public int getUserCount(String username,int userRole);
-
UserServiceImpl
// 查询记录 public int getUserCount(String username, int userRole) { Connection connection = null; int count=0; try { connection = BaseDao.getConnection(); count = userDao.getUserCount(connection, username, userRole); } catch (SQLException e) { e.printStackTrace(); }finally { BaseDao.closeResource(connection,null,null); } return count; }
2.获取用户列表:
-
UserDao:
// 获取用户列表 public List<User> getUserList(Connection connection,String userName,int userRole,int currentPageNo,int pageSize)throws SQLException;
-
UserDaoImpl:
public List<User> getUserList(Connection connection, String userName, int userRole, int currentPageNo, int pageSize) throws SQLException { PreparedStatement pstm=null; ResultSet rs=null; List<User>userList=new ArrayList<User>(); if(connection!=null){ StringBuffer sql=new StringBuffer(); sql.append("select u.*,r.roleName as userRoleName from smbms_user u,smbms_role r where u.userRole=r.id"); List<Object>list=new ArrayList<Object>(); if(!StringUtils.isNullOrEmpty(userName)){ sql.append(" and u.userName like ?"); list.add("%"+userName+"%"); } if(userRole>0){ sql.append(" and u.userRole= ?"); list.add(userRole); } sql.append(" order by creationDate DESC limit ?,?"); currentPageNo=(currentPageNo-1)*pageSize; list.add(currentPageNo); list.add(pageSize); Object[] params = list.toArray(); System.out.println("getUserList-->sql"+" "+sql); rs=BaseDao.execute(connection,pstm,rs,sql.toString(),params); while (rs.next()){ User _user=new User(); _user.setId(rs.getInt("id")); _user.setUserCode(rs.getString("userCode")); _user.setUserName(rs.getString("userName")); _user.setGender(rs.getInt("gender")); _user.setBirthday(rs.getDate("birthday")); _user.setPhone(rs.getString("phone")); _user.setUserRole(rs.getInt("userRole")); _user.setUserRole(rs.getInt("userRoleName")); userList.add(_user); } BaseDao.closeResource(null,pstm,rs); } return userList; }
-
UserService:
// 根据条件查询用户列表 public List<User>getUserList(String queryUserName,int queryUserRole,int currentPageNo,int pageSize);
-
UserServiceImpl:
public List<User> getUserList(String queryUserName, int queryUserRole, int currentPageNo, int pageSize) { Connection connection=null; List<User>userList=null; System.out.println("queryUserName---->"+queryUserName); System.out.println("queryUserRole---->"+queryUserRole); System.out.println("currentPageNo---->"+currentPageNo); System.out.println("pageSize---->"+pageSize); try { connection=BaseDao.getConnection(); userList=userDao.getUserList(connection,queryUserName,queryUserRole,currentPageNo,pageSize); } catch (SQLException e) { e.printStackTrace(); }finally { BaseDao.closeResource(connection,null,null); } return userList; }
3.获取角色操作:
为了我们职责统一,可以把角色得到操作单独放在一个包中,和entity类对应
-
RoleDao:
public interface RoleDao { // 获取角色列表 public List<Role> getRoleList(Connection connection) throws SQLException; }
-
RoleDaoImpl;
public class RoleDaoImpl implements RoleDao{ // 获取角色列表 public List<Role> getRoleList(Connection connection) throws SQLException { PreparedStatement pstm=null; ResultSet rs=null; ArrayList<Role> roleList = new ArrayList<Role>(); if(connection!=null){ String sql="select * from smbms_role"; Object[]params={}; rs = BaseDao.execute(connection, pstm, rs, sql, params); while (rs.next()){ Role _role=new Role(); _role.setId( rs.getInt("id")); _role.setRoleCode( rs.getString("roleCode")); _role.setRoleName( rs.getString("roleName")); roleList.add(_role); } BaseDao.closeResource(null,pstm,rs); } return roleList; } }
-
RoleService:
public interface RoleService { // 获取角色列表 public List<Role> getRoleList(); }
-
RoleServiceImpl:
public class RoleServiceImpl implements RoleService { // 业务层都会调用dao层,所以我们要引入Dao层 private RoleDao roleDao; public RoleServiceImpl(){ roleDao=new RoleDaoImpl(); } public List<Role> getRoleList() { Connection connection=null; List<Role> roleList=null; try { BaseDao.getConnection(); roleList=roleDao.getRoleList(connection); } catch (SQLException e) { e.printStackTrace(); }finally { BaseDao.closeResource(connection,null,null); } return roleList; }
-
UserServlet:
// 重点,难点 public void query(HttpServletRequest req, HttpServletResponse resp){ // 查询用户列表 // 从前端获取数据 String queryUserName = req.getParameter("queryUserName"); String temp = req.getParameter("queryUserRole"); String pageIndex = req.getParameter("pageIndex"); int queryUserRole=0; // 获取用户列表 UserServiceImpl userService = new UserServiceImpl(); List<User> userList=null; // 第一次走这个请求,一定是第一页,页面大小是固定的 int pageSize=5; // 可以写到配置文件中,方便以后修改 int currentPageNo=1; if (queryUserName==null){ queryUserName=""; } if (temp!=null&& !temp.equals("")){ queryUserRole=Integer.parseInt(temp); // 给查询的赋值 } if(pageIndex!=null){ try { currentPageNo=Integer.parseInt(pageIndex); } catch (NumberFormatException e) { try { resp.sendRedirect("error.jsp"); } catch (IOException ex) { ex.printStackTrace(); } } } // 获取用户总数(分页:上一页 下一页) int totalCount = userService.getUserCount(queryUserName, queryUserRole); // 总页数支持 PageSupport pageSupport=new PageSupport(); // 当前页码 pageSupport.setCurrentPageNo(currentPageNo); // 设置页面容量 pageSupport.setPageSize(pageSize); // 设置总数 pageSupport.setTotalCount(totalCount); // 总页数 int totalPageCount=pageSupport.getTotalPageCount(); // 总共有几页 // 控制首页和尾页 // 如果页面要小于1就显示第一页 if(currentPageNo<1){ currentPageNo=1; }else if(currentPageNo>totalPageCount){ // 当前页面大于最后一页 currentPageNo=totalPageCount; } // 获取用户的展示 userList = userService.getUserList(queryUserName, queryUserRole, currentPageNo, pageSize); req.setAttribute("userList",userList); // 获取角色列表 RoleServiceImpl roleService = new RoleServiceImpl(); List<Role> roleList = roleService.getRoleList(); req.setAttribute("roleList",roleList); req.setAttribute("totalCount",totalCount); req.setAttribute("currentPageNo",currentPageNo); req.setAttribute("totalPageCount",totalPageCount); req.setAttribute("queryUserName",queryUserName); req.setAttribute("queryUserRole",queryUserRole); // 返回前端 try { req.getRequestDispatcher("userlist.jsp").forward(req,resp); } catch (ServletException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } }
16. 文件传输:
-
需要的jar包:
- https://mvnrepository.com/artifact/commons-io/commons-io/2.6
- https://mvnrepository.com/artifact/commons-fileupload/commons-fileupload/1.4
-
文件上传注意事项:
- 为保证服务器的安全,上传文件应该放在外界无法直接访问的目录下,比如WEB-INF目录下
- 为防止文件覆盖的现象发生,要为上传文件产生一个唯一的文件名(时间戳,UUID,MD5等,防止文件名重复)
- 要限制上传文件的最大值
- 可以限制上传文件的类型,在收到上传文件名时,判断后缀名是否合法
-
需要用到的类详解:
- ServletFileUpload负责处理上传的文件数据,并将表单中每个输入项封装成一个File对象,在使用ServletFileUpload对象解析请求时需要DiskFileItemFactory对象,所以,我们需要在进行解析工作前构造好DiskFileItemFactory对象,通过ServletFileUpload对象的构造方法或setFileItemFactory()方法设置ServletFileUpload对象的fileItmFactory属性。
-
表单如果包含一个文件上传输入项的话,这个表单的enctpe属性就必须设置为multipart/form-data
-
常用方法介绍:
// 判断FileItem类对象封装的数据是一个普通文本表单 // 还是一个文件表单,如果是普通表单字段则返回true,否则返回false boolean isFormField(); // getFieldName()方法用于返回表单标签name属性的值 String getFieldName(); // getString()方法用于将FileItem对象中保存的数据流内容以一个字符串返回 String getString(); // getName()方法用于获得文件上传字段中的文件名 String getName(); // 以流的形式返回上传文件的数据内容 InputStream getInputStream(); // delete()方法用来清空FileItem类对象中存放的主题内容 // 如果主体内容被保存在临时文件中,delete方法将删除该临时文件 void delete();
1. ServletFileUpload类:
ServletFileUpload负责处理上传的文件数据,并将表单中每个输入项封装成一个FileItem对象中,使用其parseRequest(HttpServletRequest)方法可以通过表单中每一个HTML标签提交的数据封装成一个FileItem对象,然后以List列表的形式返回。使用该方法处理上传文件简单易用。
17、邮件发送:
-
传输协议
- SMTP协议:发送邮箱:我们通常把处理用户smtp请求(邮件发送请求)的服务器称之为SMTP服务器(邮件发送服务器)。
- POP3协议:接收邮箱:我们通常把处理用户pop3请求(邮件接收请求)的服务器称之为POP3服务器(邮件接收服务器)。
-
需要准备两个包
- mail.jar:https://mvnrepository.com/artifact/javax.mail/mail/1.4.7
- activation.jar:https://mvnrepository.com/artifact/javax.activation/activation/1.1.1
-
主要的四个核心类:
- 一封简单的邮件:
// 一封简单的邮件
public class MailDemo01 {
public static void main(String[] args) throws GeneralSecurityException, MessagingException {
Properties prop=new Properties();
prop.setProperty("mail.host","smtp.qq.com"); // 设置QQ邮箱服务器
prop.setProperty("mail.transport.protocol", "smtp");// 邮箱发送协议
prop.setProperty("mail.smtp.auth", "true");// 需要验证用户名密码
// 如果是QQ邮箱,还要设置SSL加密,加上以下代码即可 其他邮箱不需要
MailSSLSocketFactory sf=new MailSSLSocketFactory();
sf.setTrustAllHosts(true);
prop.put("mail.smtp.ssl","true");
prop.put("mail.smtp.ssl.socketFactory",sf);
// 使用JavaMail发送邮箱的5个步骤
// 1.创建定义整个应用程序所需要的环境信息的Session对象
// QQ才有,其他邮箱不需要
Session session= Session.getDefaultInstance(prop, new Authenticator() {
@Override
protected PasswordAuthentication getPasswordAuthentication() {
// 发件人邮箱用户名 授权码
return new PasswordAuthentication("发送人邮箱","授权码");
}
});
// 开启Session的debug模式,这样就可以查看到程序发送Email的运行状态
session.setDebug(true);
// 2.通过session得到transport对象
Transport ts = session.getTransport();
// 3. 使用邮箱的用户名和授权码连上邮件服务器
ts.connect("smtp.qq.com","发送人邮箱","授权码");
// 4. 创建邮件
// 注意,需要传递Session
MimeMessage message = new MimeMessage(session);
// 指明邮件的发件人
message.setFrom(new InternetAddress("发送人邮箱"));
// 指明邮件的收件人
message.setRecipient(MimeMessage.RecipientType.TO,new InternetAddress("接收人邮箱"));
// 邮件主题
message.setSubject("吼吼");
// 邮箱的文本内容
message.setContent("<h1 style='color:pink'>小马同学</h1>","text/html;charset=UTF-8");
// 5. 发送邮件
ts.sendMessage(message,message.getAllRecipients());
// 6. 关闭连接
ts.close();
}
}