因为要实现一个WebServer,写完了一个简单的WebServer后突发奇想,何不实现一个JSP 的WebServer呢? 有了这个想法后,就开始考虑JSP的实现原理,其实JSP的实现也很简单,说白了就是将其转换成一个Java文件,再对这个Java文件进行编译,生成类文件,接着再由服务器使用这个类文件。从总体上分,可以将JSP文件的内容划分为两大块,即输出语句和控制语句。 至于JSP文件中的HTML语句就是我们要输出的内容,而JSP标签则是控制HTML语句的输出。例如,有一个JSP文件的内容如下: <html> <body> <%! java.util.Date date=new java.util.Date();         int  size=10; %> <%  for(int i=0;i<size;i++){ %> <h1>Hello<%= i %> </h1> <% } %> <%=date%> </body> </html> 这个JSP文件将产生如下效果: 
Hello 0 
Hello 1 
Hello 2 
Hello 3 
Hello 4 
Hello 5 
Hello 6 
Hello 7 
Hello 8 
Hello 9 
Thu Dec 16 18:30:21 CST 2004
  那么,现在的问题就是我们就生成什么样的java文件,才能获得正确的结果。 首选写一个Interface,该接口就是我们要生成的JAVA文件将要继承的接口。其定义如下: package httpserver; import java.net.Socket; import java.io.IOException;
  public interface Jsp2HtmlInterface{  void printHtml(Socket socket)throws IOException; }
  接下来就写一个类JavaServerPage负责解析JSP文件,在这个类中将会动态生成JAVA文件,并编译生成一个可用类。再调用这个类文件的printHtml(socket)  方法,将响应信息写给客户端(其中socket为服务器accept返回的套接口。这样客户就能看到JSP正确运行的结果。这个解析JavaServerPage的类框架如下: public class JavaServerPage{     public JavaServerPage(File file,Socket socket)throws Exception{}    //file为客户端请求的一个JSP文件                                                                                                             //socket为服务器同客户端联接的套接口     public void writeReponse() {                                                            //这个函数将负责根据指定的JSP文件根据某种          GeneratJava();                                                                            //动态生成java文件并将编译,           Process proc=Runtime.getRuntime().exec("javac ...");          try{                   proc.waitFor();                }catch(InterruptedException ie){      }           httpserver.Jsp2HtmlInterface object=(httpserver.Jsp2HtmlInterface)Class.forName("...").newInstance();           object.printHtml(socket);     } }
  假设我们用某种算法将上面的JSP文件生成如下的JAVA文件 package httpserver; import java.io.*; import java.awt.*; import java.net.Socket; public class test implements Jsp2HtmlInterface{ private PrintStream out; public test(){} public void println(Object object)throws IOException{ out.print(object.toString()+"\r\n");} public void println(int i)throws IOException{ out.print(i+"\r\n");} public void println(float i)throws IOException{ out.print(i+"\r\n");} public void println(double i)throws IOException{ out.print(i+"\r\n");} public void println(long i)throws IOException{ out.print(i+"\r\n");} public void println(char ch)throws IOException{ out.print(ch+"\r\n");} public void println(char[] ch)throws IOException{ out.print(ch+"\r\n");} public void println()throws IOException{ out.print("\r\n"); } public void close(){ out.close(); } public void printHtml(Socket socket)throws IOException{ out=new PrintStream(socket.getOutputStream()); this.println("<html>"); this.println("<title>Hello</title>"); this.println("<body>");  java.util.Date date=new java.util.Date();     int size=10; 
  
for(int i=0;i<size;i++){   
this.println(" <h1>Hello"); this.println( i ); this.println(); this.println(" </h1>");  }  
this.println( date ); this.println(); this.println("</body>"); this.println("</html>"); this.println(); this.close(); } }
 
  通过调用该类printHtml()方法即可实现JSP文件的解析。 本方法在JBuilderX环境下编译通过。 说到这里,大家是不是想动手实现一个自已的tomcat或者实现别的什么动态脚本语言了。
  <%! java.util.Date date=new java.util.Date(); 
    int size=10;
%><% 
for(int i=0;i
 <% } %><%= date %> 
 
  |