- 浏览: 409281 次
- 性别:
- 来自: 秦皇岛
最新评论
-
prayjourney:
了解了,讲的不错
DataInputStream和DataOutputStream类 -
niunianss:
将字节退回的时候,需要添加判断,最后一个字符是英文时,实际数组 ...
PushbackInputStream -
cctt_1:
不要误人子弟,那根本就不是 解释器模式!!!那是Composi ...
Interpreter(解释器)模式 -
java-大神:
[i][i]引用引用引用引用[/img][/img][/img ...
BufferedReader和BufferedWriter -
百合不是茶:
你的程序在文件输入输出流中传入agrs[0]时,会报错越界 ...
DataInputStream和DataOutputStream类
1. 当在命令行模式下执行java XXX.class 指令后,java运行程序会尝试找到JRE安装的所在目录,然后寻找jvm.dll(默认是在JRE目录下bin\client目录中),接着启动JVM并进行初始化动作,产生Bootstrap Loader,Bootstrap Loader会加载Extended Loader,并设定Extended Loader的parent为Bootstrap Loader。Bootstrap Loader会加载System Loader,并将System Loader的parent设定为Extended Loader。
2. Bootstrap Loader通常由C编写而成,Extended Loader是由Java编写而成,实际是对应于sun.misc.Launcher$ExtClassLoader(Launcher中的内部类)。System Loader是由Java编写而成,实际对应于sun.misc.Launcher$AppClassLoader(Launcher中的内部类)。
3.java程序启动与加载类的顺序如下:java XXX.class----->找到JRE目录----->找到jvm.dll---->JVM初始化--->产生BootStrap Loader ----(parent)---->载入ExtClassLoader---(parent)---->载入AppClassLoader---->载入XXX.class。
4. BootStrap Loader会搜索系统参数sun.boot.class.path中指定位置的类,默认是JRE所在目录的classes下的.class文件,或lib目录下.jar文件中的类并加载。可以使用"System.getProperty("sun.boot.class.path")语句来显示sun.boot.class.path中指定的路径。
5. Extended Loader(sun.misc.Launcher$ExtClassLoader)会搜索系统参数java.ext.dirs中指定位置的类,默认是JRE目录下lib\ext\classes目录下的.class文件,或lib\ext目录下的.jar文件中的类并加载。可以使用"System.getProperty("java.ext.dirs")语句来显示java.ext.dirs中指定的路径。
6. System Loader(sun.misc.Launcher$AppClassLoader)会搜索系统参数java.class.path中指定位置的类,也就是Classpath所指定的路径,默认是当前工作路径下的.class文件。可以使用System.getProperty("java.class.path")语句来显示java.class.path中指定的路径,在使用java运行程序时,也可以加上-cp来覆盖原有的Classpath设定,例如:
java -cp ./classes SomeClass
7.Bootstrap Loader会在JVM启动后产生,然后它会加载Extended Loader并将其parent设为Bootstrap Loader,接着Bootstrap Loader会加载System Loader并将其parent设定为Extended Loader,最后System Loader开始加载指定的类。在加载类时,每个类加载器会先将加载类的任务交给其parent,如果parent找不到,再由自己负责加载。所以在加载时,会以Bootstrap Loader--->Extended Loader---->System Loader的顺序来寻找类,如果都找不到,会抛出NoClassDefFoundError。
8. 类加载器在Java中以java.lang.ClassLoader类型存在,每个类被加载后,都会有一个Class的实例来代表,而每个Class的实例都会记得自己是由哪个ClassLoader加载的。可以由Class的getClassLoader()方法取得加载该类的ClassLoader,而从ClassLoader的getParent()方法可以取得自己的parent。
getClass() getClassLoader() getParent()
实例SomeClass-----------------> class SomeClass-------------------------->AppClassLoader-------------------
getParent()
>ExtClassLoader------------------------->Bootstrap Loader.
下面是一个实际例子。
package ysu.hxy; public class SomeClass { public static void main(String[] args) { //建立SomeClass实例 SomeClass some = new SomeClass(); //取得SomeClass的Class实例 Class c = some.getClass(); //取得ClassLoader ClassLoader loader = c.getClassLoader(); System.out.println(loader); //取得父ClassLoader System.out.println(loader.getParent()); System.out.println(loader.getParent().getParent()); } }
运行结果如下:
D:\hxy>java ysu.hxy.SomeClass
sun.misc.Launcher$AppClassLoader@19821f
sun.misc.Launcher$ExtClassLoader@addbf1
null (这里的null并不是表示ExtClassLoader没有设定parent,而是因为BootStrapLoader通常是由C语言编写的,在Java中并没有一个实际的类来表示它,所以才会显示为null的)
运行结果解释:
ysu.hxy.SomeClass是一个自定义类,在当前工作目录下运行程序,首先AppClassLoader会将加载类的任务交给ExeClassLoader,而ExtClassLoader会将加载类的任务交给Bootstrap Loader.由于Bootstrap Loader在它的路径(sun.boot.class.path)下找不到类,所以由ExtClassLoader来试着寻找。而ExtClassLoader在它的路径(java.ext.dirs)下也找不到类,所以由AppClassLoader来试着寻找,AppClassLoader最后在Classpath(java.class.path)设定下找到指定的类并加载。
如果将SomeClass的.class文件移到JRE目录下的lib\ext\classes下(由于设定了包,所以实际上SomeClass.class要放置在JRE目录下的lib\ext\classes\ysu\hxy下)。重新运行程序(在任何目录下),会看到以下结果:
C:\Documents and Settings\Administrator>java ysu.hxy.SomeClass
sun.misc.Launcher$ExtClassLoader@addbf1
null
Exception in thread "main" java.lang.NullPointerException
at ysu.hxy.SomeClass.main(SomeClass.java:16)
由于SomeClass这次可以在ExtClassLoader的设定路径下找到,所以会由ExtClassLoader来加载SomeClass类。而ExtClassLoader的parent显示为null,所以再由null上尝试调用getParent()方法就会丢出NullPointerException异常。
若再将SomeClass上移至JRE目录下的class目录下(实际上是JRE目录下的classes/ysu/hxy下)。在任意目录下重新运行程序结果如下:
C:\Documents and Settings\Administrator>java ysu.hxy.SomeClass
null
Exception in thread "main" java.lang.NullPointerException
at ysu.hxy.SomeClass.main(SomeClass.java:15)
原因与上面类似。
9.取得ClassLoader的实例之后,可以使用它的loadClass()方法来加载类。使用loadClass()方法加载类时,不会运行静态区块。静态区块的运行会等到真正使用类来建立实例时。如下范例:
package ysu.hxy; public class TestClass2 { static { System.out.println("[运行静态区块]"); } }
package ysu.hxy; public class ForNameDemoV3 { public static void main(String[] args) { try { System.out.println("载入TestClass2"); ClassLoader loader = ForNameDemoV3.class.getClassLoader(); //使用loader加载类,不会执行静态区块,要等到创建对象时才执行静态区块 Class c = loader.loadClass("ysu.hxy.TestClass2"); System.out.println("使用TestClass2声明参考名称"); TestClass2 test = null; System.out.println("使用TestClass2创建对象"); test = new TestClass2(); } catch(ClassNotFoundException e) { System.out.println("找不到指定的类"); } } }
运行结果:
D:\hxy>java ysu.hxy.ForNameDemoV3
载入TestClass2
使用TestClass2声明参考名称
使用TestClass2创建对象
[运行静态区块]
10. 使用自己的ClassLoader:
ExtClassLoader与AppClassLoader都是java.net.URLClassLoader的子类,可以在使用java启动程序时,使用以下的指令来指定ExtClassLoader的搜索路径:
java -D java.ext.dirs=c:\workspace\YourClass
可以在使用java启动程序时,使用-classpath或-cp来指定AppClassLoader的搜索路径,也就是设定ClassPath:
java -classpath c:\workspace\YourClass
ExtClassLoader与AppClassLoader在程序启动后会在虚拟机中存在一份,在程序运行过程中无法再改变它的搜索路径。如果在程序运行过程中,打算动态从其他路径加载类,就要产生新的类加载器。
可以使用URLClassLoader来产生新的类加载器,它需要java.net.URL作为其参数来指定类加载的搜索路径。例如:
URL url = new URL("file:/d:/workspace/"); ClassLoader urlClassLoader = new URLClassLoader(new URL[] {url}); Class c = urlClassLoader.loadClass("SomeClass");
ClassLoader 是J2SE的标准API之一,可以在rt.jar中找到,因而会由Bootstrap Loader来载入ClassLoader类。在新增了ClassLoader实例后,可以使用它的loadClass()方法指定要加载的类名称。在新增ClassLoader 时,会自动将新增的ClassLoader 的parent设定为AppClassLoader,并在每次加载类时,先委托parent代为搜索。所以上例中搜索SomeClass类时,会一直往上委托至Bootstrap Loader开始搜索,接着是ExtClassLoader、AppClassLoader。如果都找不到,才使用新建的ClassLoader搜索。
Java的类加载器层次架构除了可以达到动态加载类的目的之外,还有着安全上的考虑。因为每次寻找类时都是委托parent开始寻找,所以除非有人可以侵入您的计算机,置换掉标准J2SE API与您自己安装的延伸包,否则不可能通过编写自己的类载入器来载入恶意类,以置换掉标准J2SE API与您自己安装的延伸包。
由于每次的类载入是由子ClassLoader委托父ClassLoader先尝试加载,但父ClassLoader看不到子ClassLoader,所以同一层次的子ClassLoader不会被误用,从而避免了加载错误类的可能性。
由同一个ClassLoader加载的类文件,会只有一份Class实例。如果同一个类文件由两个不同的ClassLoader载入,则会有两份不同的Class实例。注意,如果有两个不同的ClassLoader搜索同一个类,而在parent的AppClassLoader搜索路径中可以找到指定类,则Class实例就会有一个,因为两个不同的ClassLoader都是在委托父ClassLoader时找到该类的。如果父ClassLoader找不到,而是由各自的ClassLoader搜索到,则Class的实例会有两份。
下面范例是一个简单的示范,可用来测试加载路径与Class实例是否为同一对象。
package ysu.hxy; import java.net.MalformedURLException; import java.net.URL; import java.net.URLClassLoader; public class ClassLoaderDemo { public static void main(String[] args) { try { //测试路径 String classPath = args[0]; //测试类 String className = args[1]; URL url1 = new URL(classPath); //建立ClassLoader ClassLoader loader1 = new URLClassLoader(new URL[]{url1}); //加载指定类 Class c1 = loader1.loadClass(className); //显示类描述 System.out.println(c1); URL url2 = new URL(classPath); //建立ClassLoader ClassLoader loader2 = new URLClassLoader(new URL[]{url2}); //加载指定类 Class c2 = loader2.loadClass(className); //显示类描述 System.out.println(c2); System.out.println("C1与C2为同一实例?"+(c1 == c2)); } catch(ArrayIndexOutOfBoundsException e) { System.out.println("没有指定类加载路径与名称"); } catch(MalformedURLException e) { System.out.println("加载路径错误"); } catch(ClassNotFoundException e) { System.out.println("找不到指定的类"); } } }
可以设计任意一个测试类,例如Test,其中classpath可以输入不为ExtClassLoader或AppClassLoader的搜索路径,例如file:d:/workspace/。这样同一个类会分由两个ClassLoader载入,结果会有两份Class实例。测试c1与c2是否为同一实例时,结果会显示false。一个运行结果如下:
D:\hxy>java ysu.hxy.ClassLoaderDemo file:/d:/ Test3
class Test3
class Test3
C1与C2为同一实例?false
如果在运行程序时,以-cp将file:/d:/加入为Classpath的一部分,由于两个ClassLoader的parent都是AppClassLoader,而AppClassLoader会在ClassPath中找到指定的类,所以最后会只有一个指定的类的Class实例。测试c1与c2是否为同一实例时,结果会显示true。一个运行结果如下:
D:\hxy>java -cp .;d:\ ysu.hxy.ClassLoaderDemo file:d:/ Test3
class Test3
class Test3
C1与C2为同一实例?true
使用-cp指定Classpath时,会覆盖原有的Classpath定义,也就是连现行工作目录的路径也覆盖了。
发表评论
-
内部类总结
2009-11-27 14:28 1222一、方法及作用域内的内部类:1.在一个方法内定义的类2.在一个 ... -
finalize()方法终结条件验证 示例代码
2009-09-20 09:23 1348package Initialization; clas ... -
Proxy(代理)模式二
2009-05-15 21:16 15532. 重新思考图像代理: 现在需要思考设计模式是否 ... -
Junit简介
2009-04-08 17:46 16501. 单元测试(Unit Test) 一个单元(Un ... -
Ant简介
2009-04-08 13:10 18531. Ant可以自动完成的任务: (1)编译Java源代 ... -
专题制作--文字编辑器(文字编辑与保存)
2009-04-08 10:43 22161. 文字编辑与保存: (1). 打开文件的处理流 ... -
专题制作--文字编辑器(逻辑实现部分)
2009-04-07 22:35 19061. 事件处理: 在Java中事件以具体的对象来表 ... -
专题制作--文字编辑器(接口部分)
2009-04-07 20:28 21491. Swing入门: 若要使用J2SE来开发窗口应用 ... -
信息绑定(国际化处理)
2009-04-07 20:02 15851. 程序中的一些文字信息可以将之定义在一个属性文件中,而不定 ... -
日志(Logging)
2009-04-07 16:14 17281. 日志(Logging) 程序不免会出现错误,当 ... -
Java中的日期和时间
2009-04-07 11:26 19591. 使用Date: 使用System.cu ... -
meta-annotation
2009-04-07 09:23 30831. 所谓meta-annotation就是Annotati ... -
Annotion
2009-04-06 23:05 16841. Annotation对程序运行没有影响,它的目的在于对编 ... -
使用反射生成与操作对象(二)
2009-04-06 17:04 17561. 修改成员值: 尽管直接存取类的域成员是不被鼓励的 ... -
使用反射生成与操作对象(一)
2009-04-06 15:16 21101. 使用反射机制,可以在运行时期动态加载类并生成对象,操作对 ... -
Java中的反射(一)
2009-04-06 09:43 13931. Java提供的反射机制允许您在运行时动态加载类、查看类信 ... -
容器类的线程安全及ThreadLocal类
2009-04-05 21:28 30571. 容器类默认没有考虑 ... -
wait()和notify()
2009-04-05 19:06 13731. wait()、notify()、notifyAll() ... -
Java线程之同步化(Synchronized)主题
2009-04-05 16:44 27151. 如果一个对象所持有的数据可以被多线程同时共享存取,必须 ... -
Java线程(三)
2009-04-05 15:37 18931. Java中的每个线程都 ...
相关推荐
总的来说,Java反射机制是面向对象编程的一个重要补充,它扩展了Java程序的动态性,允许程序员在运行时访问和操作类的内部结构,增强了代码的灵活性。理解和熟练掌握反射技术,对于提升Java编程能力,尤其是处理复杂...
Java反射是Java编程语言中的一个强大特性,它允许在运行时检查类、接口、字段和方法的信息,并且能够在运行时动态地创建对象和调用方法。这个特性使得Java具有了高度的灵活性,常用于框架开发、插件系统、元编程等...
Java反射是Java编程语言中的一个重要特性,它允许程序在运行时动态地获取类的信息并操作类的对象。在Java中,反射机制提供了强大的能力,包括在运行时检查类的结构、创建对象实例、调用方法以及访问和修改字段值。...
#### 二、Java中的类反射 Java中的反射功能允许运行中的程序对自身进行检查,并能直接操作程序的内部属性,即所谓的“自省”能力。这是Java区别于其他编程语言如Pascal、C或C++的一个重要特性,因为在这些语言中...
在Java反射中,访问类的字段和方法也十分灵活。`getFields()`返回所有public字段,`getField()`获取指定名称的public字段。对于私有或其他访问修饰符的字段,可以使用`getDeclaredFields()`和`getDeclaredField()`。...
Java反射机制允许运行中的程序检查自身,并能直接操作程序的内部属性。这是其他许多编程语言(如Pascal、C或C++)不具备的能力。 **1.1 Reflection的工作机制** 为了展示反射如何工作,我们来看一个简单的例子: ...
在Java编程中,反射(Reflection)是一个强大的工具,它允许我们在运行时检查和操作类、接口、字段和方法。然而,反射操作通常比直接的Java代码执行慢,因为它涉及到动态类型检查和方法调用。因此,为了提高性能,...
#### 二、Java反射机制简介 Java反射机制允许程序在运行时获取类的信息,这使得Java具有了一定程度上的动态性。具体来说,Java反射机制提供了以下功能: 1. **获取类的信息**:可以在运行时获取类的修饰符、父类、...
在Java反射中,针对类的不同组成部分(构造函数、字段和方法),`java.lang.Class`类提供了多种反射调用方式来获取信息。以下是几种常用的反射调用: - **获取构造函数**:`Constructor getConstructor(Class[] ...
java反射机制java反射机制.zipjava反射机制.zipjava反射机制.zipjava反射机制.zipjava反射机制.zipjava反射机制.zipjava反射机制.zipjava反射机制.zipjava反射机制.zipjava反射机制.zipjava反射机制.zipjava反射机制...
Java反射是Java编程语言中的一个强大工具,它允许运行中的Java程序对自身进行检查并且可以直接操作程序的内部属性。在Java中,反射主要用于在运行时分析类和对象,包括访问私有成员、调用私有方法、创建对象、获取类...
反射机制的核心是`Class`类,它代表了Java中的每一个类型。通过`Class.forName()`方法,我们可以根据类名获取`Class`对象。一旦有了`Class`对象,我们就可以实例化对象,调用构造器,以及获取类的信息,如类名、...
Java反射是Java编程语言中的一个强大特性,它允许运行时的程序访问并操作类、接口、字段和方法等信息,即使这些信息在编译时并未明确知晓。在Java中,反射通常通过`java.lang.Class`类和相关的API来实现。本实例将...
在本文中,我们将深入探讨如何使用Java反射来获取并执行某个特定的方法。 首先,我们需要了解Java反射的基本概念。`java.lang.Class`类是反射的核心,它代表了Java中的每一个类。我们可以通过以下方式获取到一个...
Java反射机制是Java编程语言中的一个强大特性,它允许运行中的Java程序对自身进行检查并且可以直接操作程序的内部属性。这个特性使得Java具有了高度的灵活性和动态性,尤其是在处理元数据、创建对象、调用私有方法...
#### 二、Java反射的应用场景 Java反射广泛应用于框架开发、动态代理、测试工具等领域。例如,在Spring框架中,反射被用来动态创建对象和调用方法;在Hibernate等ORM框架中,反射用于处理数据库记录与Java对象之间...
在Java编程语言中,反射(Reflection)是一种强大的工具,它允许程序在运行时检查和操作类、接口、字段和方法等对象。枚举(Enumeration)是Java中的一个特殊类类型,用于定义一组常量。本项目"test-enum-demo-...
2. **类对象** - 在Java中,每个类都有一个对应的Class对象,它是Java反射的入口。我们可以通过Class对象来实例化对象、获取类的构造器、方法和字段信息。例如,`Class<?> clazz = Class.forName("全限定类名");` 这...
Java反射机制是Java编程语言中一个强大的特性,它允许程序在运行时动态地访问、检测和修改类、接口、字段和方法等对象。然而,反射操作通常会引入额外的开销,这在性能敏感的应用场景下可能成为一个瓶颈。本文将深入...