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又声明了两个类载器:

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);

}

}

以上是 java 类加载与自定义类加载器详解 的全部内容, 来源链接: utcz.com/z/333802.html

回到顶部