在上一节中我们学习了如何搭建一个简单的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注释消失了,如下图所示:

这表明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指令常用的属性见下图(图片来源于网络):

值得注意的是,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文件调用了!");
%>

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

我们发现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指令将这个文件包含。这样做不仅可以提高开发效率还有利于网站的维护。

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

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

JavaWeb从0开始学(二)-----JSP基本语法与编译指令的更多相关文章

  1. JavaWeb从0开始学(一)-----搭建第一个Web应用程序与JSP工作原理

    以往学习的时候大多是看完书或者看完视频,动手实践一下就OK了.然而过了一段时间我发现东西都忘差不多了,需要复习才能重新掌握.现在开始学习JavaWeb了,我将在这里记录自己的学习的一点一滴,不仅便于自 ...

  2. 02.JSP的3个编译指令

    本章介绍JSP的3个编译指令,在JSP中常见的编译指令有如下3个:         1.page:用于针对当前页面的指令.         2.include:用于指定包含另一个页面.         ...

  3. 二、JSP的3个编译指令,7个动作指令,9个内置对象

    JSP 3个编译指令 1)  page指令(基本不需要用到,使用默认即可) 主要属性: 1.Language:指定脚本所采用的语言类型,现在只支持java 2.Extends:定义当前jsp产生的se ...

  4. jsp 的 3 个编译指令

    JSP 的编译指令是通知 JSP 引擎的消息,它不直接生成输出. 常见的编译指令有如下三个: 1.page:该指令是针对当前页面的指令 2.include:用于指定包含另一个页面 3.taglib:用 ...

  5. 从0开始学Java——JSP和Servlet——jsp转servlet出错的三个典型场景

    由于jsp终究是要转换为servlet的java文件,然后再编译为.class文件,最后才执行,那么在这过程的任何一个步骤都可能有问题,主要包括三个方面,下面逐一分析: 一.JSP转换为Servlet ...

  6. 从0开始学Java——JSP&Servlet——Tomcat和Apache的区别

    从<JSP & Servlet 学习笔记>的第一章,了解到web容器:“Web容器是Servlet/jsp唯一认得的http服务器”. 在Java中,容器的类型有多种,这里要说的是 ...

  7. 从0开始学java——JSP&Servlet——web容器搜索class的路径顺序

    在web应用程序如果要用到某个类,会按照如下的顺序来搜索: 1)在WEB-INF/classes目录下搜索: 2)如果该目录下没有,则会到WEB-INF/lib目录下的jar文件中搜索: 3)如果还没 ...

  8. 从0开始学Java——JSP&Servlet——HttpServletRequest相关的几个路径信息

    在HttpServletRequest中有几个获取路径的接口:getRequestURI/getContextPath/getServletPath/getPathInfo 这些接口互相之间有什么区别 ...

  9. 从0开始学Java——JSP&Servlet——如何部署web应用程序

    web容器要求应用程序部署时,需要像下面这样组织其目录结构: 手动去创建这样的目录结构还是挺麻烦的,所幸我们有开发工具,所以可以像下面这样来部署一个web项目. 1)确认程序代码已经完成: 2)在ec ...

随机推荐

  1. SSM整合XML版(Maven Project)

    一.使用IntelliJ IDEA 创建Maven管理的JAVA Web项目 ①新建项目,选择Maven Module 在本地磁盘用户的..m2目录下生成settings文件: Maven工程搭建完毕 ...

  2. 关于GCD同步组实现多个异步线程的同步执行中的注意点

    在App开发中经常会遇到多个线程同时向服务器取数据, 如果每个线程取得数据后都去刷新UI会造成界面的闪烁 也有可能出现部分数据还没有获取完毕造成程序crash 之前在网上看到很多是利用dispatch ...

  3. android Android性能优化之如何避免Overdraw

    什么是Overdraw? Overdraw就是过度绘制   怎么来消灭overdraw呢?总的原则就是:尽量避免重叠不可见元素的绘制,基于这个原则,我们大概可以想出以下几招: 第一招:合理选择控件容器 ...

  4. Android7.0 Phone应用源码分析(三) phone拒接流程分析

    本文主要分析Android拒接电话的流程,下面先来看一下拒接电话流程时序图 步骤1:滑动按钮到拒接图标,会调用到AnswerFragment的onDecline方法 com.android.incal ...

  5. 无限二等分[0,1]这个区间之后还剩下啥?what's left after dividing an unit interval [0,1] infinitely many times?

    Dividing an unit interval \([0,1]\) into two equal subintervals by the midpoint \(\dfrac {0+1} {2}=\ ...

  6. java集合框架02——Collection架构与源码分析

    Collection是一个接口,它主要的两个分支是List和Set.如下图所示: List和Set都是接口,它们继承与Collection.List是有序的队列,可以用重复的元素:而Set是数学概念中 ...

  7. 面向UI编程:ui.js 1.1 使用观察者模式完成组件之间数据流转,彻底分离组件之间的耦合,完成组件的高内聚

    开头想明确一些概念,因为有些概念不明确会导致很多问题,比如你写这个框架为什么不去解决啥啥啥的问题,哎,心累. 什么是框架? 百度的解释:框架(Framework)是整个或部分系统的可重用设计,表现为一 ...

  8. 【译文】什么是Docker

    What is Docker? By Tim Butler • 14 May 2015 • https://www.conetix.com.au/blog/what-is-docker Unless ...

  9. 特性Attribute 的使用

    [IdentityAuthorize]           public ActionResult Index()        {             return View("~/V ...

  10. JAVA设计模式:单例设计

    1.单例设计Singleton的引出 单例设计,从名字上首先可以看出单---即只有一个,例---只的是实例化对象:那么单例也就是说一个类,只产生了一个实例化对象.但是我们都知道,一个类要产生实例化对象 ...