java 类加载与自定义类加载器详解
类加载 所有类加载器,都是ClassLoader的子类。 类加载器永远以.class运行的目录为准。 读取classpath根目录下的文件有以下几种方式: 1 在Java项目中可以通过以下方式获取classspath下的文件: public void abc(){ //每一种读取方法,使用某个类获取Appclassloader ClassLoader cl = ReadFile.class.getClassLoader(); URL url = cl.getResource("a.txt"); System.err.println("url1 is:"+url.getPath()); //第二种方式,直接使用ClassLoader URL url2 = ClassLoader.getSystemResource("a.txt"); System.err.println("url2 is:"+url2.getPath()); } 在Tomcat中tomcat又声明了两个类载器: StandardClassLoaderC 加载tomcat/lib/*.jar - serlvetapi.jar Webappclassloader /加载 tomcat/webapps/project/web-inf/lib/*.jar && web-inf/classes/*.class 在任何的项目中,获取类的加载器都应该使用以下方式: SomeClass(你写的).class.getClassLoader().getResource ;获取到这个类的类加载器 在java项目中是:AppClassLoader 在Web项目中:WebAppClassLoader 测试父类加载器: public class OneServlet extends HttpServlet { @Override public void doGet(HttpServletRequest req,HttpServletResponse resp) throws ServletException,IOException { ClassLoader loader = OneServlet.class.getClassLoader();//WebAppClassLoader int index=1; while(loader!=null){ System.err.println((index++)+"类加载器是:"+loader.getClass()); loader=loader.getParent();//获取父类加载器 } } } 运行的结果: 1类加载器是:class org.apache.catalina.loader.WebappClassLoader 2类加载器是:class org.apache.catalina.loader.StandardClassLoader 3类加载器是:class sun.misc.Launcher$AppClassLoader 4类加载器是:class sun.misc.Launcher$ExtClassLoader 自定义类加载器 JDK以哪一个类加载器读取A类的字节码,则A类就是被哪一个类加载器加载 的。 一个同名的类,是否可以相互转换,要看是否是在同个类加载器中。 package cn.hx.demo; import java.io.File; import java.io.FileInputStream; import java.io.InputStream; import java.net.URL; public class MyClassLoader2 extends ClassLoader { /** * name:cn.itcast.demo.Person * 根据包名找到.class文件 * cn.itcast.demo.person = > cn/itcast/demo/Person.class */ public Class<?> findClass(String name ) throws ClassNotFoundException { String classNameWithPackage=name; Class<?> cls = null; try { //先将 name = name.replace(".","/"); name +=".class"; //确定目录 URL url = MyClassLoader2.class.getClassLoader().getResource(name); System.err.println(">>:"+url.getPath()); File file = new File(url.getPath()); InputStream in = new FileInputStream(file); //读取这个.class文件的字节码 byte[] b = new byte[in.available()];//直接声明这个字节大小为这个文件的大小 int len = in.read(b);//len=621 System.err.println(len); /** * 第一个参数是类名 */ cls = defineClass(classNameWithPackage,b,len); } catch (Exception e) { e.printStackTrace(); } return cls; } } 测试类自定义类加载器 public class ClassLoaderDemo { public static void main(String[] args) throws Exception { MyClassLoader2 mc = new MyClassLoader2(); Class cls = mc.findClass("cn.itcast.demo.Person"); Object o = cls.newInstance(); System.err.println("toString:"+o+","+o.getClass().getClassLoader()); //直接使用peron是 AppClassLoader System.err.println(">>:"+Person.class.getClassLoader()); //由于o是由mc加载的。而Person是由App加载的,所有不可以转换=来自于两个不同的加载器 //Person p = (Person) o;//类型转换错误ClassCastException //System.err.println(p); } } 以上就是本文的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,同时也希望多多支持编程小技巧! (编辑:李大同) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |