3 - 3 - 2 : JSPコンパイラを頭に
JSPをコンパイルして作成されるJAVAファイルがどのような構成になるかを理解しておけば、スコープが複雑なJSPの動作を解析し易くなる。 ※ここでは tomcat4.0 のJSPコンパイラを利用する。
まずスクリプトレットを何も書かなかった場合。
[JspToJava1.jsp]
HTML only
[JspToJava1$jsp.java]
package org.apache.jsp;
import javax.servlet.*;
import javax.servlet.http.*;
import javax.servlet.jsp.*;
import org.apache.jasper.runtime.*;
public class JspToJava1$jsp extends HttpJspBase {
static {
}
public JspToJava1$jsp( ) {
}
private static boolean _jspx_inited = false;
public final void _jspx_init() throws org.apache.jasper.runtime.JspException {
}
public void _jspService(HttpServletRequest request, HttpServletResponse response)
throws java.io.IOException, ServletException {
JspFactory _jspxFactory = null;
PageContext pageContext = null;
HttpSession session = null;
ServletContext application = null;
ServletConfig config = null;
JspWriter out = null;
Object page = this;
String _value = null;
try {
if (_jspx_inited == false) {
synchronized (this) {
if (_jspx_inited == false) {
_jspx_init();
_jspx_inited = true;
}
}
}
_jspxFactory = JspFactory.getDefaultFactory();
response.setContentType("text/html;charset=ISO-8859-1");
pageContext = _jspxFactory.getPageContext(this, request, response,
"", true, 8192, true);
application = pageContext.getServletContext();
config = pageContext.getServletConfig();
session = pageContext.getSession();
out = pageContext.getOut();
// HTML // begin [file="/JspToJava1.jsp";from=(0,0);to=(4,7)]
out.write("\r\n\r\nHTML only\r\n\r\n");
// end
} catch (Throwable t) {
if (out != null && out.getBufferSize() != 0)
out.clearBuffer();
if (pageContext != null) pageContext.handlePageException(t);
} finally {
if (_jspxFactory != null) _jspxFactory.releasePageContext(pageContext);
}
}
}
JspToJava1.jsp が変換されて JspToJava1$jsp.java が作成される。このjavaファイルの形式はtomcatのコンパイラによるが、太字の _jspService メソッドの宣言や application, config, session, out の取得方法はServletの仕様に従っている。
_jspService メソッドは JAVAファイルでは必ず定義する(J2EEのフレームワーク)。
これらは session と pageContext.getSession() は同じかどうか?といった問題が出題された際に参考にすればいい。
ここではひとまず_jspService メソッド内で HTML を出力している ということが重要で、このことを常に意識しておく必要がある。
次にスクリプトレットでローカル変数を宣言した場合。
[JspToJava2.jsp]
<% String localString = ""; %>
[JspToJava2$jsp.java] (// HTML // begin のみ抜粋)
// HTML // begin [file="/JspToJava2.jsp";from=(0,0);to=(2,0)]
out.write("\r\n\r\n");
// end
// begin [file="/JspToJava2.jsp";from=(2,2);to=(2,28)]
String localString = "";
// end
// HTML // begin [file="/JspToJava2.jsp";from=(2,30);to=(4,7)]
out.write("\r\n\r\n");
localString は BODYタグの次に宣言されている。つまりJSPの記述どおりに出力される。
これは
<%= localString%>
<% String localString = ""; %>
はJAVAでは
out.write(localString);
String localString = "";
となるため、JAVA>クラス のコンパイル時にコンパイルエラーとなることが理解できる。
JSP>JAVAコンパイル時にはエラーとならないことに注意。
次はクラス変数を宣言した場合。
[JspToJava3.jsp]
<%! int counter = 0; %>
<%= counter++ %>
[JspToJava3$jsp.java]
package org.apache.jsp;
import javax.servlet.*;
import javax.servlet.http.*;
import javax.servlet.jsp.*;
import org.apache.jasper.runtime.*;
public class JspToJava3$jsp extends HttpJspBase {
// begin [file="/JspToJava3.jsp";from=(2,3);to=(2,21)]
int counter = 0;
// end
static {
}
public JspToJava3$jsp( ) {
}
private static boolean _jspx_inited = false;
public final void _jspx_init() throws org.apache.jasper.runtime.JspException {
}
public void _jspService(HttpServletRequest request, HttpServletResponse response)
throws java.io.IOException, ServletException {
JspFactory _jspxFactory = null;
PageContext pageContext = null;
HttpSession session = null;
ServletContext application = null;
ServletConfig config = null;
JspWriter out = null;
Object page = this;
String _value = null;
try {
if (_jspx_inited == false) {
synchronized (this) {
if (_jspx_inited == false) {
_jspx_init();
_jspx_inited = true;
}
}
}
_jspxFactory = JspFactory.getDefaultFactory();
response.setContentType("text/html;charset=ISO-8859-1");
pageContext = _jspxFactory.getPageContext(this, request, response,
"", true, 8192, true);
application = pageContext.getServletContext();
config = pageContext.getServletConfig();
session = pageContext.getSession();
out = pageContext.getOut();
// HTML // begin [file="/JspToJava3.jsp";from=(0,0);to=(2,0)]
out.write("\r\n\r\n");
// end
// HTML // begin [file="/JspToJava3.jsp";from=(2,23);to=(3,0)]
out.write("\r\n");
// end
// begin [file="/JspToJava3.jsp";from=(3,3);to=(3,14)]
out.print( counter++ );
// end
// HTML // begin [file="/JspToJava3.jsp";from=(3,16);to=(5,7)]
out.write("\r\n\r\n");
// end
} catch (Throwable t) {
if (out != null && out.getBufferSize() != 0)
out.clearBuffer();
if (pageContext != null) pageContext.handlePageException(t);
} finally {
if (_jspxFactory != null) _jspxFactory.releasePageContext(pageContext);
}
}
}
クラス変数として
int counter = 0;
が宣言されているため、クラスインスタンスが存在する限り、 counter も存在する。
基本的にJSPが一度実行されるとその1つのインスタンスがサーブレットコンテナに保存され、他のユーザーからも利用される。インスタンスはアプリケーションスコープに存在するので counter は全ユーザーで共通なオブジェクトとなる。 したがってブラウザからアクセスする毎に counter の値は インクリメントされる。
逆にクラス変数になることを利用して <%! static final COUNTER_MAX = 0; %>
と定数を定義することもできる。
このように、たまにバグの原因となる <%! int counter = 0; %> タグのスクリプトレットも理解しておく必要がある。
通常のJAVAの記述と同様、クラス変数を宣言する前に参照することもできる。
[JspToJava3_2.jsp]
<%= counter++ %>
<%! int counter = 0; %>
一見コンパイルエラーになりそうだが、counter++ はクラス変数へのアクセスなので問題ない。
いつでもJSPの記述順に処理されると勘違いしていると、このような問題に引っかかるかもしれない。
最後にメソッド宣言の例を挙げる。
[JspToJava4.jsp]
<%! private int add(int a, int b) { return a+b;} %>
<%= add(1,2) %>
[JspToJava4$jsp.java]
package org.apache.jsp;
import javax.servlet.*;
import javax.servlet.http.*;
import javax.servlet.jsp.*;
import org.apache.jasper.runtime.*;
public class JspToJava4$jsp extends HttpJspBase {
// begin [file="/JspToJava4.jsp";from=(2,3);to=(2,49)]
private int add(int a, int b) { return a+b;}
// end
static {
}
public JspToJava4$jsp( ) {
}
private static boolean _jspx_inited = false;
public final void _jspx_init() throws org.apache.jasper.runtime.JspException {
}
public void _jspService(HttpServletRequest request, HttpServletResponse response)
throws java.io.IOException, ServletException {
JspFactory _jspxFactory = null;
PageContext pageContext = null;
HttpSession session = null;
ServletContext application = null;
ServletConfig config = null;
JspWriter out = null;
Object page = this;
String _value = null;
try {
if (_jspx_inited == false) {
synchronized (this) {
if (_jspx_inited == false) {
_jspx_init();
_jspx_inited = true;
}
}
}
_jspxFactory = JspFactory.getDefaultFactory();
response.setContentType("text/html;charset=ISO-8859-1");
pageContext = _jspxFactory.getPageContext(this, request, response,
"", true, 8192, true);
application = pageContext.getServletContext();
config = pageContext.getServletConfig();
session = pageContext.getSession();
out = pageContext.getOut();
// HTML // begin [file="/JspToJava4.jsp";from=(0,0);to=(2,0)]
out.write("\r\n\r\n");
// end
// HTML // begin [file="/JspToJava4.jsp";from=(2,51);to=(3,0)]
out.write("\r\n");
// end
// begin [file="/JspToJava4.jsp";from=(3,3);to=(3,13)]
out.print( add(1,2) );
// end
// HTML // begin [file="/JspToJava4.jsp";from=(3,15);to=(5,7)]
out.write("\r\n\r\n");
// end
} catch (Throwable t) {
if (out != null && out.getBufferSize() != 0)
out.clearBuffer();
if (pageContext != null) pageContext.handlePageException(t);
} finally {
if (_jspxFactory != null) _jspxFactory.releasePageContext(pageContext);
}
}
}
クラス変数宣言と同様に、宣言 を利用してメソッドも定義できる。
もちこん
<% private int add(int a, int b) { return a+b;} %>
と記述することはできない。JSP>JAVAは成功するが JAVA>クラスのコンパイルでエラーが発生する。
メソッド内では out, session といったオブジェクトはまだ取得していないため、
<%! private int add(int a, int b) { out.write(a+b);} %>
といった記述も不可能。この辺は案外ややこしい。
application, session, out, config といった暗黙オブジェクトは _jspService メソッド内で宣言されているローカル変数であることを 認識していれば、どのような場合にエラーになるか理解できる。