码迷,mamicode.com
首页 > 其他好文 > 详细

类加载与自定义类加载器(67)

时间:2015-05-22 11:18:22      阅读:148      评论:0      收藏:0      [点我收藏+]

标签:

类加载

所有类加载器,都是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又声明了两个类载器:

       StandardClassLoader– 加载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,0,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);
    }
}

 

 

类加载与自定义类加载器(67)

标签:

原文地址:http://www.cnblogs.com/zhenghongxin/p/4519438.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!