- 浏览: 403706 次
- 性别:
- 来自: 秦皇岛
最新评论
-
prayjourney:
了解了,讲的不错
DataInputStream和DataOutputStream类 -
niunianss:
将字节退回的时候,需要添加判断,最后一个字符是英文时,实际数组 ...
PushbackInputStream -
cctt_1:
不要误人子弟,那根本就不是 解释器模式!!!那是Composi ...
Interpreter(解释器)模式 -
java-大神:
[i][i]引用引用引用引用[/img][/img][/img ...
BufferedReader和BufferedWriter -
百合不是茶:
你的程序在文件输入输出流中传入agrs[0]时,会报错越界 ...
DataInputStream和DataOutputStream类
1. Java提供的反射机制允许您在运行时动态加载类、查看类信息、生成对象或操作生成的对象。反射机制的一个应用实例,就是在整合式开发环境中所提供的方法提示或是类查看工具。另外像JSP中的JavaBean自动收集请求信息也使用到反射,而一些软件开发框架也常见到反射机制的使用,以达到动态加载使用者自定义类的目的。
2. 简介Class与类加载:
Java在真正需要使用一个类时才会加以加载。一个java.lang.Class对象代表了Java应用程序在运行时所加载的类或接口实例,也用来表达enum(属于类的一种)、annotation(属于接口的一种)、数组、初始类型、void。Class类没有公开的构造函数。Class对象由JVM自动产生,每当一个类被加载时,JVM就自动为其生成一个Class对象。
3. 可以通过Object的getClass()方法来取得每个对象对应的Class对象,或者是通过class常量,在取得Class对象之后,就可以操作Class对象上的一些公开方法来取得类的基本信息。
package onlyfun.caterpillar; public class ClassDemo { public static void main(String[] args) { String name = "caterpillar"; Class stringClass = name.getClass(); System.out.println("类名称:" + stringClass.getName()); System.out.println("是否为接口:" + stringClass.isInterface()); System.out.println("是否为基本类型:" + stringClass.isPrimitive()); System.out.println("是否为数组对象:" + stringClass.isArray()); System.out.println("父类名称:" + stringClass.getSuperclass().getName()); } }
也可以直接使用以下的方式来取得String类的Class对象:
Class stringClass = String.class;
Java在真正需要类时才会加载类,所谓真正需要通常指的是要使用指定的类生成对象时(或是使用者指定要加载类时,例如使用Class.forName()加载类,或是使用ClassLoader的loadClass()加载类)。使用类名称来声明参考名称并不会类的加载。下面是印证这个说法的一个测试类。
package ysu.hxy; public class TestClass { static { System.out.println("类被载入"); } }
上此范例中定义了一个静态区块,默认在类第一次被加载时会运行静态区块,通过运行在命令模式下的显示信息,可以了解类何时被加载:
package ysu.hxy; public class LoadClassTest { public static void main(String[] args) { TestClass test = null; System.out.println("声明TestClass参考名称"); test = new TestClass(); System.out.println("生成TestClass实例"); } }
运行结果如下:
D:\hxy>java ysu.hxy.LoadClassTest
声明TestClass参考名称
类被载入
生成TestClass实例
从运行结果可以看出,声明参考名称并不导致TestClass类被加载,而是在使用new生成对象时才会加载类。
4. Class的信息是在编译时期就被加入至.class文件中,这是Java支持运行时期类型辨识(Run-Time Type Information或Run-Time Type Identification,RTTI)的一种方式:在编译时期编译器会先检查对应的.class文件,而运行时期JVM在使用某类时,会先检查对应的Class对象是否已经加载,如果没有加载,则会寻找对应的.class文件并载入。一个类在JVM中只会有一个Class实例,每个类的实例都会记得自己是由哪个Class实例所生成,可以使用getClass()或.class来取得Class实例。
5. 在Java中,数组也是一个对象,也有其对应的Class实例。这个对象由具有相同元素与维度的数组所共享,而基本类型和关键词void,也都有对应的Class对象。可以用类常量来取得这些对象。
package ysu.hxy; public class ClassDemo2 { public static void main(String[] args) { System.out.println(boolean.class); System.out.println(void.class); int[] iarr = new int[10]; System.out.println(iarr.getClass().toString()); double[] darr = new double[10]; System.out.println(darr.getClass().toString()); } }
6. 使用Class.forName()加载类:
Class的静态forName()方法可以实现动态加载类的目的。如下例:
package ysu.hxy; public class ForNameDemo { public static void main(String[] args) { try { Class c = Class.forName(args[0]); System.out.println("类名称:" + c.getName()); System.out.println("是否为接口:" + c.isInterface()); System.out.println("是否为基本类型:" + c.isPrimitive()); System.out.println("是否为数组:" + c.isArray()); System.out.println("父类:" + c.getSuperclass().getName()); } catch(ArrayIndexOutOfBoundsException e) { System.out.println("没有指定类名称"); } catch(ClassNotFoundException e) { System.out.println("找不到指定的类"); } } }
一个运行的结果如下:
D:\hxy>java ysu.hxy.ForNameDemo java.util.Scanner
类名称:java.util.Scanner
是否为接口:false
是否为基本类型:false
是否为数组:false
父类:java.lang.Object
Class的静态forName()方法有两个版本,上面使用的是只指定类名称的版本,而另一个版本可以让您指定类名称、加载类时是否运行静态区块、指定类加载器:
static Class forName(String name,boolean initialize,ClassLoader loader)
默认在加载类的时候,如果类中有定义静态区块则会运行它。你可以使用forName()的第二个版本,将initialize设定为false,这样在加载类时并不会立即运行静态区块,而会在使用类建立对象时才运行静态区块。以下是一个测试类:
package ysu.hxy; public class TestClass2 { static { System.out.println("[运行静态区块]"); } }
使用第一个版本forName()方法的测试代码:
package ysu.hxy; public class ForNameDemoV1 { public static void main(String[] args) { try { System.out.println("载入TestClass2"); Class c = Class.forName("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.ForNameDemoV1
载入TestClass2
[运行静态区块]
使用TestClass2声明参考名称
使用TestClass2建立对象
使用第二个版本的forName()代码:
package ysu.hxy; public class ForNameDemoV2 { public static void main(String[] args) { try { System.out.println("载入TestClass2"); Class c = Class.forName( "ysu.hxy.TestClass2", false, // 加载类时不运行静态区块, 在使用类建立对象时才运行静态区块,此版本的forName()需要一个类加载器,这里使用的是主线程的类加载器 Thread.currentThread().getContextClassLoader()); 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.ForNameDemoV2
载入TestClass2
使用TestClass2声明参考名称使用TestClass2建立对象
[运行静态区块]
7. 从Class中获取信息:Class对象表示所加载的类,取得Class对象之后,就可以取得与类相关联的信息,像包、构造函数、方法成员、域成员等的信息,而每个信息,也会有相应的类类型。如包的对应类型是java.lang.Package;构造函数的对应类型是java.lang.reflect.Constructor;方法成员的对应类型是java.lang.reflect.Method;域成员的对应类型是java.lang.reflect.Field等。可以取回Field、Constructor、Method等对象,它们分别代表域成员、构造函数与方法成员。下面是个范例:
package ysu.hxy; import java.lang.reflect.*; public class SimpleClassViewer { public static void main(String[] args) { try { Class c = Class.forName(args[0]); //取得包代表对象 Package p = c.getPackage(); System.out.printf("package %s;%n",p.getName()); //取得类型修饰,像class,interface int m = c.getModifiers(); System.out.print(Modifier.toString(m) + " "); if(Modifier.isInterface(m)){ System.out.print("interface "); } else { System.out.print("class "); } System.out.println(c.getName() + " {"); //取得声明的域成员对象 Field[] fields = c.getDeclaredFields(); for(Field field: fields) { //显示权限修饰,像public、protected、private System.out.print("\t"+Modifier.toString(field.getModifiers())); //显示类型名称 System.out.print(" " + field.getType().getName() + " "); //显示域成员名称 System.out.println(field.getName()+";"); } //取得声明的构造函数代表对象 Constructor[] constructors= c.getDeclaredConstructors(); for(Constructor constructor: constructors) { //显示权限修饰,像public,private,protected System.out.print("\t"+Modifier.toString(constructor.getModifiers())); //显示构造函数名称 System.out.println(" " +constructor.getName()+"();"); } //取得声明的方法成员代表对象 Method[] methods = c.getDeclaredMethods(); for(Method method:methods) { //显示权限修饰 System.out.print("\t"+Modifier.toString(method.getModifiers())); //显示返回类型 System.out.print(" "+method.getReturnType().getName()+ " "); //显示方法名称 System.out.println(" " + method.getName()+"();"); } System.out.println("}"); } catch(ArrayIndexOutOfBoundsException e) { System.out.println("没有指定类"); } catch(ClassNotFoundException e) { System.out.println("找不到指定类"); } } }
运行结果:
D:\hxy>java ysu.hxy.SimpleClassViewer java.util.ArrayList
package java.util;
public class java.util.ArrayList {
private static final long serialVersionUID;
private transient [Ljava.lang.Object; elementData;
private int size;
public java.util.ArrayList();
public java.util.ArrayList();
public java.util.ArrayList();
public boolean add();
public void add();
public java.lang.Object get();
public java.lang.Object clone();
public int indexOf();
public void clear();
public boolean contains();
public boolean isEmpty();
public int lastIndexOf();
public boolean addAll();
public boolean addAll();
public int size();
public [Ljava.lang.Object; toArray();
public [Ljava.lang.Object; toArray();
public boolean remove();
public java.lang.Object remove();
private void writeObject();
private void readObject();
public java.lang.Object set();
public void ensureCapacity();
protected void removeRange();
public void trimToSize();
private void RangeCheck();
private void fastRemove();
}
发表评论
-
内部类总结
2009-11-27 14:28 1179一、方法及作用域内的内部类:1.在一个方法内定义的类2.在一个 ... -
finalize()方法终结条件验证 示例代码
2009-09-20 09:23 1296package Initialization; clas ... -
Proxy(代理)模式二
2009-05-15 21:16 15182. 重新思考图像代理: 现在需要思考设计模式是否 ... -
Junit简介
2009-04-08 17:46 15521. 单元测试(Unit Test) 一个单元(Un ... -
Ant简介
2009-04-08 13:10 17401. Ant可以自动完成的任务: (1)编译Java源代 ... -
专题制作--文字编辑器(文字编辑与保存)
2009-04-08 10:43 21491. 文字编辑与保存: (1). 打开文件的处理流 ... -
专题制作--文字编辑器(逻辑实现部分)
2009-04-07 22:35 18541. 事件处理: 在Java中事件以具体的对象来表 ... -
专题制作--文字编辑器(接口部分)
2009-04-07 20:28 21031. Swing入门: 若要使用J2SE来开发窗口应用 ... -
信息绑定(国际化处理)
2009-04-07 20:02 15301. 程序中的一些文字信息可以将之定义在一个属性文件中,而不定 ... -
日志(Logging)
2009-04-07 16:14 16221. 日志(Logging) 程序不免会出现错误,当 ... -
Java中的日期和时间
2009-04-07 11:26 19111. 使用Date: 使用System.cu ... -
meta-annotation
2009-04-07 09:23 30171. 所谓meta-annotation就是Annotati ... -
Annotion
2009-04-06 23:05 16361. Annotation对程序运行没有影响,它的目的在于对编 ... -
使用反射生成与操作对象(二)
2009-04-06 17:04 17231. 修改成员值: 尽管直接存取类的域成员是不被鼓励的 ... -
使用反射生成与操作对象(一)
2009-04-06 15:16 20511. 使用反射机制,可以在运行时期动态加载类并生成对象,操作对 ... -
Java中的反射(二)
2009-04-06 10:42 19801. 当在命令行模式下执行java XXX.class 指令后 ... -
容器类的线程安全及ThreadLocal类
2009-04-05 21:28 30161. 容器类默认没有考虑 ... -
wait()和notify()
2009-04-05 19:06 13231. wait()、notify()、notifyAll() ... -
Java线程之同步化(Synchronized)主题
2009-04-05 16:44 26751. 如果一个对象所持有的数据可以被多线程同时共享存取,必须 ... -
Java线程(三)
2009-04-05 15:37 18531. Java中的每个线程都 ...
相关推荐
java反射 java反射java反射java反射java反射java反射java反射java反射java反射java反射java反射java反射java反射java反射
通过学习后,保证初学者对反射有一个很好的理解,有助于之后对反射的进一步...其中LEAD/LEAD++ 、OpenC++ 、MetaXa和OpenJava等就是基于反射机制的语言。最近,反射机制也被应用到了视窗系统、操作系统和文件系统中。
Java中的反射
java学习例子 java java例子 java反射
Java中的反射机制Java中的反射机制Java中的反射机制
Java反射 调用空参方法 调用Object类型参数的方法 调用基本类型参数的方法 调用基本类型数组参数的方法 调用String数组参数的方法 调用Object数组参数的方法 调用私有方法
java之反射优化(缓存思路)源码 编程中,抽象是一个重要思想。一个无状态(业务less)的抽象化工具方法,往往可以起到减少代码量的作用,使得程序员能更加focus更加重要的业务逻辑中去
JAVA 反射机制应用JAVA 反射机制应用JAVA 反射机制应用JAVA 反射机制应用JAVA 反射机制应用JAVA 反射机制应用JAVA 反射机制应用
java 反射中操作数组 java 反射中操作数组 java 反射中操作数组
java反射与EJBjava反射与EJBjava反射与EJBjava反射与EJBjava反射与EJBjava反射与EJB
java反射机制java反射机制.zipjava反射机制.zipjava反射机制.zipjava反射机制.zipjava反射机制.zipjava反射机制.zipjava反射机制.zipjava反射机制.zipjava反射机制.zipjava反射机制.zipjava反射机制.zipjava反射机制...
这是一本关于Java反射机制论文集合的书。其中包含基本的理论及各种实例。 所有文章均从网络途径获得,本人只作收集整理的工作,个别地方有改动。每篇文章的标题遵循原作者的表述,本人不再作任何更改。 关键字有:...
Java反射经典实例 Java反射经典实例 Java反射经典实例 Java反射经典实例
JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法;这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。 Java...
java 通过反射获取枚举类,及枚举类的值,枚举类枚举实例名。本项目为普通java项目
java反射java反射java反射java反射java反射
JAVA提高反射效率,Java代码精粹,高手实现,反射高效代码
Java反射机制是1.5新增的功能,希望大家能学习愉快!
java中反射知识总结 1.Chp16 2.类对象 3.反射包 4.反射的作用
java反射机制源码java反射机制源码java反射机制源码