1. Jsp技术:Jsp是一种 html代码+ Java代码 + Jsp页面代码的技术,并且其不用像Servlet哪样须要配置访问路径。css
2. Jsp执行原理:第一次访问服务器web项目中的jsp页面时,会首先把jsp页面翻译成为一个标准Servlet,也就是 .java 的文件,而后将这个java文件编译生成字节码文件,而后再执行。html
3. Jsp中Java代码的编写:java
<%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%> <% String path = request.getContextPath(); String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/"; %> <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"> <html> <head> <base href="<%=basePath%>"> <title>My JSP 'test1.jsp' starting page</title> <meta http-equiv="pragma" content="no-cache"> <meta http-equiv="cache-control" content="no-cache"> <meta http-equiv="expires" content="0"> <meta http-equiv="keywords" content="keyword1,keyword2,keyword3"> <meta http-equiv="description" content="This is my page"> <!-- <link rel="stylesheet" type="text/css" href="styles.css"> --> </head> <body> <%--定义局部变量a,b,c --%> <% int a=10;%> <% int b=10;%> <% int c=a+b;%> <%= "c=" %> <% out.println(c); %> <br> <%--输出3行3列的表格 --%> <table cellspacing="0" cellpadding="0" border="1"> <% for(int i=1;i<=3;i++){%> <tr> <% for(int x=1;x<=3;x++){%> <td> <%=i+","+x %> </td> <%}%> </tr> <%}%> </table> <%--构造LinkedList实例对象并添加元素,而后输出 --%> <% LinkedList<Integer> list=new LinkedList<Integer>(); for(int i=0;i<=10;i++){ list.add(i); } %> <%=list.toString() %></br> <%--输出当前时间信息 --%> <% Date now=new Date();%> <%="当前时间:北京时间" %> <%=now.toLocaleString() %> </body> </html>
1. page指令用于设置Jsp文件属性,形式为<%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%>,其中可使用一个属性,也可使用多个属性,可是只有import属性能够屡次出现,具体经常使用属性以下:web
<error-page> <error-code>404</error-code> <location>/jsp/error.jsp</location> </error-page>
1. 用于页面包含,能够将某些公共页面片断提取出来,用的时候在加入进行组合,指令形式如<%@ include file="" %>,具体经常使用属性以下apache
1. 在 tomcat服务器目录路径+\work\Catalina\localhost\+项目名+\org\apache\jsp\+jsp文件目录下的jsp文件对应的翻译过来的java代码文件,打开这个java代码文件,咱们能够发现一系列的jsp内置对象。内置对象能够在java代码块内直接使用,而exception对象只有isErrorPage属性为true的页面内才能使用数组
好比路径为D:\apache-tomcat-8.5.24\work\Catalina\localhost\Servlet\org\apache\jsp\jsp\test1_jsp.java,打开后代码为浏览器
package org.apache.jsp.jsp; import javax.servlet.*; import javax.servlet.http.*; import javax.servlet.jsp.*; import java.util.*; public final class test1_jsp extends org.apache.jasper.runtime.HttpJspBase implements org.apache.jasper.runtime.JspSourceDependent, org.apache.jasper.runtime.JspSourceImports { //删除部分代码,下面的就是jsp中的内置对象 final javax.servlet.jsp.PageContext pageContext; javax.servlet.http.HttpSession session = null; final javax.servlet.ServletContext application; final javax.servlet.ServletConfig config; javax.servlet.jsp.JspWriter out = null; final java.lang.Object page = this; javax.servlet.jsp.JspWriter _jspx_out = null; javax.servlet.jsp.PageContext _jspx_page_context = null; try { response.setContentType("text/html;charset=UTF-8"); pageContext = _jspxFactory.getPageContext(this, request, response, null, true, 8192, true); _jspx_page_context = pageContext; application = pageContext.getServletContext(); config = pageContext.getServletConfig(); session = pageContext.getSession(); out = pageContext.getOut(); _jspx_out = out; } catch (java.lang.Throwable t) { if (!(t instanceof javax.servlet.jsp.SkipPageException)){ out = _jspx_out; if (out != null && out.getBufferSize() != 0) try { if (response.isCommitted()) { out.flush(); } else { out.clearBuffer(); } } catch (java.io.IOException e) {} if (_jspx_page_context != null) _jspx_page_context.handlePageException(t); else throw new ServletException(t); } } finally { _jspxFactory.releasePageContext(_jspx_page_context); } } }
2. 内置对象用法:tomcat
request:表示HttpServletRequest对象服务器
response:表示HttpServletResponse对象cookie
session:表示HttpSession对象
application:表示ServletContext对象,也就是整个web应用
out:表示JspWriter对象,和response.getWriter()同样,都用于向返回客户端的html页面输出数据。但JspWriter与response.getWriter()获得的字符输出流对象确定不是同一个,JspWriter会将输出的数据先保存在本身的缓冲区中,而后再将缓冲区中的数据发送到response.getWriter()中,再向页面写出,由于只有response.getWriter()才能向页面输出数据。
pageContext:表示PageContext对象,该对象能够获取另外的8个对象,且此对象也是jsp中独有的一个域对象,但其中的数据只能在本页面生效,而且能够向其余的域对象中存取值。方法以下
pageContext.setAttribute(String name,Object value, int scope):将一个属性值,存储到request、session、application三个域对象内,好比 pageContext.setAttribute("name", "vn", pageContext.APPLICATION_SCOPE),就是将属性存储在application域内,其余两个域对象范围用 REQUEST_SCOPE,SESSION_SCOPE 表示,对于PAGE_SCOPE就表示pageContext域对象。
pageContext.getAttribute(name, scope):指定在某个域对象范围内获取属性。好比pageContext.getAttribute("name", pageContext.REQUEST_SCOPE)
pageContext.setAttribute(String name,Object value):向pageContext域对象内存储属性值。
pageContext.getAttribute(name):从pageContext域对象内获取属性值。
pageContext.findAttribute(name):先从pageContext对象自身内部查找属性,而后依次从request、session、application三个域对象中获取属性,找到以后就会返回属性值,不会继续查找。
page:Object(表明当前Servlet对象的Object类型引用)
config:表示ServletConfig对象
exception:表示Throwable对象
JSP中的4个域对象访问范围(由小到大):
pageContext:仅当前jsp页面范围,经过该域可获得其余全部域中的值,默认从自身存储的数据中开始查找,而后范围有小到大访问其余域
request:请求范围,在请求生效的范围中都可访问其中的值
session:整个会话范围
application:整个web应用范围
1. <jsp:forward page="url"></jsp:forward>:转发标签,实现转发功能。page属性指转发到哪一个页面上去,转发的路径的写法和Servlet中同样,写服务器绝对路径(不包含包含项目名称)
2. <jsp:param value="" name=""/>:转发时传递参数,须要被包括在jsp:forward 标签以内。name 指参数的名称, value表明参数的值
3. <jsp:include page=""></jsp:include>:动态包含(和静态包含结果是同样的,原理是不同的),page属性指被包含的页面的路径,动态包含会在源码的文件下,对每一个jsp都生成了.java和.class的文件。
4. 在Jsp中封装数据所使用的标签:经过这些标签能够实例化一个类的对象,一般能够将请求中携带的参数,经过其属性名也就是name与类的属性名称进行对比,相同名称的将会进行自动注入。假如如今有一个请求被一个jsp页面接受到,请求里包括了两个请求参数,分别是
name="vn"&password="123",而有一个用于封装数据的标准JavaBean规范的类以下
package servlet.JSP; /** * @ClassName:User * @Description:一个符合JavaBean规范的数据封装工具类,其必须知足三大规范 * 必须有无参构造器 * 实例变量必须是私有的,对外不可见 * 对于每一个实例变量都必须有一个public修饰的get与set方法,来实现外界对实例变量的读写 * @author: * @date:2018年9月11日 */ public class User { private String name; private String password; public String getName() { return name; } public void setName(String name) { this.name = name; } public String getPassword() { return password; } public void setPassword(String password) { this.password = password; } }
<jsp:useBean id="u" class="servlet.JSP.User" scope="request"></jsp:useBean> <jsp:getProperty property="name" name="u"/> <jsp:setProperty property="name" name="u" param="name" value="uzi"/>
1. 内省技术的简单使用
import java.beans.BeanInfo; import java.beans.IntrospectionException; import java.beans.Introspector; import java.beans.MethodDescriptor; import java.beans.PropertyDescriptor; import java.lang.reflect.InvocationTargetException; import java.lang.reflect.Method; import org.junit.Test; /** * @ClassName:TestIntrospector * @Description:Java内省技术,其底层经过反射技术实现。主要的类为Introspector * @author: * @date:2018年9月11日 */ public class TestIntrospector { @Test public void testIntrospector() throws IntrospectionException, IllegalAccessException, IllegalArgumentException, InvocationTargetException{ User user=new User(); //测试内省,获取User这个JavaBean类的信息 BeanInfo info=Introspector.getBeanInfo(user.getClass()); //获取这个JavaBean类的全部属性信息 PropertyDescriptor[] pros= info.getPropertyDescriptors(); for(PropertyDescriptor p:pros){ if(!"class".equals(p.getName())){ System.out.println(p.getName());//获取属性名 Method method1=p.getWriteMethod();//获取该属性的写入方法,也就是set方法 method1.invoke(user, "vn");//调用每一个属性的set方法 } } for(PropertyDescriptor p:pros){ if(!"class".equals(p.getName())){ System.out.println(p.getName());//获取属性名 Method method2=p.getReadMethod();//获取该属性的读取方法,也就是get方法 System.out.println(method2.invoke(user, null));//调用每一个属性的读取方法,并输出 } } //获取这个JavaBean类的全部方法信息 MethodDescriptor[] meths=info.getMethodDescriptors(); for(MethodDescriptor m:meths){ System.out.println(m.getName());//获取方法名 Method method=m.getMethod();//转换为反射中的Method类 } } }
2. 经过内省技术在一个Servlet类里实现数据封装:
import java.beans.BeanInfo; import java.beans.IntrospectionException; import java.beans.Introspector; import java.beans.PropertyDescriptor; import java.io.IOException; import java.lang.reflect.InvocationTargetException; import java.lang.reflect.Method; import java.util.Map; import javax.servlet.ServletException; import javax.servlet.annotation.WebServlet; import javax.servlet.http.HttpServlet; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; /** * Servlet implementation class UserServlet */ @WebServlet("/UserServlet") public class UserServlet extends HttpServlet { private static final long serialVersionUID = 1L; protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { Map<String, String[]> parameter = request.getParameterMap(); User user=new User(); try { populate(user,parameter); } catch (IllegalAccessException | IllegalArgumentException | InvocationTargetException | IntrospectionException e) { // TODO Auto-generated catch block e.printStackTrace(); } } private static void populate(Object o,Map<String, String[]> map) throws IntrospectionException, IllegalAccessException, IllegalArgumentException, InvocationTargetException{ //获取数据封装类型信息 BeanInfo info=Introspector.getBeanInfo(o.getClass()); //获取类中全部属性的信息 PropertyDescriptor[] pds=info.getPropertyDescriptors(); for(PropertyDescriptor p:pds){ if(!"class".equals(p) && map.containsKey(p.getName())){ Method m=p.getWriteMethod(); m.invoke(o, map.get(p.getName())[0]); } } }
3. 使用BeanUtils工具类来封装数据到一个对象中:
public class TestBeanUtils { @Test /** * @Title:testBeanUtils * @Description:BeanUtils的简单使用,首先须要导入commons-logging和commons-beanutils两个jar包 * @author: * @date: * @param: * @return:void * @throws InvocationTargetException * @throws IllegalAccessException * @throws: */ public void testBeanUtils() throws IllegalAccessException, InvocationTargetException{ User user=new User(); //模拟从请求中取得的参数Map Map<String, String[]> map=new HashMap<String, String[]>(10); String[] s1={"vn"}; String[] s2={"12345"}; map.put("name", s1); map.put("password", s2); //使用工具类封装数据,将数据封装到user对象中 BeanUtils.populate(user, map); //查看是否封装成功 System.out.println(user.getName()); System.out.println(user.getPassword()); } }
4. 编写代码,实现将字符串转换为本身想要的数据类型(以将字符串转换为日期类型数据为例):
import java.text.ParseException; import java.text.SimpleDateFormat; import java.util.Date; import org.apache.commons.beanutils.Converter; public class StringToDateConverter implements Converter{ /** * @Title:convert * @Description:将字符串转为日期类型 * @param c * @param o 须要被转换的数据 * @return * @see org.apache.commons.beanutils.Converter#convert(java.lang.Class, java.lang.Object) */ @Override public Object convert(Class c, Object o) { String str=(String)o; SimpleDateFormat f=new SimpleDateFormat("yyyy-MM-dd");//字符串格式指定 Date date=null; try { date=f.parse(str); } catch (ParseException e) { // TODO Auto-generated catch block e.printStackTrace(); } return date; } } //编写一个包含Date类型属性的JavaBean import java.util.Date; public class Birthday{ Date birth; public Date getBirth() { return birth; } public void setBirth(Date birth) { this.birth = birth; } } //测试数据封装 public class TestBeanUtils { @Test public void testMyConverter() throws IllegalAccessException, InvocationTargetException{ //将本身的转换器注册到ConvertUtils中,第一个参数是本身编写的转换器的实例对象,第二个参数是数据 //转换的目标类型,当把数据封装到某一个对象的时候,若是对象中有Date类的属性,就会使用该转换器 ConvertUtils.register(new StringToDateConverter(), Date.class); Birthday day=new Birthday(); Map<String, String[]> map=new HashMap<String, String[]>(); String[] s=new String[]{"1997-10-24"}; map.put("birth", s); BeanUtils.populate(day, map); System.out.println(day.getBirth().toString()); } }
Jsp内置的一种表达式,EL表达式能够获取数据、执行运算、获取web开发经常使用对象、调用Java方法四个用途。
语法形式为: ${ 表达式内容 }
1. 获取域对象中的属性值:从四个域中取属性值,默认从范围最小的域中开始取值,即顺序为:pageContext->request->session->application
<%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%> <% String path = request.getContextPath(); String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/"; %> <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"> <html> <head> <base href="<%=basePath%>"> <title>My JSP 'demo5.jsp' starting page</title> </head> <% pageContext.setAttribute("username", "1"); request.setAttribute("username", "2"); session.setAttribute("username", "3"); application.setAttribute("username", "4"); %> <body> <h5> ${username} <!-- 从四个域中取属性值,默认从范围最小的域中开始取值,即顺序为:pageContext->request->session->application --> </h5> <div>存取数组:<br/> <% int[] arr={1,2,3,4}; pageContext.setAttribute("arr", arr); %> <p>${ arr[0] }</p> </div> <div>存取集合:<br/> <% List<String> list=new ArrayList<String>(); list.add("1"); list.add("2"); list.add("3"); pageContext.setAttribute("list", list); %> <p>${ list[0] }</p> </div> <div>存取Map集合:<br/> <% Map<String,String> map=new HashMap<String,String>(); map.put("1", "1"); map.put("2", "2"); map.put("3", "3"); map.put("a.a", "4"); map.put("a", "a"); pageContext.setAttribute("map", map); %> <p>${ map.a }</p> <p>${ map["1"] }</p> <p>${ map["a.a"] }</p><!-- 特殊字符使用中括号 --> </div> </body> </html>
2. EL表达式经常使用的部份内部隐藏对象:
3. EL表达式支持运算:加减乘除,关系运算,逻辑运算:
<div>EL表达式支持运算:加减乘除,关系运算,逻辑运算<br/> <%pageContext.setAttribute("n1", 10); request.setAttribute("n2", 20); session.setAttribute("n3", 30); application.setAttribute("n4", 40); %> <p> <!-- ${n1 }就至关于 ${ pageContext.n1 } --> ${n1 }+${n2 }=${n1+n2 } ${n1==n2 }或${n1 eq n2 } ${true && true } </p> </div>
1. 引入JSTL核心标签库:<%@taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c"%>
<%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%> <% String path = request.getContextPath(); String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/"; %> <%--JSTL标签库:jstl,即jsp内置标签,其运行在服务器中并与html标签区分,用于简化标签,去掉<%%> --%> <%--引入JSTL标签库: --%> <%@taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c"%> <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"> <html> <head> <base href="<%=basePath%>"> <title>My JSP 'demo6.jsp' starting page</title> </head> <%pageContext.setAttribute("n", 10); %> <body> <div> <c:if test="${n==10 }"> <p>test=true</p> </c:if> <c:choose></c:choose> </div> </body> </html>
2. 使用方法:<c:标签名 属性>输出内容(能够是HTML标签)</c:标签名>
3. 经常使用标签库:
1. <c:out value="" default="" escapeXml="true"></c:out>:用于查找jsp四个域对象中的属性,并输出值
2. <c:set property="" scope="" value="" target="" var=""></c:set>:用于设置jsp四个域对象中存储属性,也能够向JavaBean或Map对象中设置属性
3. <c:remove var="" scope=""/>:用于删除域对象中的属性
4. <c:catch var=""></c:catch>:用于捕获发生的异常,var属性表示若是发生了异常,就把异常信息保存到var指定的变量中。
5. <c:if test="" scope="" var=""></c:if>:用于判断,标签间能够写当判断为true时,则输出JSTL标签间的内容,能够是HTML标签
6. <c:choose><c:when test=""></c:when><c:otherwise></c:otherwise></c:choose>:也用于判断,至关于if-else语句
7. <c:forEach items="" begin="" end="" step="" var="" varStatus=""></c:forEach>:用于遍历数组或迭代数据集合
8. <c:url context="" scope="" value="" var=""></c:url>:与提交的地址相关,好比超连接等
<%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%> <% String path = request.getContextPath(); String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/"; %> <%--JSTL标签库:jstl,即jsp内置标签,其运行在服务器中并与html标签区分,用于简化标签,去掉<%%> --%> <%--引入JSTL标签库: --%> <%@taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c"%> <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"> <html> <head> <base href="<%=basePath%>"> <title>My JSP 'demo6.jsp' starting page</title> </head> <%pageContext.setAttribute("n", 10); %> <body> <!-- JSTL核心标签库中的out标签,用于查找jsp四个域对象中的属性,并输出值 --> <div> <c:out value="${n}" default="查找失败"></c:out> <c:out value="<a href='http://www.baidu.com'>百度</a>" escapeXml="false"></c:out> </div> <!-- set标签,用于向jsp四个域对象中存储更改属性,也能够向JavaBean或Map对象中设置属性 --> <div> <c:set property="" scope="session" value="vn" target="" var="name"></c:set> <c:out value="${name }"></c:out> </div> <!-- remove标签,用于向jsp四个域对象中存储属性 --> <div> <c:remove var="n" scope=""/> </div> <!-- catch标签,用于捕获发生的异常 --> <div> <c:catch var="e"> <%!int a=10/0; %> </c:catch> <c:out value="${e.message }"></c:out> </div> <!-- if标签 ,用于判断条件--> <div> <c:if test="${n==10 }" scope="session" var="flag"> <p>test=true</p> </c:if> <c:out value="${i }"></c:out> </div> <!-- choose标签 ,用于判断条件--> <c:choose> <c:when test="${i }"> 条件为true </c:when> <c:when test="${i == null }"> 条件为空 </c:when> <c:otherwise> 条件为false </c:otherwise> </c:choose> <!-- forEach标签,用于遍历数组或迭代数据集合 --> <div> <%! int[] arr={1,23,4,5,12}; List<Integer> list=new ArrayList<Integer>(); list.add(2); list.add(4); list.add(52); list.add(22); request.setAttribute("list", list); %> <!-- 至关于加强for循环 ,即 for(int i:list){} --> <c:forEach items="${list }" var="i"> <c:out value="${i }"></c:out> </c:forEach> <!-- 普通for循环 ,即 for(int i=0;i<10;i++){} --> <c:forEach items="${list }" var="i" begin="1" end="3" step="1" varStatus="status"> <c:out value="${i }"></c:out> <c:out value="${step }"></c:out> <c:out value="${status.count }"></c:out> </c:forEach> </div> <!-- url标签,value中的路径能够不写项目名--> <div> <c:url value="/jsp/demo5.jsp"></c:url> </div> </body> </html>
1. EL函数:用来操做字符串,须要引入<%@taglib uri="http://java.sun.com/jsp/jstl/functions" prefix="fn"%>
2. 用法:
<!-- EL 函数标签 --> <%pageContext.setAttribute("str", "abcdefg"); %> ${fn:contains(str,"ab") } ${fn:indexOf(str,"c") }