JavaWeb从0开始学(二)-----JSP基本语法与编译指令

时间:2023-03-08 17:30:03
JavaWeb从0开始学(二)-----JSP基本语法与编译指令

在上一节中我们学习了如何搭建一个简单的Web应用,并且已经知晓了一个JSP页面主要由静态的HTML内容和动态的Java脚本共同组成。JSP的基本语法共有JSP注释、JSP声明、输出JSP表达式与JSP程序片段四种,用来控制动态向页面输出内容的。现在我们就一起来学习JSP的这几种基本语法。

(一)JSP的基本语法

1.JSP注释语句

JSP注释用于标注在程序开发过程中的开发提示,不会输出到客户端。

JSP注释的格式如下:

<%--我是JSP注释的内容--%>

其实我们可以发现JSP的注释与HTML的注释十分类似,下面是HTML的注释格式:

<!--我是HTML注释的内容-->

下面我们通过一个例子来体会一下这二者在JSP页面中的区别:

<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<!DOCTYPE html>
<html>
  <head>
    <title>注释实例</title>
  </head>
  <body>
        下面的是JSP注释:<br>
        <%--我是JSP注释--%><br>
        下面的是HTML注释:<br>
        <!--我是HTML注释-->
  </body>
</html>

在浏览器中浏览该页面,并且右击查看网页源代码,我们会看到JSP注释消失了,如下图所示:

JavaWeb从0开始学(二)-----JSP基本语法与编译指令

这表明JSP注释只在服务器端有效,但是并不会作为HTML代码的一部分发送到客户端。

2.JSP脚本语句

(1)JSP声明

JSP声明用于声明Servlet类的成员变量和方法

JSP的声明格式如下:

<%! 声明内容 %>

下面来看一个使用JSP声明的示例页面:

<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<!DOCTYPE html>
<html>
  <head>
    <title>JSP声明</title>
  </head>

  <%!
    public int count;

    public String sayHello(){
        return "hello";
    }
  %>
  <body>
        <%
        //这行代码的意思是向屏幕输出count的值后再加1
        out.println(count++);
        %>
        <br/>
        <%
        //这行代码的意思是输出sayHello方法的返回值
        out.println(sayHello());
        %>
  </body>
</html>

当我们在浏览器中查看时,可以看到count的值与字符hello的输出。

对于JSP声明有下列三点需要注意:

  1. 可以一次性声明多个变量和方法,前提是符合Java语法。
  2. 一个声明仅在一个页面中有效
  3. 声明的变量为全局变量,所有执行此JSP页面的用户会共享此变量。

在JSP声明中声明方法似乎不需要任何类作为载体而独立存在,事实上并不是这样。根据我们学过的知识我们知道JSP实质 就是Servlet,JSP会被转化为一个Servlet实例。而我们在JSP页面中声明的内容会被转换为对应Servlet实例的成员变量和成员方法。

下面我们打开F:\Tomcat\work\Catalina\localhost\myThreeProject\org\apache\jsp下的.java文件,我们会看到如下的java代码:

*       generation to assist with modification tracking.
 */
package org.apache.jsp;

import javax.servlet.*;
import javax.servlet.http.*;
import javax.servlet.jsp.*;

public final class index_jsp extends org.apache.jasper.runtime.HttpJspBase
    implements org.apache.jasper.runtime.JspSourceDependent,
                 org.apache.jasper.runtime.JspSourceImports {

    public int count;

    public String sayHello(){
        return "hello";
    }
    ...
}

以上就是由JSP页面转换成的的Servlet类,我们可以看到count变量与sayHello方法是这个类的成员。

由于JSP声明的变量与方法会转换为对应Servlet实例的成员变量与成员方法,所以我们可以对JSP声明部分定义的变量和方法使用private和public等访问控制符,也可以使用static,但是不能使用abstract修饰声明部分的方法,因为抽象方法将导致JSP对应的Servlet变成抽象类,从而导致无法实例化。

(2)输出JSP表达式

JSP提供了如下方式的输出表达式语法:

<%=JSP表达式%>

值得注意的是JSP表达式后面不能有分号。

下面来看一个实例:

<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<!DOCTYPE html>
<html>
  <head>
    <title>输出JSP表达式</title>
  </head>

  <%!
    public int count;

    public String sayHello(){
        return "hello";
    }
  %>
  <body>
        <%=count++%>
        <br/>
        <%=sayHello()%>
  </body>
</html>

可以看到这个页面与前一个页面的区别仅仅是把out.println输出语句替换成了'=',这两个页面的执行效果是一致的。其实,JSP/Servlet引擎在解释执行JSP页面时会把'='替换成Servlet实例类中的out输出语句。

(3)JSP程序片段

JSP脚本中可以包含任何可执行性代码

JSP程序片段简称Scriptlet,就是脚本小部件的意思。我们可以通过Scriptlet把代码嵌入到静态HTML页面中去,JSP的核心代码也都是写在这里面的。同样,它的内容也是包裹在'<%%>'中,如下所示:

<% 代码 %>

下面我们来看一个例子:

<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<!DOCTYPE html>
<html>
  <head>
    <title>JSP脚本</title>
  </head>

  <body>
    <%
      String name = "Jom";
      if (name.equals("Jom")){
    %>
      He is Jom.
    <% } else { %>
      He is not Jom.
    <% } %>
  </body>
</html>

输出结果为:

He is Jom.

下面我们打开此项目生成的.java文件

public void _jspService(final javax.servlet.http.HttpServletRequest request, final javax.servlet.http.HttpServletResponse response)
      throws java.io.IOException, javax.servlet.ServletException {
    ...
    try {
     ...
      out.write("\r\n");
      out.write("<!DOCTYPE html>\r\n");
      out.write("<html>\r\n");
      out.write("  <head>\r\n");
      out.write("    <title>JSP脚本<title>\r\n");
      out.write("  </head>\r\n");
      out.write("  \r\n");
      out.write("  <body>\r\n");
      out.write("    ");

      String name = "Jom";
      if (name.equals("Jom")){
        out.write(" \r\n");
        out.write("      He is Jom.\r\n");
        out.write("    ");
      } else {
          out.write("\r\n");
          out.write("      He is not Jom.\r\n");
          out.write("    ");
      }
        out.write("\r\n");
        out.write("  </body>\r\n");
        out.write("</html>");
    }
    ...
  }

我们可以看出,JSP脚本与静态的HTML内容已经全部被转换为_jspService方法的可执行代码。由此我们可以得出两个结论:

  1. 我们可以在JSP脚本中声明变量,只是声明的只是局部变量。
  2. JSP脚本中不能定义方法,因为按照java语法,方法中不能定义方法。

(二)JSP编译指令

JSP编译指令用于提供整个JSP网页的相关信息、设定JSP网页的相关属性,例如网页的编码方式、语法等。JSP编译指令语法如下:

<%@ 指令名 属性1="属性值1" 属性2="属性值2" ...  %>

编译指令都有默认值,所以我们无需为每个指令设置值。常见的编译指令有page、include、taglib三个,下面主要介绍page、include两个指令,关于taglib指令在后续章节中会详细学习。

1.page指令

page指令定义了JSP文件的全局属性,可以指定所使用的的脚本语言、JSP对应的Servlet所实现的接口等。

page指令的语法格式如下:

<%@ page 属性1="属性值1" 属性2="属性值2" ...  %>

page指令常用的属性见下图(图片来源于网络):

JavaWeb从0开始学(二)-----JSP基本语法与编译指令

值得注意的是,contentType的charset属性和pageEncoding属性都涉及JSP网页的编码方式,但是两者有所区别。contentType的charset属性是指服务器发送给客户端时的内容编码方式,pageEncoding是JSP文件本身的编码方式。另外,errorPage属性实质是JSP的异常处理机制,JSP脚本不要求强制处理异常,包括checked异常。

使用page指令需要注意一下几点:

  1. page指令作用于整个页面,包括静态的包含文件,不包括动态的包含文件。
  2. page指令在一个页面中可以出现多次,除了import属性外,其他属性只能出现一次。
  3. 为了JSP程序的可读性,我们一般将page指令置于顶部

2.include指令

使用include指令可以将外部文件包含进当前JSP页面,这个外部文件可以是JSP文件、HTML文件或者文本文件。

include指令的语法格式如下:

<%@ include 属性1="属性值1" 属性2="属性值2" ...  %>

这种包含是静态包含,即将被包含文件插入到JSP指定的位置,解析这个页面,同时把目标文件的其他编译指令也包含进来。

下面来看一个使用静态导入的示例:

 <%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
  <head>
    <title>静态include测试</title>
  </head>

  <body>
       <%@include file="text.jsp" %>
  </body>
</html>

在这个JSP页面里包含了当前路径下的text.jsp文件,该文件里只有一条JSP输出语句:

<%
    out.println("我是text.jsp文件,我被index.jsp文件调用了!");
%>

下面我们启动这个项目,在浏览器中我们会看到如下的输出:

JavaWeb从0开始学(二)-----JSP基本语法与编译指令

我们发现text.jsp文件里的内容在index.jsp文件里被成功执行了。下面我们再打开对应的.java文件,我们会看到如下部分内容:

    out.write("\r\n");
    out.write("<html>\r\n");
    out.write("  <head>\r\n");
    out.write("   <title>静态include测试itle>\r\n");
    out.write("  </head>\r\n");
    out.write("  \r\n");
    out.write("  <body>\r\n");
    out.write("       ");
    out.write("\r\n");
    out.write("<html>\r\n");
    out.write("  <head>\r\n");
    out.write("    <title>JSP-include调用</title>\r\n");
    out.write("  </head>\r\n");
    out.write("  \r\n");
    out.write("  <body>\r\n");
    out.write("       \r\n");
    out.write("        ");
    out.println("我是text.jsp文件,我被index.jsp文件调用了!");
    out.write("\r\n");
    out.write("  </body>\r\n");
    out.write("</html>");
    out.write("\r\n");
    out.write("  </body>\r\n");
    out.write("</html>");

我们发现,在该包含文件被引入处那条语句被执行了。所以,我们得出结论:include指令将包含的页面与本页面融合成一个了页面。

鉴于此,如果多数JSP网页包含相同的内容,我们可以将这些相同的内容单独放在一个文件中,其他JSP页面通过include指令将这个文件包含。这样做不仅可以提高开发效率还有利于网站的维护。

下面以下图总结本次学习内容(图片来源于网络):

JavaWeb从0开始学(二)-----JSP基本语法与编译指令

以上就是本次学习内容,下次我们将学习JSP的动作指令。