1. jsp介绍
如:Servlet 回传 html 页面数据的代码:
public class PringHtml extends HttpServlet {
@Override
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
// 通过响应的回传流回传 html 页面数据
response.setContentType("text/html; charset=UTF-8");
PrintWriter writer = response.getWriter();
writer.write("<!DOCTYPE html>\r\n");
writer.write(" <html lang=\"en\">\r\n");
writer.write(" <head>\r\n");
writer.write(" <meta charset=\"UTF-8\">\r\n");
writer.write(" <title>Title</title>\r\n");
writer.write(" </head>\r\n");
writer.write(" <body>\r\n");
writer.write(" 这是 html 页面数据 \r\n");
writer.write(" </body>\r\n");
writer.write("</html>\r\n");
writer.write("\r\n");
}
}
结果:
jsp 回传一个简单 html 页面的代码:
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>Title</title>
</head>
<body>
这是一个html页面
</body>
</html>
结果:
1.1 如何创建 jsp 的页面?
1.2 访问jsp页面
比如:在 web 目录下有如下的文件:web 目录a.html 页面访问地址是 =======>>>>>> http://ip:port/工程路径/a.htmlb.jsp 页面访问地址是 =======>>>>>> http://ip:port/工程路径/b.jsp
2. jsp 的本质
jsp 页面本质上是一个 Servlet 程序。
3. jsp 的三种语法
3.1 jsp 头部的 page 指令
<%@ page contentType="text/html;charset=UTF-8" language="java" %>i. language 属性 表示 jsp 翻译后是什么语言文件。暂时只支持 java。ii. contentType 属性 表示 jsp 返回的数据类型是什么。也是源码中 response.setContentType()参数值iii. pageEncoding 属性 表示当前 jsp 页面文件本身的字符集。iv. import 属性 跟 java 源代码中一样。用于导包,导类。================== 两个属性是给 out 输出流使用 ==========================v. autoFlush 属性 设置当 out 输出流缓冲区满了之后,是否自动刷新冲级区。默认值是 true。vi. buffer 属性 设置 out 缓冲区的大小。默认是 8kb
vii. errorPage 属性 设置当 jsp 页面运行时出错,自动跳转去的错误页面路径。errorPage 表示错误后自动跳转去的路径这个路径一般都是以斜杠打头,它表示请求地址为 http://ip:port/ 工程路径 / 映射到代码的 Web 目录viii. isErrorPage 属性 设置当前 jsp 页面是否是错误信息页面。默认是 false。如果是 true 可以获取异常信息。ix. session 属性 设置访问当前 jsp 页面,是否会创建 HttpSession 对象。默认是 true。x. extends 属性 设置 jsp 翻译出来的 java 类默认继承谁。
3.2 jsp 中的常用脚本
3.2.1 声明脚本(极少使用)
示例代码:
<%@ page import="java.util.Map" %>
<%@ page import="java.util.HashMap" %>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>Title</title>
</head>
<body>
这是一个html页面
<%-- 1、声明类属性 --%>
<%!
private Integer id;
private String name;
private static Map<String, Object> map;
%>
<%-- 2、声明 static 静态代码块 --%>
<%!
static {
map = new HashMap<String, Object>();
map.put("key1", "value1");
map.put("key2", "value2");
map.put("key3", "value3");
map.put("key4", "value4");
}
%>
<%-- 3、声明类方法 --%>
<%!
public int getA() {
return 12;
}
%>
<%-- 4、声明内部类--%>
<%!
public static class A{
private Integer id = 12;
private String name = "a";
}
%>
</body>
</html>
查看被翻译到的对应的源码中:
3.2.2 表达式脚本(常用)
代码示例:
<br/>
<%--1. 输出整型 --%>
<%= 12 %><br/>
<%--2. 输出浮点型 --%>
<%= 12.21 %><br/>
<%--3. 输出字符串 --%>
<%= "你好" %><br/>
<%--4. 输出对象--%>
<%= map %><br/>
<%= request.getParameter("username")%><br/>
结果:
源码翻译对照:
3.2.3 代码脚本
代码示例1:
<%--1. 代码脚本----if 语句 --%>
<%
int i = 11;
if (i == 11) {
System.out.println("i等于11");
} else {
System.out.println("i不等于11");
}
%>
<%--2. 代码脚本----for 循环语句 --%>
<%
for (int j = 0; j < 10; j++) {
System.out.println(j);
}
%>
<%--3. 翻译后 java 文件中_jspService 方法内的代码都可以写--%>
<%
String username = request.getParameter("username");
System.out.println(username);
%>
结果;
代码示例2:
<%--多个代码脚本块组合完成一个完整的 java 语句。--%>
<%
for (int j = 0; j < 10; j++) {
%>
<%
System.out.printf("%d < ",j);
}
%>
结果:
代码示例3:
<%--代码脚本还可以和表达式脚本一起组合使用,在 jsp 页面上输出数据--%>
<table border="1">
<%
for (int j = 0; j < 10; j++) {
%>
<tr>
<td>
<%= "第" + (j + 1) + "行"%>
</td>
</tr>
<%
}
%>
</table>
结果:
3.3 jsp 中的三种注释
3.3.1 html 注释
html 注释会被翻译到 java 源代码中。在_jspService 方法里,以 out.writer 输出到客户端
3.3.2 java 注释
3.3.3 jsp 注释
4. jsp 九大内置对象
jsp 九大内置对象
1. request 请求对象
2. response 响应对象
3. pageContext jsp是上下文对象
4. session 会话对象
5. exception 异常对象 (注:该对象只有当头部的page指令中isErrorPage=true时才会出现 )
6. application ServletContext对象
7. config ServletConfig对象
8. out jsp输出流对象
9. page 指向当前jsp的对象
5.jsp 四大域对象
四个域对象分别是:pageContext (PageContextImpl 类)当前 jsp 页面范围内有效request (HttpServletRequest 类)一次请求内有效session (HttpSession 类)一个会话范围内有效(打开浏览器访问服务器,直到关闭浏览器)application (ServletContext 类)整个 web 工程范围内都有效(只要 web 工程不停止,数据都在)
代码示例1:
<body>
<h1>这是scope页面</h1>
<%
// 往四个域对象中分别保存数据
pageContext.setAttribute("key","pageContext");
request.setAttribute("key", "request");
session.setAttribute("key","session");
application.setAttribute("key","application");
%>
pageContext 域是否有值: <%= pageContext.getAttribute("key") %><br/>
request 域是否有值: <%= request.getAttribute("key") %><br/>
session 域是否有值: <%= session.getAttribute("key") %><br/>
application 域是否有值: <%= application.getAttribute("key") %><br/>
</body>
结果:
代码示例2:
scope1
<h1>这是scope页面</h1>
<%
// 往四个域对象中分别保存数据
pageContext.setAttribute("key","pageContext");
request.setAttribute("key", "request");
session.setAttribute("key","session");
application.setAttribute("key","application");
%>
pageContext 域是否有值: <%= pageContext.getAttribute("key") %><br/>
request 域是否有值: <%= request.getAttribute("key") %><br/>
session 域是否有值: <%= session.getAttribute("key") %><br/>
application 域是否有值: <%= application.getAttribute("key") %><br/>
<%
request.getRequestDispatcher(("/scope2.jsp")).forward(request,response);
%>
scope2:
<body>
<h1>这是scope2页面</h1>
pageContext 域是否有值: <%= pageContext.getAttribute("key") %><br/>
request 域是否有值: <%= request.getAttribute("key") %><br/>
session 域是否有值: <%= session.getAttribute("key") %><br/>
application 域是否有值: <%= application.getAttribute("key") %><br/>
</body>
用浏览器打开scope.jsp页面:
结果:
用浏览器直接打开scope2.jsp页面:
结果:
先关闭浏览器,再直接打开scope2.jsp页面:
结果:
先重启程序运行,再刷新scope2.jsp页面:
结果:
6. jsp 中的 out 输出和 response.getWriter 输出的区别
response 中表示响应,我们经常用于设置返回给客户端的内容(输出)
out 也是给用户做输出使用的。
7. jsp 的常用标签
7.1 jsp 静态包含
<%@ include file=""%> 就是静态包含file 属性指定你要包含的 jsp 页面的路径地址中第一个斜杠 / 表示为 http://ip:port/工程路径/ 映射到代码的 web 目录静态包含的特点:1、静态包含不会翻译被包含的 jsp 页面。2、静态包含其实是把被包含的 jsp 页面的代码拷贝到包含的位置执行输出。
代码示例:
main.jsp:
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>main</title>
</head>
<body>
头部信息<br/>
主体内容<br/>
<%@include file="/include/footer.jsp"%><br/>
</body>
</html>
footer.jsp:
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>footer</title>
</head>
<body>
底部信息
</body>
</html>
结果:
查看源代码:
把被包含的 jsp 页面的代码拷贝到包含的位置执行输出
7.2 jsp 动态包含
<jsp:include page=""></jsp:include>这是动态包含page 属性是指定你要包含的 jsp 页面的路径动态包含也可以像静态包含一样。把被包含的内容执行输出到包含位置动态包含的特点:1、动态包含会把包含的 jsp 页面也翻译成为 java 代码2、动态包含底层代码使用如下代码去调用被包含的 jsp 页面执行输出。JspRuntimeLibrary.include(request, response, "/include/footer.jsp",out, false);3、动态包含,还可以传递参数
代码示例:
main.jsp:
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>main</title>
</head>
<body>
头部信息<br/>
主体内容<br/>
<%-- <%@include file="/include/footer.jsp"%><br/>--%>
<jsp:include page="/include/footer.jsp">
<jsp:param name="username" value="username"/>
<jsp:param name="password" value="password"/>
</jsp:include><br/>
</body>
</html>
footer.jsp:
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>footer</title>
</head>
<body>
底部信息<br/>
<%= request.getParameter("password") %>
</body>
</html>
结果:
查看源代码:
main_jsp.java :
footer_jsp.java :
动态包含的底层原理:
7.3 jsp 标签-转发
<jsp:forward page=""></jsp:forward> 是请求转发标签,它的功能就是请求转发page 属性设置请求转发的路径
<jsp:forward page="/scope2.jsp"></jsp:forward>
8. jsp练习
8.1 在 jsp 页面中输出九九乘法口诀表
代码:
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>test1</title>
<style type="text/css">
table{
width:60%;
}
</style>
</head>
<body>
<h1 align="center">九九乘法口诀表</h1>
<table align="center">
<% for (int i = 1; i < 10; i++) { %>
<tr>
<% for (int j = 1; j <= i; j++) { %>
<td><%= j + "X" + i + "=" + (i * j) %></td>
<% } %>
</tr>
<% } %>
</table>
</body>
</html>
结果:
8.2 jsp 输出一个表格,里面有 10 个学生信息。
代码:
User类(构造器、set\get方法略):
public class User {
private Integer id;
private String name;
private Integer age;
private String phone;
}
test2.jsp:
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>test2</title>
<style type="text/css">
table{
border:1px blue solid;
width:650px;
border-collapse: collapse;
}
tr,th,td{
border:1px blue solid;
text-align: center;
}
</style>
</head>
<body>
<%
List<User> list = new ArrayList<>();
for (int i = 0; i < 10; i++) {
int t = i + 1;
list.add(new User(t, "name" + t, 17 + t, "phone" + t));
}
%>
<table>
<tr>
<th>编号</th>
<th>姓名</th>
<th>年龄</th>
<th>手机号</th>
<th>操作</th>
</tr>
<% for (User user : list) { %>
<tr>
<td><%= user.getId() %></td>
<td><%= user.getName() %></td>
<td><%= user.getAge() %></td>
<td><%= user.getPhone() %></td>
<td><%= "update/delete" %></td>
</tr>
<% } %>
</table>
结果:
使用请求转发的方式:
代码:
SearchStudentServlet .java
public class SearchStudentServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
// 获取请求的参数
// 发 sql 语句查询学生的信息
// 使用 for 循环生成查询到的数据做模拟
List<User> list = new ArrayList<>();
for (int i = 0; i < 10; i++) {
int t = i + 1;
list.add(new User(t, "name" + t, 17 + t, "phone" + t));
}
// 保存查询到的结果(学生信息)到 request 域中
request.setAttribute("list", list);
// 请求转发到 showStudent.jsp 页面
request.getRequestDispatcher("/test/SearchStudentServlet.jsp").forward(request,response);
}
}
SearchStudentServlet .jsp:
9. Listener 监听器(了解皮毛)
ServletContextListener 监听器
public interface ServletContextListener extends EventListener { /** * 在 ServletContext 对象创建之后马上调用,做初始化 */ public void contextInitialized(ServletContextEvent sce); /** * 在 ServletContext 对象销毁之后调用 */ public void contextDestroyed(ServletContextEvent sce); }
如何使用 ServletContextListener 监听器监听 ServletContext 对象。使用步骤如下:1、编写一个类去实现 ServletContextListener2、实现其两个回调方法3、到 web.xml 中去配置监听器
监听器实现类:
public class MyServletContextListenerImpl implements ServletContextListener {
@Override
public void contextInitialized(ServletContextEvent sce) {
/* This method is called when the servlet context is initialized(when the Web application is deployed). */
System.out.println("servletContext对象被创建了");
}
@Override
public void contextDestroyed(ServletContextEvent sce) {
/* This method is called when the servlet Context is undeployed or Application Server shuts down. */
System.out.println("servletContext对象被销毁了");
}
}
xml配置文件:
<listener>
<listener-class>com.example._09_jsp.listener.MyServletContextListenerImpl</listener-class>
</listener>