JSP编译成Servlet(五)JDT Compiler编译器

简介: 通过JSP编译器编译后生成了对应的java文件,接下去要把Java文件编译成class文件。对于这部分完全没有必要重新造轮子,常见的优秀编译工具有Eclipse JDT Java编译器和Ant编译器。

通过JSP编译器编译后生成了对应的java文件,接下去要把Java文件编译成class文件。对于这部分完全没有必要重新造轮子,常见的优秀编译工具有Eclipse JDT Java编译器和Ant编译器。Tomcat其实是同时支持两个编译器的,通过配置可以选择,而默认是使用Eclipse JDT编译器。

通过调用这些现成的编译器的API就可以方便地实现对java文件的编译,由于两个编译器功能基本一样,我们就挑默认编辑器看下它是如何进行编译的,下面仅看如何用Eclipse JDT编译器编译java文件。

Eclipse JDT提供了Compiler类用于编译,它的构造函数比较复杂,如下所示,其实就是实现自定义构造函数包含的参数即基本完成了编译工作。

public Compiler(

  INameEnvironment environment,

  IErrorHandlingPolicy policy,

  CompilerOptions options,

  final ICompilerRequestor requestor,

  IProblemFactory problemFactory) {

}

为了说明方便直接上一个简单的编译实现,如下:

public class JDTCompile {

 

    private static final File WORKDIR = new File("D:\\Program Files\\tomcat7\\work\\Catalina\\localhost\\test");

 

    public static void main(String[] args) {

 

        INameEnvironment nameEnvironment = new INameEnvironment() {

            public NameEnvironmentAnswer findType(final char[][] compoundTypeName) {

                return findType(join(compoundTypeName));

            }

 

            public NameEnvironmentAnswer findType(final char[] typeName, final char[][] packageName) {

                return findType(join(packageName) + "." + new String(typeName));

            }

 

            private NameEnvironmentAnswer findType(final String name) {

                File file = new File(WORKDIR, name.replace('.', '/') + ".java");

                if (file.isFile()) {

                    return new NameEnvironmentAnswer(new CompilationUnit(file), null);

                }

                try {

                    InputStream input =

                            this.getClass().getClassLoader().getResourceAsStream(name.replace(".", "/") + ".class");

                    if (input != null) {

                        byte[] bytes = IOUtils.toByteArray(input);

                        if (bytes != null) {

                            ClassFileReader classFileReader = new ClassFileReader(bytes, name.toCharArray(), true);

                            return new NameEnvironmentAnswer(classFileReader, null);

                        }

                    }

                } catch (ClassFormatException e) {

                    throw new RuntimeException(e);

                } catch (IOException e) {

                    throw new RuntimeException(e);

                }

                return null;

            }

 

            public boolean isPackage(char[][] parentPackageName, char[] packageName) {

                String name = new String(packageName);

                if (parentPackageName != null) {

                    name = join(parentPackageName) + "." + name;

                }

 

                File target = new File(WORKDIR, name.replace('.', '/'));

                return !target.isFile();

            }

 

            public void cleanup() {}

        };

 

        ICompilerRequestor compilerRequestor = new ICompilerRequestor() {

 

            public void acceptResult(CompilationResult result) {

                if (result.hasErrors()) {

                    for (IProblem problem : result.getErrors()) {

                        String className = new String(problem.getOriginatingFileName()).replace("/", ".");

                        className = className.substring(0, className.length() - 5);

                        String message = problem.getMessage();

                        if (problem.getID() == IProblem.CannotImportPackage) {

                            message = problem.getArguments()[0] + " cannot be resolved";

                        }

                        throw new RuntimeException(className + ":" + message);

                    }

                }

 

                ClassFile[] clazzFiles = result.getClassFiles();

                for (int i = 0; i < clazzFiles.length; i++) {

                    String clazzName = join(clazzFiles[i].getCompoundName());

                    File target = new File(WORKDIR, clazzName.replace(".", "/") + ".class");

                    try {

                        FileUtils.writeByteArrayToFile(target, clazzFiles[i].getBytes());

                    } catch (IOException e) {

                        throw new RuntimeException(e);

                    }

                }

            }

        };

 

        IProblemFactory problemFactory = new DefaultProblemFactory(Locale.ENGLISH);

        IErrorHandlingPolicy policy = DefaultErrorHandlingPolicies.exitOnFirstError();

 

        org.eclipse.jdt.internal.compiler.Compiler jdtCompiler =

                new org.eclipse.jdt.internal.compiler.Compiler(nameEnvironment, policy, getCompilerOptions(),

                        compilerRequestor, problemFactory);

 

        jdtCompiler

                .compile(new ICompilationUnit[] {new CompilationUnit(new File(WORKDIR, "org\\apache\\jsp\\HelloWorld_jsp.java"))});

    }

 

    public static CompilerOptions getCompilerOptions() {

        Map settings = new HashMap();

        String javaVersion = CompilerOptions.VERSION_1_7;

        settings.put(CompilerOptions.OPTION_Source, javaVersion);

        settings.put(CompilerOptions.OPTION_Compliance, javaVersion);

        return new CompilerOptions(settings);

    }

 

    private static class CompilationUnit implements ICompilationUnit {

 

        private File file;

 

        public CompilationUnit(File file) {

            this.file = file;

        }

 

        public char[] getContents() {

            try {

                return FileUtils.readFileToString(file).toCharArray();

            } catch (IOException e) {

                throw new RuntimeException(e);

            }

        }

 

        public char[] getMainTypeName() {

            return file.getName().replace(".java", "").toCharArray();

        }

 

        public char[][] getPackageName() {

            String fullPkgName = this.file.getParentFile().getAbsolutePath().replace(WORKDIR.getAbsolutePath(), "");

            fullPkgName = fullPkgName.replace("/", ".").replace("\\", ".");

            if (fullPkgName.startsWith("."))

                fullPkgName = fullPkgName.substring(1);

            String[] items = fullPkgName.split("[.]");

            char[][] pkgName = new char[items.length][];

            for (int i = 0; i < items.length; i++) {

                pkgName[i] = items[i].toCharArray();

            }

            return pkgName;

        }

 

        public boolean ignoreOptionalProblems() {

            return false;

        }

 

        public char[] getFileName() {

            return this.file.getName().toCharArray();

        }

    }

 

    private static String join(char[][] chars) {

        StringBuilder sb = new StringBuilder();

        for (char[] item : chars) {

            if (sb.length() > 0) {

                sb.append(".");

            }

            sb.append(item);

        }

        return sb.toString();

    }

 

}

为了更好理解,我们根据构造函数的参数分别看看,

INameEnvironment接口,主要需要实现的方法是findTypeisPackagefindType是帮助JDT找到相应的java源文件或者class字节码,根据传进来的包名和类名去寻找。例如传入“java.lang.String”或“org.apache.jsp.HelloWorld_jsp”则分别要找到JDK自带的String字节码及tomcat中编译的HelloWorld_jsp.java文件。接着按要求封装这些对象返回JDT规定的NameEnvironmentAnswer对象。而isPackage则提供是否是包的判断。

IErrorHandlingPolicy接口,用于描述错误策略,可直接使用DefaultErrorHandlingPolicies.exitOnFirstError(),表示第一个错误就退出编译。

CompilerOptions对象,指定编译时的一些参数,例如这里指定编译的Java版本为1.7

ICompilerRequestor接口,它只有一个acceptResult方法,这个方法用于处理编译后的结果,如果包含了错误信息则抛异常,否则则把编译成功的字节码写到指定路径的HelloWorld_jsp.class文件中,即生成字节码。

IProblemFactory接口,主要是控制编译错误信息的格式。

所有Compiler构造函数需要的参数对象都已经具备,传入这些参数后创建一个Compiler对象,然后调用compile方法即可对指定的java文件进行编译。这里完成了HelloWorld_jsp.java的编译,结果生成了HelloWorld_jsp.class字节码。实际的tomcat中基本也是类似这样使用JDT实现servlet的编译,但它使用的某些策略可能不相同,例如使用DefaultErrorHandlingPolicies.proceedWithAllProblems()作为错误策略。

通过这两章节“从JSPServlet”及“从Servletclass字节码”,我们已经清楚tomcatJSP编译处理的整个过程了,先根据JSP语法解析生成类似xxxx.javaServlet,然后再通过Eclipse JDTxxxx.java编译,最后生成了JVM能识别的class字节码。


点击订购作者《Tomcat内核设计剖析》



目录
相关文章
|
Java 容器
【学习笔记】Jsp与Servlet技术
【学习笔记】Jsp与Servlet技术
280 0
|
SQL Java 数据库
jsp中使用Servlet查询SQLSERVER数据库中的表的信息,并且打印在屏幕上
该博客文章介绍了在JSP应用中使用Servlet查询SQL Server数据库的表信息,并通过JavaBean封装图书信息,将查询结果展示在Web页面上的方法。
jsp中使用Servlet查询SQLSERVER数据库中的表的信息,并且打印在屏幕上
|
前端开发 安全 Java
在Java服务器端开发的浩瀚宇宙中,Servlet与JSP犹如两颗璀璨的明星,它们联袂登场,共同编织出动态网站的绚丽篇章。
在Java服务器端开发的浩瀚宇宙中,Servlet与JSP犹如两颗璀璨的明星,它们联袂登场,共同编织出动态网站的绚丽篇章。
154 1
|
供应链 前端开发 Java
JSP+servlet+mybatis+layui服装库存管理系统(大三上学期课程设计)
这篇文章通过一个服装库存管理系统的实例,展示了在Spring Boot项目中使用Ajax、JSON、layui、MVC架构和iframe等技术,涵盖了注册登录、权限管理、用户管理、库存管理等功能,并提供了系统运行环境和技术要求的详细说明。
JSP+servlet+mybatis+layui服装库存管理系统(大三上学期课程设计)
|
自然语言处理 前端开发 Java
Servlet与JSP:Java Web开发的基石技术详解
【6月更文挑战第23天】Java Web的Servlet与JSP是动态网页的核心。Servlet是服务器端的Java应用,处理HTTP请求并响应;JSP则是结合HTML与Java代码的页面,用于动态内容生成。Servlet通过生命周期方法如`init()`、`service()`和`destroy()`工作,而JSP在执行时编译成Servlet。两者在MVC架构中分工,Servlet处理逻辑,JSP展示数据。尽管有Spring MVC等框架,Servlet和JSP仍是理解Web开发基础的关键。
311 12
|
存储 Java 关系型数据库
基于Servlet和JSP的Java Web应用开发指南
【6月更文挑战第23天】构建Java Web应用,Servlet与JSP携手打造在线图书管理系统,涵盖需求分析、设计、编码到测试。通过实例展示了Servlet如何处理用户登录(如`LoginServlet`),JSP负责页面展示(如`login.jsp`和`bookList.jsp`)。应用基于MySQL数据库,包含用户和图书表。登录失败显示错误信息,成功后展示图书列表。部署到Tomcat服务器测试功能。此基础教程为深入Java Web开发奠定了基础。
307 10
|
缓存 小程序 前端开发
Java服务器端技术探秘:Servlet与JSP的核心原理
【6月更文挑战第23天】Java Web开发中的Servlet和JSP详解:Servlet是服务器端的Java小程序,处理HTTP请求并响应。生命周期含初始化、服务和销毁。创建Servlet示例代码展示了`doGet()`方法的覆盖。JSP则侧重视图,动态HTML生成,通过JSP脚本元素、声明和表达式嵌入Java代码。Servlet常作为控制器,JSP处理视图,遵循MVC模式。优化策略涉及缓存、分页和安全措施。这些技术是Java服务器端开发的基础。
279 9
|
缓存 安全 Java
Java服务器端技术:Servlet与JSP的集成与扩展
【6月更文挑战第23天】Java Web开发中,Servlet和JSP是构建动态Web应用的基础。Servlet处理逻辑,JSP专注展示。示例展示了Servlet如何通过`request.setAttribute`传递数据给JSP渲染。JSP自定义标签提升页面功能,如创建`WelcomeTag`显示欢迎消息。Servlet过滤器,如`CacheControlFilter`,用于预处理数据或调整响应头。这些集成和扩展技术增强了应用效率、安全性和可维护性,是Java服务器端开发的关键。
191 7
|
前端开发 安全 Java
Java服务器端开发实战:利用Servlet和JSP构建动态网站
【6月更文挑战第23天】**Servlet和JSP在Java Web开发中扮演关键角色。Servlet处理业务逻辑,管理会话,JSP则结合HTML生成动态页面。两者协同工作,形成动态网站的核心。通过Servlet的doGet()方法响应请求,JSP利用嵌入式Java代码创建动态内容。实战中,Servlet处理数据后转发给JSP展示,共同构建高效、稳定的网站。虽然新技术涌现,Servlet与JSP仍为Java Web开发的基石,提供灵活且成熟的解决方案。**
323 8
|
缓存 负载均衡 安全
Servlet与JSP在Java Web应用中的性能调优策略
【6月更文挑战第23天】在Java Web中,Servlet和JSP调优至关重要,以应对高并发和复杂业务带来的性能挑战。优化包括Servlet复用、线程安全、数据库连接池,以及JSP的编译优化、使用JSTL、页面缓存和静态内容分离。全局优化涉及负载均衡、异步处理和缓存策略。通过这些实践,开发者能提升应用响应速度和吞吐量,确保高负载下的稳定运行。
314 7