JavaWeb基础01
Servlet
Servlet简介
Servlet就是sun公司开发动态web的一门技术
x 1
编写一个类,实现Servlet接口
把开发好的java类部署到web服务器中
我们把实现了Servlet接口的java程序叫做 Servlet
HelloServlet
servelet接口 Sun公司有两个默认实现类 HttpServlet GenericServlet
构建一个普通的Maven项目 , 删除里面的src目录 , 以后我们学习就在这个项目里面建立Model . pom.xml 这个空的工程就是Maven主工程
关于Maven父子工程的理解 :
父项目中会有
<modules> <module>servlet-01</module> </modules> 子项目中会有
<groupId>com.kuang</groupId> <artifactId>servlet-01</artifactId> <version>1.0-SNAPSHOT</version> 父项目中的java 子项目可以直接使用
son extends father
Maven环境优化
- 修改web.xml为最新的
<?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>
- 将maven的结构搭建完整
编写一个servlet程序
编写一个普通类
实现Servlet接口 这里我们直接继承HttpServlet
public class HelloServlet extends HttpServlet { // 由于get或者post只是请求实现的不同方式, 可以相互调用, 业务逻辑都一样 @Override protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { PrintWriter writer = resp.getWriter(); // 响应流 writer.print("Hello,Servlet"); } @Override protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { doGet(req, resp); } }
编写servlet的映射
为什么需要映射 --> 我们写的是JAVA程序, 但是需要浏览器访问, 而浏览器需要连接web服务器, 所以我们需要在web服务器中注册我们写的Servlet , 还需要给它一个浏览器能够访问的路径
<!--注册Servlet--> <servlet> <servlet-name>hello</servlet-name> <servlet-class>com.kuang.HelloServlet</servlet-class> </servlet> <!--Servlet的请求路径--> <servlet-mapping> <servlet-name>hello</servlet-name> <url-pattern>hello</url-pattern> </servlet-mapping>配置Toncat
注意 --> 配置项目的发布路径就可以了
启动测试
Servlet原理
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>/hello2</url-pattern>
</servlet-mapping>
<servlet-mapping>
<servlet-name>hello</servlet-name>
<url-pattern>/hello3</url-pattern>
</servlet-mapping>
<servlet-mapping>
<servlet-name>hello</servlet-name>
<url-pattern>/hello4</url-pattern>
</servlet-mapping>
<servlet-mapping>
<servlet-name>hello</servlet-name>
<url-pattern>/hello5</url-pattern>
</servlet-mapping>
一个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>
指定一些后缀或者前缀等等...
<!-- 可以自定义注释实现请求映射
注意 *前面不能加项目映射路径 /
但是 hello/suiyi.xiaoliu 也可以访问到Servlet
-->
<servlet-mapping>
<servlet-name>hello</servlet-name>
<url-pattern>*.xiaoliu</url-pattern>
</servlet-mapping>
优先级问题
指定了固有的映射路径 优先级最高 , 如果找不到就会走默认的处理请求
public class HelloServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
System.out.println("进入了DoGet方法");
System.out.println("进入了DoGet方法");
System.out.println("进入了DoGet方法");
System.out.println("进入了DoGet方法");
PrintWriter writer = resp.getWriter();
writer.print("Hello,Servlet");
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
public class ErrorServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
resp.setContentType("text/html");
resp.setCharacterEncoding("utf8");
PrintWriter writer = resp.getWriter();
writer.print("<h1>404</h1>");
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
<!--注册Servlet-->
<servlet>
<servlet-name>hello</servlet-name>
<servlet-class>com.kuang.servlet.HelloServlet</servlet-class>
</servlet>
<!-- Servlet的请求路径-->
<!-- localhost:8080/s1/hello-->
<servlet-mapping>
<servlet-name>hello</servlet-name>
<url-pattern>/hello</url-pattern>
</servlet-mapping>
<!-- 404页面-->
<servlet>
<servlet-name>error</servlet-name>
<servlet-class>com.kuang.servlet.ErrorServlet</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>error</servlet-name>
<url-pattern>/*</url-pattern>
</servlet-mapping>
ServletContext
web容器在启动的时候 它会为每个web程序都创建一个对应的ServletContext对象 它代表了当前的web应用
共享数据
我在这个Servlet中保存的数据 可以在另一个Servlet中拿到
public class HelloServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
ServletContext context = this.getServletContext();
String username = "小刘"; //数据
context.setAttribute("username",username); // 将一个数据保存在servletContext中,名字为: username 值 username
}
}
public class GetServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
ServletContext context = this.getServletContext();
String username = (String) context.getAttribute("username");
resp.setContentType("text/html");
resp.setCharacterEncoding("utf8");
resp.getWriter().print("名字 " + username);
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
<servlet>
<servlet-name>hello</servlet-name>
<servlet-class>com.liu.servlet.HelloServlet</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>hello</servlet-name>
<url-pattern>/hello</url-pattern>
</servlet-mapping>
<servlet>
<servlet-name>getContext</servlet-name>
<servlet-class>com.liu.servlet.GetServlet</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>getContext</servlet-name>
<url-pattern>/getContext</url-pattern>
</servlet-mapping>
测试访问结果
获取初始化参数
<!-- 配置一些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 context = this.getServletContext();
String url = context.getInitParameter("url");
resp.getWriter().print(url);
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
请求转发
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
ServletContext context = this.getServletContext();
System.out.println("进入了ServletDemo04");
//RequestDispatcher requestDispatcher = context.getRequestDispatcher("/getParameter"); //转发的请求路径
//requestDispatcher.forward(req,resp); //调用forward实现请求转发
context.getRequestDispatcher("/getParameter").forward(req,resp);
}
读取资源文件
Properties
- 在java目录下新建properties
- 在resources目录下新建properties
发现 都被打包到了同一个路径下 classes 我们俗称这个路径为classpath
思路 需要一个文件流
username=root
password=631062568
public class ServletDemo05 extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
InputStream inputStream = this.getServletContext().getResourceAsStream("/WEB-INF/classes/db.properties");
Properties prop = new Properties();
prop.load(inputStream);
String username = prop.getProperty("username");
String password = prop.getProperty("password");
resp.getWriter().print(username + " : " + password);
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doGet(req, resp);
}
}
HttpServletResponse
下载文件
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// 获取下载文件的路径
String realPath = "D:\\ideaProgram\\JavaWeb\\javaweb-02-servlet\\response\\target\\classes\\小小.png";
System.out.println("下载文件的路径 --> " + realPath);
// 下载文件名是啥
String fileName = realPath.substring(realPath.lastIndexOf("\\") + 1);
// 设置想办法让浏览器能够支持(Content-Disposition)下载我们需要的东西,中文文件名URLEncoder.encode编码,否则可能会乱码
resp.setHeader("Content-Disposition","attachment;filename" + URLEncoder.encode(fileName,"utf8"));
// 获取下载文件的输入流
FileInputStream fileInputStream = new FileInputStream(realPath);
// 创建缓冲区
int len = 0;
byte[] buffer = new byte[1024];
// 获取OutputStream对象
ServletOutputStream outputStream = resp.getOutputStream();
// 将FileOutputStream流写入到buffer缓冲区,使用OutputStream将缓冲区中的数据输出到客户端
while ((len = fileInputStream.read(buffer)) > 0) {
outputStream.write(buffer,0,len);
}
fileInputStream.close();
outputStream.close();
}
验证码功能
验证怎么来的?
- 前端实现
- 后端实现 需要用到java的图片类 产生一个图片
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// 如何让浏览3秒自动刷新一次
resp.setHeader("refresh", String.valueOf(3));
// 在内存中创建一个图片
BufferedImage bufferedImage = new BufferedImage(80, 20, BufferedImage.TYPE_INT_RGB);
// 得到图片
Graphics2D graphics2D = (Graphics2D) bufferedImage.getGraphics();
// 设置图片背景颜色
graphics2D.setColor(Color.white);
graphics2D.fillRect(0,0,80,20);
// 给图片写数据
graphics2D.setColor(Color.blue);
graphics2D.setFont(new Font(null,Font.BOLD,20));
graphics2D.drawString(makeNum(),0,20);
// 告诉浏览器, 这个请求用图片的方式打开
resp.setContentType("image/jpeg");
// 网站存在缓存, 不让浏览器缓存
resp.setDateHeader("expires",-1);
resp.setHeader("Cahe-Control","no-cache");
resp.setHeader("Prama","no-cache");
// 把图片写给浏览器
ImageIO.write(bufferedImage,"jpg",resp.getOutputStream());
}
// 生成随机数
private String makeNum(){
Random random = new Random();
String randomNum = random.nextInt(9999) + "";
StringBuffer stringBuffer = new StringBuffer();
for (int i = 0; i < 5 - randomNum.length(); i++) {
stringBuffer.append(0);
}
randomNum = stringBuffer.toString() + randomNum;
return randomNum;
}
实现重定向 (重点)
B web资源收到客户端A请求后 B会通知A客户端去访问另一个web资源C 这个过程叫重定向
使用场景
- 用户登录
HttpServletRequest
HttpServletRequest代表客户端的请求 用户通过HTTP协议访问服务器 HTTP请求中的所有信息会被封装到HttpServletRequest
通过这个HttpServletRequest的方法 我们可以获得客户端的所有信息
面试题 请你聊聊重定向和转发的区别?
相同点
- 页面都会跳转
不同点
- 请求转发的时候 URL不会发生变化 307
- 重定向的时候 URL会发生变化 302
请求转发
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
req.setCharacterEncoding("utf8");
resp.setCharacterEncoding("utf8");
String username = req.getParameter("username");
String password = req.getParameter("password");
String[] hobbies = req.getParameterValues("hobbies");
System.out.println(username);
System.out.println(password);
System.out.println(Arrays.toString(hobbies));
System.out.println(req.getContextPath());
// 通过请求转发
// 这里的 / 代表当前web应用
req.getRequestDispatcher("/success.jsp").forward(req,resp);
}
Cookie 和 Session
Session 会话
会话 --> 用户打开一个浏览器 点击了很多超链接 访问多个web资源 关闭浏览器 这个过程可以称之为会话
有状态会话 --> 一个同学来过教室 下次再来教室 我们会知道这个同学曾经来过 称之为有状态会话
比如 --> 你怎么证明你是你们学习的学生
你 学校
发票 学校给你开发票
学校登记 学校登记你来过
一个网站如何证明你来过
客户端 服务端
服务端给客户端一个信件 客户端下次访问服务端带上信件就可以了 cookie
服务器登记你来过 下次你来的时候服务器来匹配你 session
保存会话的两种技术
cookie
客户端技术 ( 响应 请求 )
session
服务器技术 利用这个技术 可以保存用户的会话信息 我们可以把信息或者数据放在Session中
常见 --> 网站登录之后 我们下次不用再登录 第二次访问就直接上去了
Cookie
从请求中拿到cookie信息
服务器响应给客户端cookie
Cookie[]cookies =req.getcookies();//获得cookie
cookie.getName();//获得cookie中的key
cookie.getvalue();//获得cookie中的vaue
new cookie("lastLoginTime",system.currentTimeMillis()+""); //新建一个cookie
cookie.setMaxAge(24*60*60)://设置cookie的有效期
6resp.addcookie(cookie);//响应给客户端一个cookie
cookie:一般会保存在本地的 用户目录下 appdata;
一个网站cookie是否存在上限 ! 聊聊细节问题
- 一个Cookie只能保存一个信息
- 一个web站点可以给浏览器发送多个cookie 最多存放20个cookie
- Cookie大小限制4kb
- 300个cookie浏览器上限
删除Cookie
- 不设置有效期 关闭浏览器 自动失效
- 设置有效时间为 0
编码解码
// 编码
Cookie cookie = new Cookie("name",URLEncoder.encode("小刘","utf8"));
// 解码
out.write(URLDecoder.decode(cookie.getValue(),"utf8"));
Session 重点
Session 和 Cookie 的区别
- Cookie是把用户的数据写给 用户的浏览器 浏览器保存 ( 可以保存多个 )
- Session是把用户的数据写到 用户独占Session中 服务器端保存 ( 保存重要的信息 减少服务器资源的浪费 )
- Session对象由服务器创建
使用场景 -->
- 保存一个登录用户的信息
- 购物车信息
- 在整个网站中经常会使用的数据 我们将它保存在Session中
使用Session -->
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
req.setCharacterEncoding("utf8");
resp.setCharacterEncoding("utf8");
resp.setContentType("text/html;charset=utf8");
// 得到session
HttpSession session = req.getSession();
// 给session中存东西
session.setAttribute("name",new Person("小刘",21));
// 获得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 doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
req.setCharacterEncoding("utf8");
resp.setCharacterEncoding("utf8");
resp.setContentType("text/html;charset=utf8");
// 得到session
HttpSession session = req.getSession();
// String name = (String) session.getAttribute("name");
// System.out.println(name);
Person person = (Person) session.getAttribute("name");
System.out.println(person.toString());
}
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// 获得session
HttpSession session = req.getSession();
session.removeAttribute("name");
// 手动注销Session
session.invalidate();
}
会话自动过期 -->
<!-- 设置Session默认的失效时间-->
<session-config>
<!-- 十五分钟后Session自动失效 以分钟为单位-->
<session-timeout>1</session-timeout>
</session-config>
总结 -->

浙公网安备 33010602011771号