看到JAVA书上说到如题的话,不是特别理解“不能互相访问”的含义,比如一个简单的类:
public class CLTest {
public void parse(String s){
System.out.println(s);
}
public static void main(String[] args) {
String str = new String("ok"); //String 类是又启动类装载器装载的
CLTest test = new CLTest(); //CLTest类是有系统类装载器装载的
test.parse(str); //这叫做互相访问吗?
}
}
答案当然肯定不是,那么什么样的访问才叫做“互相访问”?
还有在上边的例子里,String类和CLTest类拥有不同的“命名空间”吗(因为书上说不同的类装载器装载的类有不同的命名空间)?
我肯定是没有理解这个概念,请大家指教。
public class CLTest {
public void parse(String s){
System.out.println(s);
}
public static void main(String[] args) {
String str = new String("ok"); //String 类是又启动类装载器装载的
CLTest test = new CLTest(); //CLTest类是有系统类装载器装载的
test.parse(str); //这叫做互相访问吗?
}
}
答案当然肯定不是,那么什么样的访问才叫做“互相访问”?
还有在上边的例子里,String类和CLTest类拥有不同的“命名空间”吗(因为书上说不同的类装载器装载的类有不同的命名空间)?
我肯定是没有理解这个概念,请大家指教。
解决方案 »
- 新手求救
- 如何将String转化为outputStream?
- 在控制台输入任意一个数(1-9),输出该数字的电子图像
- Solaris中装eclipse IDE
- 大家遇没有遇到struts+hibernate的上传图片的问题
- 请教高手,J2SE 1.4.1中javax.crypto里面有没有带RSA加密的算法啊?
- 怎么把sql server8.0的驱动加入到jb8里面去?
- 谁能帮我解释以下几个词语。。。(分数是少,但愿真情回报)
- javadoc问题
- 在html页面可以用&#xxxxx;显示字符,如"A"为A,java中如何得到一个文字的对应&#代码?
- 方法调用返回值问题
- 2个面试题~
你提到的不同的classloader我理解应该是没有直接父子关系(或直系祖先关系)的两个CL之间装载的类不能相互访问。
例如,有如下CL关系:
CL1
|-CL2
|-CL3CL1的Path中加载了Class1那么在CL2,CL3中使用Class1是获取的是同一个Class实例(这里的Class实例不是Object)
CL2加载了自身path中的一个Class2,同时CL3也自身加载了一个Class2。虽然类名称一样但其实是两个完全不同的Class实例。(通过Class的static变量值不同就可以看出)你可以按照这个思路写一个Test看看
我就是不太理解什么叫不能互相访问
也就是它们装载同一个类,会有不同的Class实例。
即使是静态成员,也会有自己单独的内存空间。
每个类装载器都有自己的命名空间,
不同命名空间的两个类是不可见的,
但如果持有类所对应的Class对象的引用,
还是可以访问另一命名空间的类。(我一直这么理解的)
比方说class A 分别被两个分支上的classLoader 加载,分别生成a1,a2两个实例,
如果两个实例都持有对方的引用, 在a1 中调 a2.f();或在a2中调a1.f();都会提示class not found但是通过接口访问没问题,比方说A implements IA ,不同classloader 加载的类可以通过IA的引用互访
如下图中:
bootstrap
ExtClassloader
AppClassloader
-自定义clsloadr1
-自定义clsloadr2
自定义clsloadr1和自定义clsloadr2是平级的,被他们load的class是不能相互访问的。
同时,被处于上层的classloader加载的class不能访问被处于下层的classloader加载的class。
比如有如下的类加载器结构:
bootstrap
ExtClassloader
AppClassloader
-自定义clsloadr1
-自定义clsloadr2
如果用“自定义clsloadr1”加载java.lang.String类,那么根据双亲委派最终bootstrap会加载此类,那么bootstrap类就叫做该类的“定义类加载器”,而包括bootstrap的所有得到该类class实例的类加载器都叫做“初始类加载器”。二,所说的“命名空间”,是指jvm为每个类加载器维护的一个“表”,这个表记录了所有以此类加载器为“初始类加载器”(而不是定义类加载器,所以一个类可以存在于很多的命名空间中)加载的类的列表,所以,题目中的问题就可以解释了:
CLTest是AppClassloader加载的,String是通过加载CLTest的类加载器也就是AppClassloader进行加载,但最终委派到bootstrap加载的(当然,String类其实早已经被加载过了,这里只是举个例子)。所以,对于String类来说,bootstrap是“定义类加载器”,AppClassloader是“初始类加载器”。根据刚才所说,String类在AppClassloader的命名空间中(同时也在bootstrap,ExtClassloader的命名空间中,因为bootstrap,ExtClassloader也是String的初始类加载器),所以CLTest可以随便访问String类。这样就可以解释“处在不同命名空间的类,不能直接互相访问”这句话了。三,一个类,由不同的类加载器实例加载的话,会在方法区产生两个不同的类,彼此不可见,并且在堆中生成不同Class实例。四,那么由不同类加载器实例(比如-自定义clsloadr1,-自定义clsloadr2)所加载的classpath下和ext下的类,也就是由我们自定义的类加载器委派给AppClassloader和ExtClassloader加载的类,在内存中是同一个类吗?
所有继承ClassLoader并且没有重写getSystemClassLoader方法的类加载器,通过getSystemClassLoader方法得到的AppClassloader都是同一个AppClassloader实例,类似单例模式。
在ClassLoader类中getSystemClassLoader方法调用私有的initSystemClassLoader方法获得AppClassloader实例,在initSystemClassLoader中:
sun.misc.Launcher l = sun.misc.Launcher.getLauncher();
scl = l.getClassLoader();
AppClassloader是sun.misc.Launcher类的内部类,Launcher类在new自己的时候生成AppClassloader实例并且放在自己的私有变量loader里:
loader = AppClassLoader.getAppClassLoader(extclassloader);
值得一提的是sun.misc.Launcher类使用了一种类似单例模式的方法,即既提供了单例模式的接口getLauncher()又把构造函数设成了public的。但是在ClassLoader中是通过单件模式取得的Launcher 实例的,所以我们写的每个类加载器得到的AppClassloader都是同一个AppClassloader类实例。
这样的话得到一个结论,就是所有通过正常双亲委派模式的类加载器加载的classpath下的和ext下的所有类在方法区都是同一个类,堆中的Class实例也是同一个。