09 Servlet详解和HttpServlet应用
1. Servelt
1.1 Servlet 简介
-
sun公司开发动态web的一门技术
-
Sun公司在这些API中提供了一个接口叫做:Servlet,如果你想开发一个Servlet程序,只需要完成两个小步骤
- 编写一个类,实现Servlet接口
- 把开发好的Java类部署到web服务器中
把实现了Servlet接口的Java程序叫做,Servlet
1.2 HelloServlet
Servlet接口Sun公司有两个默认的实现类:HttpServlet
-
构建一个普通的项目,删掉里面的src目录,以后在这个项目里面建立Module;这个空的工程就是Maven主工程
-
关于Maven父子工程的理解
父项目中
<modules>
<module>servlet01</module>
</modules>
子项目中
<parent>
<artifactId>javaweb-02-servelt</artifactId>
<groupId>com.ybl</groupId>
<version>1.0-SNAPSHOT</version>
</parent>
父项目中的java子项目可以直接使用
-
Maven环境优化
- 修改web.xml为最新
- 将maven的结构搭建完整
-
编写一个Servlet程序
-
编写一个普通类
-
实现一个Servlet接口,这里我们直接继承HttpServlet
public class HelloServlet extends HttpServlet { @Override protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { PrintWriter writer = resp.getWriter(); writer.print("hello,servlet"); } @Override protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { super.doPost(req, resp); } }
-
-
编写Servlet的映射
-
为什么需要映射:我们写的是Java程序,但是要通过浏览器访问,而浏览器需要连接Web服务器,所以我们需要在web服务器中注册我们写的Servlet,还需要一个浏览器能访问的路径
<!-- 注册Servlet--> <servlet> <servlet-name>hello</servlet-name> <servlet-class>com.ybl.servlet.HelloServlet</servlet-class> </servlet> <!-- Servlet请求路径--> <servlet-mapping> <servlet-name>hello</servlet-name> <url-pattern>/hello</url-pattern> </servlet-mapping>
-
-
配置Tomcat
如果没有Artificat 看一下我的收藏夹
7.启动测试
1.3 Servlet原理
Servlet是由Web服务器(Tomcat)调用的,web服务器在收到浏览器请求之后会: 下图其实不够清晰
1.4 Mapping问题
- 一个Servlet可以指定一个映射路径
<!-- 注册Servlet-->
<servlet>
<servlet-name>hello</servlet-name>
<servlet-class>com.ybl.servlet.HelloServlet</servlet-class>
</servlet>
<!-- Servlet请求路径-->
<servlet-mapping>
<servlet-name>hello</servlet-name>
<url-pattern>/hello</url-pattern>
</servlet-mapping>
- 一个Servlet可以指定多个映射路径
<!-- 注册Servlet-->
<servlet>
<servlet-name>hello</servlet-name>
<servlet-class>com.ybl.servlet.HelloServlet</servlet-class>
</servlet>
<!-- Servlet请求路径-->
<servlet-mapping>
<servlet-name>hello</servlet-name>
<url-pattern>/hello1</url-pattern>
</servlet-mapping>
<servlet-mapping>
<servlet-name>hello</servlet-name>
<url-pattern>/hello2</url-pattern>
</servlet-mapping>
- 一个Servlet可以指定通用映射路径
<!-- 注册Servlet-->
<servlet>
<servlet-name>hello</servlet-name>
<servlet-class>com.ybl.servlet.HelloServlet</servlet-class>
</servlet>
<!-- Servlet请求路径-->
<servlet-mapping>
<servlet-name>hello</servlet-name>
<url-pattern>/hello/*</url-pattern> 通配符 /hello/123123213都会跳转
</servlet-mapping>
- 指定一个后缀或者前缀
<!-- 注册Servlet-->
<servlet>
<servlet-name>hello</servlet-name>
<servlet-class>com.ybl.servlet.HelloServlet</servlet-class>
</servlet>
<!-- Servlet请求路径-->
<servlet-mapping>
<servlet-name>hello</servlet-name>
<url-pattern>*.do</url-pattern> 注意这里使用后缀的话没有斜线,*是通配符 ,do可以改成别的 ybl.ybb都可以
</servlet-mapping>
-
优先级问题
指定了固有的映射路径优先级最高,如果找不到就会走默认的处理请求
<servlet> <servlet-name>Error</servlet-name> <servlet-class>com.ybl.servlet.Error</servlet-class> </servlet> <servlet-mapping> <servlet-name>Error</servlet-name> <url-pattern>/*</url-pattern> </servlet-mapping>1.5 做的时候遇到的问题
1.5 做的时候遇到的问题
-
配置Tomcat的时候没有Artifacts
- 看我的csdn收藏夹
-
更改web.xml之后,target文件夹没有更新,导致找不到资源
-
还有一种就是i右边Maven刷新以下就行了
1.6 ServletContext
web容器在启动的时候,它会为每个web程序都创建一个对应的ServletContext对象,他代表了当前的web应用。
共享数据
- 我在这个Servlet中保存的数据,可以在另外一个Servlet中拿到
public class HelloServler extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// this.getInitParameter();
// this.getServletConfig(); Servlet配置 就是web.xml中servlet配置
// this.getServletContext(); Servlet上下文
ServletContext servletContext = this.getServletContext();
resp.setCharacterEncoding("utf-8");
String username="叶炳龙";
//将一个数据保存在ServletContext中,名字为username,值为username
servletContext.setAttribute("username",username);
System.out.println("servlet02");
}
}
public class GetServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
ServletContext servletContext = this.getServletContext();
resp.setContentType("text/html");
resp.setCharacterEncoding("utf-8");
String username=(String)servletContext.getAttribute("username");
resp.getWriter().print("名字"+username);
}
}
获取初始化参数
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") ; //获取web.xml中的默认初始化值
resp.getWriter().print(url);
}
}
<!-- 配置一些web应用初始化参数-->
<context-param>
<param-name>url</param-name>
<param-value>jdbc:mysql://localhost:3306/mybatis</param-value>
</context-param>
请求转发
请求转发和重定向不一样,转发的时候URL不变,重定向URL会变化
public class ServletDemo04 extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
System.out.println("进入了04类");
ServletContext servletContext = this.getServletContext();
RequestDispatcher requestDispatcher = servletContext.getRequestDispatcher("/gp");//获取转发的路径
requestDispatcher.forward(req,resp);//转发
}
}
读取资源文件
Propertoes
- 在java目录下新建properties
- 在resources目录下新建properties
如果java中的properties 打包不了,在web.xml中放入一下
<build>
<resources>
<resource>
<directory>src/main/resources</directory>
<includes>
<include>**/*.properties</include>
<include>**/*.xml</include>
</includes>
<filtering>true</filtering>
</resource>
<resource>
<directory>src/main/java</directory>
<includes>
<include>**/*.properties</include>
<include>**/*.xml</include>
</includes>
<filtering>true</filtering>
</resource>
</resources>
</build>
发现都被打包到同一额路径下:classes,我们俗称这个路径为classpath
思路需要一个文件流
public class ServletDemo05 extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//获得流 ,文件的路径是target中的路径,第一个/表示当前web项目
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);
resp.getWriter().print(pwd);
}
}
1.7 HttpServletResponse
web服务器接收到客户端的http请求,针对这个请求,分别创建一个代表请求的HttpServletRequest 对象,和代表响应的HttpServletResponse;
- 如果要获取客户端请求过来的参数:找HttpServletRequest
- 如果要给客户端响应:找HttpServletRequest
简单分类
负责向浏览器发送数据的方法
ServletOutputStream getOutputStream() throws IOException;
PrintWriter getWriter() throws IOException;
负责向浏览器发送响应头的方法
void setCharacterEncoding(String var1);
void setContentLength(int var1);
void setContentType(String var1);
void setBufferSize(int var1);
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);
常见应用 下载文件
- 向浏览器输出消息
- 下载文件
- 获取下载文件的路径
- 下载的文件名是什么
- 设置想办法让浏览器支持下载我们需要的东西
- 获取下载文件的输入流
- 创建缓冲区
- 获取OutputStream对象
- 将FileOutputStream写入到buffer缓冲区
- 使用OutputStream将缓冲区的数据输入到浏览器
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;
public class FileServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// 1. 获取下载文件的路径
//"this.getServletContext().getRealPath("/1.png");"
String realPath ="D:\\IDEAProj\\javaweb-02-servelt\\response\\src\\main\\resources\\1.png" ;
System.out.println("下载文件路径:"+realPath);
// 2. 下载的文件名是什么
String fileName = realPath.substring(realPath.lastIndexOf("\\") + 1);// 两个\\表示/ 转义字符 这样就截取。
// 3. 设置想办法让浏览器支持下载我们需要的东西
resp.setHeader("Content-disposition","attachment;filename="+fileName);
// 4. 获取下载文件的输入流
FileInputStream in = new FileInputStream(realPath);
// 5. 创建缓冲区
int len=0;
byte[] buffer = new byte[1024];
// 6. 获取OutputStream对象
ServletOutputStream out = resp.getOutputStream();
// 7. 将FileOutputStream写入到buffer缓冲区
while ((len=in.read(buffer))>0){
out.write(buffer,0,len);
}
// 8. 使用OutputStream将缓冲区的数据输入到浏览器
in.close();;
out.close();
}
}
验证码功能
验证怎来的?
- 前端实现
- 后端实现,需要用到Java的图片类
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//如何让浏览器3秒自动刷新一次;
resp.setHeader("refresh","3");
//在内存中创建图片
BufferedImage image = new BufferedImage(80,20,BufferedImage.TYPE_INT_RGB);
//得到图片
Graphics2D g=( Graphics2D) image.getGraphics(); //笔
g.setColor(Color.BLUE);
g.fillRect(0,0,80,20);
//给图片写数据
g.setColor(Color.WHITE);
g.setFont(new Font(null,Font.BOLD,20));
g.drawString(makeNum(),0,20);
// 告诉浏览器这个求情用图片的方式打开
resp.setContentType("image/jpeg");
//网站存在缓存,不让浏览器缓存
resp.setDateHeader("expires",-1);
resp.setHeader("Cache-Control","no-cache");
resp.setHeader("Pragma","no-cache");
//把图片写给浏览器
ImageIO.write(image, "jpg",resp.getOutputStream());
}
//生成随机数
private String makeNum(){
Random random = new Random();
String num=random.nextInt(999999)+" ";
StringBuffer stringBuffer = new StringBuffer();
for (int i = 0; i < 6 - num.length(); i++) {
stringBuffer.append("0");
}
num=stringBuffer.toString()+num;
return num;
}
重定向
resp.sendRedict("/webapps名字/target")
resp.setHeader("Location","/r/img")
1.8 HttpServletRequest
HTTP中请求的所有信息会被封装到HttpServletRequest,通过这个HttpServletRequest的方法获得客户端的所有信息i
获取前端参数
String username=req.getParameter("username");
Buffer();
for (int i = 0; i < 6 - num.length(); i++) {
stringBuffer.append(“0”);
}
num=stringBuffer.toString()+num;
return num;
}
#### 重定向
```java
resp.sendRedict("/webapps名字/target")
resp.setHeader("Location","/r/img")
1.8 HttpServletRequest
HTTP中请求的所有信息会被封装到HttpServletRequest,通过这个HttpServletRequest的方法获得客户端的所有信息i
获取前端参数
String username=req.getParameter("username");