一、jsp运行原理——jsp本质就是servlet(面试)
jsp在第一次被访问时会被Web容器翻译成servlet再执行
过程:
第一次访问——>helloServlet.jsp(翻译成Servlet)——>helloServlet_jsp.java(编译后的Servlet)——>编译运行
PS:被翻译后的servlet在Tomcat的work目录中可以找到
二、jsp脚本和注释
jsp脚本:
<% java代码 %>
—— 里面的java代码翻译到service方法的内部<%=java变量或表达式>
——翻译成service方法内部out.print(),直接输出内容<% ! java代码%>
——翻译成servlet的成员的内容。
jsp注释: 不同的注释可见范围是不同的
-
Html注释:
<!--注释内容-->
——可见范围: jsp源码、翻译后的servlet、浏览器页面显示html源码 -
java注释:
//单行注释 /*多行注释*/
——可见范围: jsp源码翻译后的servlet -
jsp注释:
<%--注释内容--%>
——可见范围 :jsp源码可见
案例:
<%@ page contentType="text/html;charset=UTF-8"language="java" %>
<html>
<head>
<title>$Title$</title>
</head>
<body>
<!--html注释-->
<h1>
JSP页面
</h1>
<%
int i=0;
System.out.println(i);
%>
<%=i%>
<%=1+1%>
//java注释
/*java多行注释*/
<%--jsp注释--%>
<%!
String str = "hello china";
%>
<%=str %>
</body>
</html>
三、JSP指令——(三大指令)
jsp的指令是指导jsp翻译和运行的命令,jsp包括三大指令:
1.page指令
page指令
——属性最多的指令(实际开发中page指令默认)
属性最多的一个指令,根据不同的属性,指导整个页面特性- 格式:
<%@ page 属性名1= "属性值1" 属性名2= "属性值2" ...%>
常用属性如下:
language
:jsp脚本中可以嵌入的语言种类pageEncoding
:当前jsp文件的本身编码—内部可以包含contentTypecontentType
:response.setContentType(text/html;charset=UTF-8)session
:是否jsp在翻译时自动创建sessionimport
:导入java的包。例如想要引入List 会进行报错,未在开头进行导包。errorPage
:当当前页面出错后跳转到哪个页面。isErrorPage
:当前页面是一个处理错误的页面。例如:当error.jsp=”true”时。表示当前是个错误页面。extends
:例如写“java.util.ArrayList”.,此时的jsp文件会报错。开发中不常用,一般不会随便继承。
案例:
注意:留意该jsp上面的一些常用page指令,同时我们把当前页面设置为一个处理错误的页面
<%@page import="java.util.ArrayList"%>
<%@page import="java.util.List"%>
<%@page language="java" contentType="text/html; charset=UTF-8"
pageEncoding="UTF-8" session="true" isErrorPage="true"%>
<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title>Insert title here</title>
</head>
<body>
<%List<String> list =new ArrayList<String>(); %>
<%session.setAttribute("name","zhangsan"); %>
<% out.print("this is errorPage"); %>
</body>
</html>
注意:此时我们访问该工程下面不存在的资源的时候,会跳转到当前页面。并且我们需要在web.xml
中写一些东西如下:
<error-page>
<error-code>404</error-code>
<location>/jspdemo.jsp</location> <!--表示jsp的文件名-->
</error-page>
效果如下:
2. include指令
页面包含(静态包含)指令,可以将一个jsp页面包含到另一个jsp页面中
格式:<%@ include file="被包含的文件地址"%>
例如:我们创建一个demo.jsp,并且我们在index.jsp中书写
<body>
<%@include file="demo.jsp"%>
</body>
此时我们访问index.jsp,里面的内容会包含demo.jsp的内容
3. taglib指令
在jsp页面中引入标签库(jstl标签库、struts2标签库)
格式:<%@ taglib uri="标签库地址" prefix="前缀"%>
类似于前端中引入css 和 JavaScript文件
四. jsp内置/隐式对象(9个)——(重要)
jsp被翻译成servlet之后,service方法中有9个对象定义并初始化完毕,我们在jsp 脚本中可以直接使用这9个对象
名称 | 类型 | 描述 |
---|---|---|
out | javax.servlet.jsp.JspWriter | 用于页面输出 |
request | javax.servlet.http.HttpServletRequest | 得到用户请求信息 |
response | javax.servlet.http.HttpServletResponse | 服务器向客户端的回应信息 |
config | javax.servlet.ServletConfig | 服务器配置,可以取得初始化参数 |
session | javax.servlet.http.HttpSession | 用来保存用户的信息 |
application | javax.servlet.ServletContext | 所有用户的共享信息 |
page | java.lang.Object | 指当前页面转换后的Servlet类的实例(注意下方案例的开头) |
pageContext | javax.servlet.jsp.PageContext | JSP的页面容器 |
exception | java.lang.Throwable | 表示JSP页面所发生的异常,在错误页中才起作用 |
1. out对象
out的类型:JspWriter
out作用就是为客户端输出内容——out.write()
out缓冲区默认8kb 可以设置成0 代表关闭out缓冲区 内容直接写到respons缓冲器
案例:
<%@ page language="java" contentType="text/html; charset=UTF-8"
pageEncoding="UTF-8" %>
<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title>Insert title here</title>
</head>
<body>
aaaaaaaaaaaaaa
<% out.write("bbbbbbbbbbbbbbbbbb");
response.getWriter().write("ccccccccc");
%>
<%="ddddddddddddd" %>
<% out.print("ffffffff"); %>
</body>
</html>
效果如下:
为什么这里ccccc先输出的原因是:Tomcat引擎从response缓冲区获得内容再到out缓冲区
2. pageContext对象
page对象与pageContext对象是两个对象,注意区别
jsp页面的上下文对象(pageContext对象),作用如下:
pageContext是一个域对象
同其他作用域相同,pageContext也有下面三个方法:
setAttribute(String name,Object obj)
getAttribute(String name)
removeAttrbute(String name)
pageContext可以向指定的其他域中存取数据
setAttribute(String name,Object obj,int scope)
getAttribute(String name,int scope)
removeAttrbute(String name,int scope)
案例:
<body>
<%
//使用pageContext向request域存数据
request.setAttribute("name","zhangsan");
//将name =zhangsan,设置到reques的域中,该行代码与上句代码作用一样
pageContext.setAttribute("name","lisi",PageContext.REQUEST_SCOPE);
//使用pageContext向session域存数据
pageContext.setAttribute("name","wangwu",PageContext.SESSION_SCOPE);
//使用pageContext向application域存数据
pageContext.setAttribute("name","tianqi",PageContext.APPLICATION_SCOPE);
%>
</body>
pageContext对象有一个比较牛逼的功能是可以自动寻找Attribute
findAttribute(String name)
依次从pageContext域,request域,session域,application域中获取属性,在某个域中获取后将不在向后寻找.
案例:
body>
<%
//使用pageContext向request域存数据
request.setAttribute("name","zhangsan");
//将name =zhangsan 设置到reques的域中,该行代码与上句代码作用一样 pageContext.setAttribute("name","lisi",PageContext.REQUEST_SCOPE);
pageContext.setAttribute("name","wangwu",PageContext.SESSION_SCOPE);
pageContext.setAttribute("name","tianqi",PageContext.APPLICATION_SCOPE);
%>
<%=request.getAttribute("name")%>
//获得域中的属性
<%=request.getAttribute("name")%>>
//该方法域上面的request方法一样
<%=pageContext.getAttribute("name",PageContext.REQUEST_SCOPE)%>
//从所有域中找数据,小到大中搜索域中的范围中的name
<%--从小到大 pageContext域,request域,session域,application域--%>
<%=pageContext.findAttribute("name")%>>
</body>
这里插播一个域的大小复习比较
四大作用域的总结:
page域
:当前jsp页面范围request域
:一次请求,多次转发也算一次请求。session域
:一次会话,页面开启到页面关闭。application域
:整个web应用
pageContext对象可以获得其他8大隐式对象
案例:
<%
pageContext.getOut();
pageContext.getRequest();
pageContext.getResponse()
%>
五. jsp标签(动作)
- 页面包含(动态包含):
<jsp:include page="被包含的页面"/>
- 请求转发:
<jsp:forward page="要转发的资源" />
地址不变。
静态包含与动态包含的区别?
- 静态包含:
<%@include file=”/”%>
- 动态包含:
<jsp.include page =”被包含的页面”/>
静态包含是将两个jsp页面拼凑一起合成一个页面。然而动态页面是分开的两个页面但不拼凑在一起。
案例:
做一个动态的包含
第一步:写第一个页面
body>
<%--include1中包含include2--%>
<jsp:include page="include2.jsp"></jsp:include>
<h1>This is include1 page</h1>
</body>
第二步:写第二个页面
<body>
<h1>This is include2 page</h1>
</body>
此时我们访问第一个页面的时候,包含了第二个页面的内容。实则是两个页面。