JSP入门

JSP简介

  • 概念:Java Server Pages: java服务器端页面
    • 可以理解为:一个特殊的页面,其中既可以指定定义html标签,又可以定义java代码
    • 用于简化书写
  • JSP的执行过程
    • 翻译阶段:JSP文件会被web容器中的JSP引擎转换成Java源码(继承了HttpServlet的Servlet)
    • 编译阶段:Java源码会被遍历成可执行的字节码
    • 执行阶段:容器接受了客户端的请求后,执行编译成字节码的JSP文件;处理完请求后,容器把生成的页面反馈给客户端进行显示
  • 原理:JSP本质上就是一个Servlet

jsp页面7大元素构成

alt

1.JSP指令

  • 用于配置JSP页面,导入资源文件
    <%@ 指令名称 属性名1=属性值1 属性名2=属性值2 ... %>
    
  1. page :配置JSP页面的
    • contentType:等同于response.setContentType()
      1. 设置响应体的mime类型以及字符集
      2. 设置当前jsp页面的编码(只能是高级的IDE才能生效,如果使用低级工具,则需要设置pageEncoding属性设置当前页面的字符集)
    • import:导包
    • errorPage:当前页面发生异常后,会自动跳转到指定的错误页面
    • isErrorPage:标识当前也是是否是错误页面。
      • true:是,可以使用内置对象exception
      • false:否。默认值。不可以使用内置对象exception
  2. include :页面包含的。导入页面的资源文件
    • <%@include file="top.jsp"%>
  3. taglib :导入资源
  • 如导入JSTL,先导入相关jar包 javax.servlet.jsp.jstl.jar jstl-impl.jar

2.JSP表达式

  • 在JSP页面中执行的表达式
    <%= 表达式 %><!--表达式不以分号结束-->
    

3.JSP脚本

  • 在jsp中执行java代码
    <% Java代码 %>
    

4.JSP声明

  • 在JSP页面中定义变量或者方法
    <%! Java代码 %>
    

5.JSP注释

  • html注释:只能注释html代码片段,前台查看源代码可见
    <!-- -->
    
  • jsp注释:可以注释所有,前台查看源代码不可见
    <%-- --%>
    

6.JSP动作

语法 描述
jsp:include 在页面被请求的时候引入一个文件。
jsp:useBean 寻找或者实例化一个JavaBean。
jsp:setProperty 设置JavaBean的属性。
jsp:getProperty 输出某个JavaBean的属性。
jsp:forward 把请求转到一个新的页面。
jsp:plugin 根据浏览器类型为Java插件生成OBJECT或EMBED标记。
jsp:element 定义动态XML元素
jsp:attribute 设置动态定义的XML元素属性。
jsp:body 设置动态定义的XML元素内容。
jsp:text 在JSP页面和文档中使用写入文本的模板

JSP9个内置对象

JSP内置对象是web容器创建的一组对象,在jsp页面中不需要创建,直接使用的对象,一共有9个。

变量名 真实类型 作用
pageContext PageContext 当前页面共享数据,还可以获取其他八个内置对象
request HttpServletRequest 一次请求访问的多个资源(转发)
session HttpSession 一次会话的多个请求间
application ServletContext 所有用户间共享数据
response HttpServletResponse 响应对象
page Object 当前页面(Servlet)的对象 this
out JspWriter 输出对象,数据输出到页面上
config ServletConfig Servlet的配置对象
exception Throwable 异常对象
  • 前四个内置对象pageContext,request,session,application为域对象,可用于共享数据。

  • out:字符输出流对象。可以将数据输出到页面上。和response.getWriter()类似

    • response.getWriter()和out.write()的区别:
      • 在tomcat服务器真正给客户端做出响应之前,会先找response缓冲区数据,再找out缓冲区数据。
      • response.getWriter()数据输出永远在out.write()之前

EL表达式

EL表达式简介

  • 概念:Expression Language 表达式语言
  • 作用:替换和简化jsp页面中java代码的编写

EL表达式使用

  • 语法:${表达式}
  • 注意:
    • jsp默认支持el表达式的。如果要忽略el表达式
      1. 设置jsp中page指令中:isELIgnored="true" 忽略当前jsp页面中所有的el表达式
      2. \${表达式} :忽略当前这个el表达式
  • 使用:

运算

  • 运算符:
    1. 算数运算符: + - * /(div) %(mod)
    2. 比较运算符: > < >= <= == !=
    3. 逻辑运算符: &&(and) ||(or) !(not)
    4. 空运算符: empty
      • 功能:用于判断字符串、集合、数组对象是否为null或者长度是否为0
      • ${empty list}:判断字符串、集合、数组对象是否为null或者长度为0
      • ${not empty str}:表示判断字符串、集合、数组对象是否不为null 并且 长度>0
    <%@ page import="java.util.ArrayList" %>
    <%@ page import="java.util.List" %>
    <%@ page contentType="text/html;charset=UTF-8" language="java"  %><%--jsp指令,isELIgnored="true"忽略当前页面EL表达式,做字符串输出--%>
    <html>
    <head>
        <titel>Title</titel>
    </head>
    <body>
        ${3>4}
        \${3>4}<!--忽略单个的EL表达式-->
    <hr>
    <h3>算术运算符</h3>
        ${3+4}<br>
        ${3/4}<br>
        ${3 div 4}<br>
        ${3%4}<br>
        ${3 mod 4}<br>
    <h3>比较运算符</h3>
        ${3 == 4}<br>
    <h3>逻辑运算符</h3>
        ${3 > 4 && 3 < 4 }<br>
        ${3 > 4 and 3 < 4 }<br>
    <h4>empty运算符</h4>
    <%
        String str="";
        request.setAttribute("str",str);
    
        List list =new ArrayList();
        request.setAttribute("list",list);
    %>
    ${empty str}
    ${not empty str}
    </body>
    </html>
    

alt

获取值

  • el表达式只能从域对象中获取值
  1. ${域名称.键名}:从指定域中获取指定键的值

    域名称 对象
    pageScope pageContext
    requestScope request
    sessionScope session
    applicationScope application(ServletContext)
    • 举例:在request域中存储了name=张三
    • 获取:${requestScope.name}
  2. ${键名}:表示依次从最小的域中查找是否有该键对应的值,直到找到为止。 el2.jsp

    <%@ page contentType="text/html;charset=UTF-8" language="java" %>
    <html>
    <head>
        <title>获取域中的数据</title>
    </head>
    <body>
        <%
            //在域中存储数据
            session.setAttribute("name","李四");
            request.setAttribute("name","张三");
            session.setAttribute("age","23");
        %>
    <h3>el获取值</h3>
    ${requestScope.name}<%--${域名称.键名}:从指定域中获取指定键的值--%>
    ${sessionScope.age}
    ${sessionScope.haha}<%--即使没有对应的键,也不会报错--%>
    
    ${name}<%--${键名}:表示依次从最小的域中查找是否有该键对应的值,直到找到为止--%>
    </body>
    </html>
    

alt

  1. 获取对象、List集合、Map集合的值
    1. 对象:${域名称.键名.属性名}
      • 本质上会去调用对象的getter方法
    2. List集合:${域名称.键名[索引]}
    3. Map集合:
      • ${域名称.键名.key名称}
      • ${域名称.键名["key名称"]}
    <%@ page import="cn.simplek9.domain.User" %>
    <%@ page import="java.util.*" %>
    <%@ page contentType="text/html;charset=UTF-8" language="java" %>
    <html>
    <head>
        <title>获取数据</title>
    </head>
    <body>
        <%
            User user = new User();//创建一个User对象
            user.setName("张三");//设置对象属性
            user.setAge(23);
            user.setBirthday(new Date());
            request.setAttribute("u",user);//将对象存入request域中
    
            List list = new ArrayList();//创建一个list对象
            list.add("aaa");
            list.add("bbb");
            list.add(user);
            request.setAttribute("list",list);//将list对象存入request域中
    
            Map map = new HashMap();//创建一个map对象
            map.put("sname","李四");
            map.put("gender","男");
            map.put("user",user);
            request.setAttribute("map",map);//将map对象存入request域中
        %>
        <h3>el获取对象中的值</h3>
        ${requestScope.u}<br><%--会返回request域中对象的字符串表示形式--%>
    <%--
        *通过对象的属性来获取
             *setter或getter方法,去掉set或get,将剩余部分,首字母变小写
             *setName  --> Name  --> name
    --%>
        ${requestScope.u.name}<br><%---找对应的getter 方法--%>
        ${u.age}<br><%--保证键名称不相同更方便--%>
        ${u.birthday}<br>
        ${u.birStr}<br>
    
        <h3>el获取List值</h3>
        ${list}<br>
        ${list[0]}<br>
        ${list[1]}<br>
        ${list[10]}<br><%--在EL表达式里角标越界,不会报错,也不显示--%>
        ${list[2].name}
    
        <h3>el获取Map值</h3>
        ${map.gender}<br>
        ${map["gender"]}<br>
    </body>
    </html>
    

alt

隐式对象

  • el表达式中有11个隐式对象
  • pageContext:
    • 获取jsp其他八个内置对象
      • ${pageContext.request.contextPath}:动态获取虚拟目录
    <%@ page contentType="text/html;charset=UTF-8" language="java" %>
    <html>
    <head>
        <title>el隐式对象</title>
    </head>
    <body>
        ${pageContext.request}<br>
        <h4>动态获取虚拟目录</h4>
        ${pageContext.request.contextPath}
    </body>
    </html>
    

alt

JSTL

JSTL简介

  1. 概念:JavaServer Pages Tag Library JSP标准标签库

    • 是由Apache组织提供的开源的免费的jsp标签 <标签>
  2. 作用:用于简化和替换jsp页面上的java代码

JSTL使用步骤

  1. 导入jstl相关jar包 javax.servlet.jsp.jstl.jar jstl-impl.jar
  2. 引入标签库:taglib指令: <%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %>
  3. 使用标签

常用的JSTL标签

  1. if:相当于java代码的if语句
    1. 属性:
      • test 必须属性,接受boolean表达式
        • 如果表达式为true,则显示if标签体内容,如果为false,则不显示标签体内容
        • 一般情况下,test属性值会结合el表达式一起使用
    2. 注意:
      • c:if标签没有else情况,想要else情况,则可以在定义一个c:if标签
    <%@ page import="java.util.List" %>
    <%@ page import="java.util.ArrayList" %>
    <%@ page contentType="text/html;charset=UTF-8" language="java" %>
    <%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core"%>
    <html>
    <head>
        <title>if标签</title>
    </head>
    <body>
        <c:if test="true">
            <h1>我是真的...</h1>
        </c:if>
        <%
            //判断request域中的一个list集合是否为空,如果不为null则显示遍历集合
            List list=new ArrayList();
            list.add("aaaa");
            request.setAttribute("list",list);
    
            request.setAttribute("number",3);
        %>
        <c:if test="${not empty list}">
            遍历集合...
        </c:if>
        <br>
    
        <c:if test="${number%2!=0}">
            ${number}为奇数
        </c:if>
        <c:if test="${number%2 == 0}">
            ${number}为奇数
        </c:if>
    
    </body>
    </html>
    

alt

  1. choose:相当于java代码的switch语句
    1. 使用choose标签声明 相当于switch声明
    2. 使用when标签做判断 相当于case
    3. 使用otherwise标签做其他情况的声明 相当于default
    <%@ page contentType="text/html;charset=UTF-8" language="java" %>
    <%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %>
    <html>
    <head>
        <title>choose标签</title>
    </head>
    <body>
        <%
            request.setAttribute("number",3);
        %>
        <c:choose>
            <c:when test="${number==1}">星期一</c:when>
            <c:when test="${number==2}">星期二</c:when>
            <c:when test="${number==3}">星期三</c:when>
            <c:when test="${number==4}">星期四</c:when>
            <c:when test="${number==5}">星期五</c:when>
            <c:when test="${number==6}">星期六</c:when>
            <c:when test="${number==7}">星期天</c:when>
            <c:otherwise>数字输入有误</c:otherwise>
        </c:choose>
    </body>
    </html>
    

alt

  1. foreach:相当于java代码的for语句
    1. 完成重复的操作 相当于for(int i=0;i<10;i++){}

      属性 含义
      begin 开始值
      end 结束值
      var 临时变量
      step 步长
      varStatus 循环状态对象

      varStatus还有以下属性

      • index:容器中元素的索引,从0开始
      • count:循环次数,从1开始
    2. 遍历容器 相当于for(User user : list){}

      属性 含义
      items 容器对象(相当于上面list)
      var 容器中元素的临时对象(相当于上面user)
      varStatus 循环状态对象

      varStatus还有以下属性

      • index:容器中元素的索引,从0开始
      • count:循环次数,从1开始
    <%@ page import="java.util.List" %>
    <%@ page import="java.util.ArrayList" %>
    <%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %>
    <%@ page contentType="text/html;charset=UTF-8" language="java" %>
    <html>
    <head>
        <title>foreach标签</title>
    </head>
    <body>
    <c:forEach begin="1" end="10" var="i" step="3" varStatus="s">
        ${i} ${s.index} ${s.count}<br><%--打印索引和循环次数--%>
    </c:forEach>
    <hr>
    <%
        List list=new ArrayList();
        list.add("aaa");
        list.add("bbb");
        list.add("ccc");
    
        request.setAttribute("list",list);
    %>
    <c:forEach items="${list}" var="str" varStatus="s">
        ${s.index} ${s.count} ${str}<br>
    </c:forEach>
    </body>
    </html>
    

alt

MVC开发模式

  • jsp演变历史
    1. 早期只有servlet,只能使用response输出标签数据,非常麻烦
    2. 后来有jsp,简化了Servlet的开发,如果过度使用jsp,在jsp中既写大量的java代码,又写html,造成难于维护,难于分工协作
    3. 再后来,java的web开发,借鉴mvc开发模式,使得程序的设计更加合理性

MVC 模式代表 Model-View-Controller(模型-视图-控制器) 模式。这种模式用于应用程序的分层开发。

  • Model(模型) 模型代表一个存取数据的对象或 JAVA POJO。它也可以带有逻辑,在数据变化时更新控制器。
  • View(视图) 视图代表模型包含的数据的可视化。
  • Controller(控制器) 控制器作用于模型和视图上。它控制数据流向模型对象,并在数据变化时更新视图。它使视图与模型分离开。

MVC 模式以JSP+JavaBean+Servlet模式为例,servlet负责处理用户请求,jsp负责数据显示,javabean负责封装数据。

  • 优点:
    1. 耦合性低,方便维护,可以利于分工协作
    2. 重用性高
  • 缺点:
    1. 使得项目架构变得复杂,对开发人员要求高

三层架构

  • 三层架构是一个分层式的软件体系架构设计理念。
  1. 界面层(表示层):用户看的得界面。用户可以通过界面上的组件和服务器进行交互
  2. 业务逻辑层:处理业务逻辑的。
  3. 数据访问层:操作数据存储文件。

alt

列表查询实例

alt