首页 > 学院 > 开发设计 > 正文

向高手请教ant构建工具的类装载器问题

2019-11-18 13:35:55
字体:
来源:转载
供稿:网友
问题概述:每个ClassLoader本身只能分别加载特定位置和目录中的类,但是,ClassLoader被设计成了一种委托模式,使得某一个ClassLoader可以委托它的父级类装载器去加载类,从而让应用程序可以借助某一个子级的ClassLoader去多个位置和目录中进行类的加载。这就好比“儿子”除了可以花自己的钱,他还可以花“父亲”的钱,“父亲”又可以花“父亲的父亲”的钱,所以,最终能通过“儿子”花出去的钱包括他历代前辈的钱。类装载器一级级委托到BootStrap类加载器,当BootStrap无法加载当前所要加载的类时,然后才一级级回退到子孙类装载器去进行真正的加载。当回退到最初的类装载器时,假如它自己也不能完成类的装载,那就应报告ClassNotFoundException异常。

  现在的问题是,我编写了一个类装载器去加载特定目录中的类,使用java.exe测试这个类加载器时,测试结果完全正常,可以看到委托效果。而我使用ant工具去调用测试程序时,结果就有点问题了,我编写的类装载器似乎并没有委托其父级类加载器去加载类,而总是自己加载。由于本人才学疏浅,且实在没有精力去研究ant工具的源码,无法了解其类加载内部细节,现在特针对这个问题,向真正的java高手们请教。为了便于高手们快速了解我的问题所在,也便于一些中手们学习,我写出了具体的实验步骤,对于java新手,建议不要参与讨论了,免得我耽误了您宝贵时间。

  1.源程序:MainClass.java

package cn.itcast;
public class MainClass
{
 public static void main(String [] args)
 {
  ClassLoader loader = MainClass.class.getClassLoader();
  //打印出当前的类装载器,及该类装载器的各级父类装载器
  while(loader != null)
  {
   System.out.PRintln(loader.getClass().getName());
   loader = loader.getParent();
  }
  //加载AuxiliaryClass类
  System.out.println(AuxiliaryClass.class.getName());
 }
}
  源程序:AuxiliaryClass.java

package cn.itcast;
public class AuxiliaryClass
{}
  2.源文件及build结果文件的目录结构

f:/project
__src
__cn
__itcast
__MainClass.java
__AuxiliaryClass.java
__build.xml
__classes
__cn
__itcast
__MainClass.class
__AuxiliaryClass.class
  3.build.xml文件内容

<project name="antloader" default="run">
<property name="classes.dir" value="classes" />
<property name="src.dir" value="src" />

<target name="init">
<mkdir dir="${classes.dir}" />
</target>

<target name="compile" depends="init">
<javac destdir="${classes.dir}" >
<src path="${src.dir}" />
</javac>
</target>

<target name="run" depends="compile">
<java classname="cn.itcast.MyClassLoader">
<classpath>
<pathelement location="${classes.dir}"/>
</classpath>
</java>
</target>
</project>
  4.进入project目录中运行ant,执行结果正常,如下:

org.apache.tools.ant.loader.AntClassLoader2
sun.misc.Launcher$AppClassLoader
sun.misc.Launcher$ExtClassLoader
cn.itcast.AuxiliaryClass
  5.修改build.xml文件,将最后名称为“run”的target(执行目标)修改成如下形式,即不设置其中的<classpath>子元素。

<target name="run" depends="compile">
<java classname="cn.itcast.MyClassLoader">
<!--classpath>
<pathelement location="${classes.dir}"/>
</classpath-->
</java>
</target>
  再次执行ant,将报告如下错误信息:

Could not find cn.itcast.MainClass. Make sure you have it in your classpath
at org.apache.tools.ant.taskdefs.ExecuteJava.execute(ExecuteJava.java:170)
  在执行ant的命令行窗口中设置classpath环境变量:



发表评论 共有条评论
用户名: 密码:
验证码: 匿名发表