动态资源技术JSP|Java与Html的美好相遇
前言
JSP可以用简单易懂的方式表示为:Html+Java=JSP,JSP技术使用Java编程语言编写类XML的tags和scriptlets,来封装产生动态网页的处理逻辑。网页还能通过tags和scriptlets访问存在于服务端的资源的应用逻辑。JSP将网页逻辑与网页设计和显示分离,支持可重用的基于组件的设计,使基于Web的应用程序的开发变得迅速和容易。
公众号:「浅羽的IT小屋」
1、会话
「生活中的会话:」
商量个事嘛 嘛事 借点钱叭 喂...喂...我这里信号不好...
「软件中会话:」
打开浏览器->访问应用->关闭浏览器 用户打开一个浏览器,点击多个超链接,访问服务器多个Web资源,然后关闭浏览器,整个过程称之为一个会话
「会话过程中要解决的一些问题:」
每个用户在使用浏览器与服务器进行会话的过程中,不可避免各自会产生一些数据,服务器要想办法为每个用户保存这些数据 例如:多个用户点击超链接通过一个servlet各自买了一个商品,服务器应该想办法把每一个用户购买的商品保存在各自的地方,以便于这些用户点结账servlet时,结账servlet可以得到用户各自购买的商品为用户结账
「会话技术:」
Cookie:将数据保存在浏览器客户端的技术
服务器把每个用户的数据以cookie的形式写给用户各自的浏览器。当用户使用浏览器再去访问服务器中的web资源时,就会带着各自的数据去。这样,web资源处理的就是用户各自的数据了
Session:将数据保存在服务端的技术
服务器在运行时可以为每一个用户的浏览器创建一个其独享的session对象,由于session为用户浏览器独享,所以用户在访问服务器的web资源时,可以把各自的数据放在各自的session中,当用户再去访问服务器中的其它web资源时,其它web资源再从用户各自的session中取出数据为用户服务
「原理:」
服务端创建cookie对象 发送cookie信息到浏览器
例如,响应头中增加 Set-Cookie: name=zhangsan
浏览器将得到的cookie信息保存在浏览器端 通过浏览器下次访问web应用时,请求数据中会带上 cookie信息
例如,请求头中 Cookie: name=zhangsan
服务器端收到浏览器发送的cookie信息
「使用:」
Cookie(String name, String value) 创建cookie对象的构造方法,value是字符串类型,不能使用中文
public String getName() 取得Cookie的名字
public String getValue() 取得Cookie的值
public void setValue(String newValue)设置Cookie的值
public void setMaxAge(int expiry)设置Cookie的最大保存时间,单位秒,即cookie的有效期,默认会话结束,cookie失效
public int getMaxAge()获取Cookies的有效期
public void setPath(String uri) 设置cookie的有效路径,比如把cookie的有效路径设置为"/aaa",那么浏览器访问"aaa"的web资源时,就会带上cookie
public String getPath() 获取cookie的有效路径
代码演示:
public void cookie1(HttpServletRequest request, HttpServletResponse response){
//创建cookie对象
Cookie cookie = new Cookie("name", "zhangsan");
//如果有相同name值得cookie,后一个会替换前一个cookie内容
//Cookie cookie1 = new Cookie("name", "lisi");
//cookie中的name和value都是字符串类型
//Cookie cookie1 = new Cookie("age", "10");
//cookie中不能出现中文
//Cookie cookie1 = new Cookie("name", "张三");
//通过响应将cookie数据发送到浏览器端
response.addCookie(cookie);
//response.addCookie(cookie1);
//获取请求中的cookie数据
//ctrl + 1,Enter
//ctrl + 2, l
Cookie[] cookies = request.getCookies();
if(cookies != null){
for (Cookie cookie2 : cookies) {
// 获取cookie的name和对应的value
System.out.println(cookie2.getName());
System.out.println(cookie2.getValue());
}
}
}
public void cookieTime(HttpServletRequest request, HttpServletResponse response){
Cookie cookie = new Cookie("name", "zhangsan");
//设置cookie的过期时间
//单位秒
// cookie.setMaxAge(60 * 60);
//可以实现删除已有的cookie
// cookie.setMaxAge(0);
//过期时间:会话结束时
cookie.setMaxAge(-1);
response.addCookie(cookie);
}
public void cookiePath(HttpServletRequest request, HttpServletResponse response){
Cookie cookie = new Cookie("name", "lisi");
//路径不同,name相同,不会覆盖原来的cookie
//实现将cookie传给另一个web应用
//设置cookie的路径
cookie.setPath("/W08_3_T");
cookie.setMaxAge(60 * 60);
response.addCookie(cookie);
}public void cookie1(HttpServletRequest request, HttpServletResponse response){
//创建cookie对象
Cookie cookie = new Cookie("name", "zhangsan");
//如果有相同name值得cookie,后一个会替换前一个cookie内容
//Cookie cookie1 = new Cookie("name", "lisi");
//cookie中的name和value都是字符串类型
//Cookie cookie1 = new Cookie("age", "10");
//cookie中不能出现中文
//Cookie cookie1 = new Cookie("name", "张三");
//通过响应将cookie数据发送到浏览器端
response.addCookie(cookie);
//response.addCookie(cookie1);
//获取请求中的cookie数据
//ctrl + 1,Enter
//ctrl + 2, l
Cookie[] cookies = request.getCookies();
if(cookies != null){
for (Cookie cookie2 : cookies) {
// 获取cookie的name和对应的value
System.out.println(cookie2.getName());
System.out.println(cookie2.getValue());
}
}
}
public void cookieTime(HttpServletRequest request, HttpServletResponse response){
Cookie cookie = new Cookie("name", "zhangsan");
//设置cookie的过期时间
//单位秒
// cookie.setMaxAge(60 * 60);
//可以实现删除已有的cookie
// cookie.setMaxAge(0);
//过期时间:会话结束时
cookie.setMaxAge(-1);
response.addCookie(cookie);
}
public void cookiePath(HttpServletRequest request, HttpServletResponse response){
Cookie cookie = new Cookie("name", "lisi");
//路径不同,name相同,不会覆盖原来的cookie
//实现将cookie传给另一个web应用
//设置cookie的路径
cookie.setPath("/W08_3_T");
cookie.setMaxAge(60 * 60);
response.addCookie(cookie);
}例如,请求头中 Cookie: name=zhangsan
「特点:」
一个cookie不能超过4k; 一个浏览器最多存300个cookie; 一个站点最多存20个cookie; cookie只能存字符串数据,不支持中文
「简介:」
在Web开发中,服务器可以为每个用户浏览器创建一个会话对象(session对象),可用于保存客户端用户的访问状态。 注意:一个浏览器独占一个session对象(默认情况下) Session依赖于cookie Session也属于域对象
「原理:」
1)访问应用时,如果创建session,每个session对象会分配一个id,称为JSESSIONID
2)服务器会将session的id以cookie的形式发送给浏览器端,浏览器端会存储该id
3)以后再访问应用时,浏览器将session的id发送给服务端
4)服务端收到session的id后,会根据id查找对应的session对象,如果没有找到,创建一个新的session对象
「使用:」
常用方法:
request.getSession() 获取session对象,如果不存在,创建
public void setAttribute(String name,String value)设定指定名字的属性的值,并将它添加到session会话范围内
public Object getAttribute(String name)在会话范围内获取指定名字的属性的值,返回值类型为object,如果该属性不存在,则返回null。
public void removeAttribute(String name),删除指定名字的session属性
public void invalidate(),使session失效。可以立即使当前会话失效,原来会话中存储的所有对象都不能再被访问。
public String getId( ),获取当前的会话ID。每个会话在服务器端都存在一个唯一的标示sessionID,session对象发送到浏览器的唯一数据就是sessionID,它一般存储在cookie中。
public void setMaxInactiveInterval(int interval) 设置会话的最大持续时间,单位是秒,负数表明会话永不失效。
public int getMaxInActiveInterval(),获取会话的最大持续时间。
代码演示:
//会根据session的id 查找服务端是否有对应的session对象,如果有,返回session对象;如果没有创建session对象
//创建session对象后,会将JSESSIONID以cookie的形式存在浏览器端
HttpSession session = request.getSession();
//session的id,唯一的 JSESSIONID
String id = session.getId();
System.out.println(id);
//默认情况下,JSESSIONID的过期时间是会话结束时,为了能正常访问session对象,需要手动将JSESSIONID存储到cookie中,并设置一个合适的过期时间
Cookie cookie = new Cookie("JSESSIONID", id);
cookie.setMaxAge(30 * 60);
response.addCookie(cookie);
//设置session的过期时间,单位秒,默认30分钟
session.setMaxInactiveInterval(10 * 60);
//向session域中存入数据
session.setAttribute("name", "张三");
-----------------------
//根据JSESSIONID找session对象,如果存在,返回;如果不存在,返回null
HttpSession session = request.getSession(false);
if(session != null){
//获取域中的数据
System.out.println(session.getAttribute("name"));
//获取域中所有的属性值
//session.getAttributeNames();
}在Web开发中,服务器可以为每个用户浏览器创建一个会话对象(session对象),可用于保存客户端用户的访问状态
访问一个资源时,如果没有登录过,跳转到登录界面。如果登录过,在一定时间内,可以直接访问该资源
「禁用cookie的处理:」
session依赖cookie,因为服务器需要根据sessionId,然后找到客户端的session对象,如果浏览器禁用了cookie,就需要对URL进行重写,这样服务器收到的请求就会带有sessionId 使用方法:
response.encodeURL(String url)
它会判断客户端浏览器是否禁用了cookie,如果禁用了,那么这个方法会在URL后面追加jsessionId,否则不会追加
「什么是JSP:」
Java Server Pages,它和servlet技术一样,是Java中用于开发动态Web资源的技术
Jsp相当于Java + html
JSP的最大的特点在于,写jsp就像在写html。相比html而言,html只能为用户提供静态数据,而Jsp技术允许在页面中嵌套java代码,为用户提供动态数据。相比servlet而言,servlet很难对数据进行排版,而jsp除了可以用java代码产生动态数据的同时,也很容易对数据进行排版。
不管是JSP还是Servlet,虽然都可以用于开发动态web资源。但由于这2门技术各自的特点,在长期的软件实践中,人们逐渐把servlet作为web应用中的控制器组件来使用,而把JSP技术作为数据显示模板来使用。
原因在于,程序的数据通常要美化后再输出,让jsp既用java代码产生动态数据,又做美化会导致页面难以维护;让servlet既产生数据,又在里面嵌套html代码美化数据,同样也会导致程序可读性差,难以维护。
因此最好的办法就是根据这两门技术的特点,让它们各自负责各的,servlet只负责响应请求产生数据,并把数据通过转发技术带给jsp,数据的显示jsp来做。
「JSP执行过程:」
JSP本质上就是servlet
第一次访问jsp文件,会在tomcat的work目录下生成对应的java文件,然后编译为字节码文件,执行jsp时,其实就是执行对应的class文件,并执行其中的相关方法 再次访问该jsp时,直接从work下找到对应的class文件,并执行
「JSP脚本:」
<%
Java语句
%>
注意:JSP脚本中只能出现java代码,不能出现其它模板元素(html相关标签), JSP引擎在翻译JSP页面中,会将JSP脚本中的Java代码将被原封不动地放到Servlet的_jspService方法中。
JSP脚本片断中的Java代码必须严格遵循Java语法
代码演示:
<!-- jsp脚本,脚本中声明的变量放在_jspService方法中 -->
<%
int num = 10;
String name = "zhangsan";
System.out.println(name);
Date date = new Date();
SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
String dateStr = sdf.format(date);
System.out.println(dateStr);
%>
单个脚本中的Java语句可以是不完整的,但是,多个脚本片断组合后的结果必须是完整的Java语句,例如:
<%
for (int i=1; i<5; i++)
{
%>
<H1>hahahaha</H1>
<%
}
%>
<%
for(int i = 1; i <= 6; i++){
response.getWriter().write("<h" + i + ">标题" + i + "<h" + i + "/>");
}
%>
<%
for(int i = 1; i <= 6; i++){
%>
<h<%=i %>>标题<%=i %></h<%=i %>>
<%
}
%>
「JSP表达式:」
<%=java的表达式 %>
当前时间:<%= new java.util.Date() %>
JSP引擎在翻译脚本表达式时,会将程序数据转成字符串,然后在相应位置用out.write(…) 将数据输给客户端。
注意:JSP脚本表达式中的变量或表达式后面不能有分号(;)。
代码演示:
<!-- jsp表达式 -->
<!-- 相当于out.print(name.length() ); -->
<%=name %>
<%=name.length() %>
<%!
定义变量、方法等
%>
Jsp声明中的java代码被翻译到_jspService方法的外面,即成为实例变量。JSP声明可用于定义JSP页面转换成的Servlet程序的静态代码块、成员变量和方法 。
代码演示:
<!-- jsp声明 -->
<%!
//相当于成员变量
private int aaa = 10;
//定义了一个方法
public int add(int a, int b){
return a + b;
}
int sum = add(10, 12);
//System.out.println(sum);
%>
「JSP注释:」
<%-- 注释的内容 --%>
「简介:」
JSP的指令并不直接产生任何可见输出,而只是告诉引擎如何处理JSP页面中的其余部分 JSP指令的基本语法格式:
<%@ 指令 属性名="值" %>
「include指令 理解:」
include指令用于引入其它JSP页面,如果使用include指令引入了其它JSP页面,JSP引擎将把这两个JSP翻译成一个servlet。所以include指令引入通常也称之为静态引入。
语法:
<%@ include file="relativeURL"%>
其中的file属性用于指定被引入文件的相对路径。file属性的设置值必须使用相对路径,如果以“/”开头,表示相对于当前WEB应用程序的根目录,否则,表示相对于当前文件。
注意:
被引入的文件必须遵循JSP语法。
被引入的文件可以使用任意的扩展名,即使其扩展名是html,JSP引擎也会按照处理jsp页面的方式处理它里面的内容。
由于使用include指令将会涉及到2个JSP页面,并会把2个JSP翻译成一个servlet,所以这2个JSP页面的指令不能冲突(pageEncoding和导包除外)。
代码演示:
<!-- include指令 ,相当于将多个文件合为一个文件,转换为一个java文件-->
<%@ include file="header.jsp" %>
「page指令 理解:」
用于定义JSP页面的各种属性,无论page指令出现在JSP页面中的什么地方,它作用的都是整个JSP页面,为了保持程序的可读性和遵循良好的编程习惯,page指令最好是放在整个JSP页面的起始位置。
page指令的完整语法:
<%@ page
[ language="java" ]
[ import="{package.class | package.*}, ..." ]
[ session="true | false" ]
[ buffer="none | 8kb | sizekb" ]
[ autoFlush="true | false" ]
[ isThreadSafe="true | false" ]
[ info="text" ]
[ errorPage="relative_url" ]
[ isErrorPage="true | false" ]
[ contentType="mimeType [ ;charset=characterSet ]" | "text/html ; charset=ISO-8859-1" ]
[ pageEncoding="characterSet | ISO-8859-1" ]
[ isELIgnored="true | false" ]
代码演示:
<%--
page中
session值默认为true,表示会自动创建一个session对象。当不需要时,可以设置为false
pageEncoding jsp文件本身的编码方式
contentType 服务器发送给浏览器的内容的编码
--%>
<%@ page language="java"
import="java.util.*"
contentType="text/html; charset=UTF-8"
pageEncoding="UTF-8"
session="false"
%>
「taglib指令 了解:」
用于在JSP页面中导入标签库
「原理:」
每个JSP 页面在第一次被访问时,WEB容器都会把请求交给JSP引擎(即一个Java程序)去处理。JSP引擎先将JSP翻译成一个_jspServlet(实质上也是一个servlet) ,然后按照servlet的调用方式进行调用。
由于JSP第一次访问时会翻译成servlet,所以第一次访问通常会比较慢,但第二次访问,JSP引擎如果发现JSP没有变化,就不再翻译,而是直接调用,所以程序的执行效率不会受到影响。
JSP引擎在调用JSP对应的_jspService时,会传递或创建9个与web开发相关的对象供_jspService使用。JSP技术的设计者为便于开发人员在编写JSP页面时获得这些web对象的引用,特意定义了9个相应的变量,开发人员在JSP页面中通过这些变量就可以快速获得这9大对象的引用。
「九大内置对象 了解:」
request HttpServletRequest
response HttpServletResponse
application ServletContext
session HttpSession
out JspWriter
pageContext PageContext
exception Throwable
config ServletConfig
page Object(this) 较少使用
「pageContext:」
pageContext对象是JSP技术中最重要的一个对象,它代表当前JSP页面的运行环境,这个对象不仅封装了对其它8大隐式对象的引用,它自身还是一个域对象,可以用来保存数据。并且,这个对象还封装了web开发中经常涉及到的一些常用操作,例如包含和跳转其它资源、检索其它域对象中的属性等
基本语法:
<%
//可以获取其他内置对象
pageContext.getRequest();
pageContext.getResponse();
pageContext.getServletContext();
pageContext.getSession();
pageContext.getServletConfig();
pageContext.getException();
pageContext.getOut();
pageContext.getPage();
%>
代码演示:
<%
//pageContext 域对象
//pageContext.setAttribute("msg", "page_msg");
//第三个参数,指定向哪个域中设置数据
//四个域对象
pageContext.setAttribute("msg", "page_msg", PageContext.PAGE_SCOPE);
pageContext.setAttribute("msg", "request_msg", PageContext.REQUEST_SCOPE);
pageContext.setAttribute("msg", "session_msg", PageContext.SESSION_SCOPE);
pageContext.setAttribute("msg", "application_msg", PageContext.APPLICATION_SCOPE);
%>
「四个域对象:」
request
session
application
pageContext
基本语法:
<%
System.out.println(pageContext.getAttribute("msg"));
System.out.println(request.getAttribute("msg"));
System.out.println(session.getAttribute("msg"));
System.out.println(application.getAttribute("msg"));
//从域中查找属性
//pageContext->reqeust->session->application
System.out.println(pageContext.findAttribute("msg"));
%>
「获取数据:」
EL表达式主要用于替换JSP页面中的脚本表达式,以从各种类型的域中检索java对象、获取数据。
使用EL表达式获取数据语法:${标识符}
从指定域中获取数据 获取对象属性 获取集合中值
「表达式运算:」
利用El表达式可以在JSP页面中执行一些基本的关系运算】逻辑运输和算术运算,以在JSP页面中完成一些简单的逻辑运算
「内置对象:」
EL表达式定义了一些内置(隐式)对象,利用这些隐式对象,开发人员可以获得对web中常用对象的引用,从而获的这些对象中的数据
「内置动作标签 理解:」
如下:
<jsp:useBean> 创建对象,对象会放到域中,其中:
id 创建的对象名称
class 类名
<jsp:setProperty>设置对象的属性值,其中:
name 表示对象的名称
property 属性名
value 对应的值
<jsp:getProperty> 获取属性值
<jsp:forward page="WEB-INF/aaa.jsp"></jsp:forward> 转发
「jstl标签:」
JSTL(JSP Standard Tag Library,JSP标准标签库)是一个不断完善的开放源代码的JSP标签库,是有apache的jakarta小组来维护的。 主要包括:
核心标签(c:)
JSTL函数(fn:)
格式化标签(fmt:)
数据库标签
XML标签
核心标签库
<c:set> 设置值,会向域中添加数据
<c:out>在jsp中显示值
<c:if> 条件判断
<c:choose> 多条件判断
<c:when>
<c:otherwise>
<c:forEach> 循环遍历
使用步骤:
1)导入jstl-1.2.jar文件
2)在jsp中导入相关的标签库
<!-- 引入核心标签库,前缀必须使用 "c" -->
<%@taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %>
2)格式化标签
<fmt:formatNumber> 数字格式化
<fmt:formatDate> 日期格式化
结语
本篇关于JSP的介绍就先到这里结束了,后续会出更多关于JSP系列更多文章,谢谢大家支持!
如果你觉得浅羽的文章对你有帮助的话,请在微信搜索并关注「 浅羽的IT小屋 」微信公众号,我会在这里分享一下计算机信息知识、理论技术、工具资源、软件介绍、后端开发、面试、工作感想以及一些生活随想等一系列文章。所见所领,皆是生活。慢慢来,努力一点,你我共同成长...
我建立了一个技术群,如果你想了解到更多关于IT行业的技术以及生活中遇到的问题,欢迎小伙伴进群交流,只需添加我的微信,备注进群即可,期待你们的加入。
往期推荐
「万字图文」史上最姨母级Java继承详解
尚能饭否|技术越来越新,我对老朋友jQuery还是一如既往热爱
告别祈祷式编程|单元测试在项目里的正确落地姿势
小记 | 一周上线百万级高并发系统
组件必知必会|那些年我们使用过的轮子—Filter和Proxy
【简历加分】hexo框架搭建个人博客站点,手把手教学。
ES开发指南|如何快速上手ElasticSearch
玩转Redis|学会这10点让你分分钟拿下Redis,满足你的一切疑问