`
xyheqhd888
  • 浏览: 409249 次
  • 性别: Icon_minigender_1
  • 来自: 秦皇岛
社区版块
存档分类
最新评论

Java中的反射(一)

阅读更多

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

分享到:
评论

相关推荐

    java中反射的概念

    总的来说,Java反射机制是面向对象编程的一个重要补充,它扩展了Java程序的动态性,允许程序员在运行时访问和操作类的内部结构,增强了代码的灵活性。理解和熟练掌握反射技术,对于提升Java编程能力,尤其是处理复杂...

    Java方法反射调用demo

    Java反射是Java编程语言中的一个强大特性,它允许在运行时检查类、接口、字段和方法的信息,并且能够在运行时动态地创建对象和调用方法。这个特性使得Java具有了高度的灵活性,常用于框架开发、插件系统、元编程等...

    java反射 java反射 java反射java反射

    Java反射是Java编程语言中的一个重要特性,它允许程序在运行时动态地获取类的信息并操作类的对象。在Java中,反射机制提供了强大的能力,包括在运行时检查类的结构、创建对象实例、调用方法以及访问和修改字段值。...

    Java中的反射

    在Java中,反射提供了一种强大的工具,它可以让程序在运行时动态地获取类信息,而不需要预先知道类的具体信息。 Java中的反射技术使得程序具备高度的灵活性,可以在运行时动态地加载、检测和调用类的方法和字段等。...

    java例子 java反射

    Java反射是Java编程语言中的一个强大特性,它允许运行中的Java程序对自身进行检查并且可以直接操作程序的内部属性。在Java中,反射机制的核心类是`java.lang.Class`,它代表了类的信息。当我们需要在运行时动态地...

    Java中的反射机制

    Java反射机制允许运行中的程序检查自身,并能直接操作程序的内部属性。这是其他许多编程语言(如Pascal、C或C++)不具备的能力。 **1.1 Reflection的工作机制** 为了展示反射如何工作,我们来看一个简单的例子: ...

    java之反射优化(缓存思路)源码

    在Java编程中,反射(Reflection)是一个强大的工具,它允许我们在运行时检查和操作类、接口、字段和方法。然而,反射操作通常比直接的Java代码执行慢,因为它涉及到动态类型检查和方法调用。因此,为了提高性能,...

    java面试题--反射机制

    ### Java反射机制详解 #### 一、引言 在Java面试中,经常会出现与反射...以上内容不仅解释了Java反射机制的相关知识点,还通过示例代码进行了实践演示,希望能够帮助你在Java面试中更好地理解和运用这一重要特性。

    java 通过反射获取枚举类,及枚举类的值,枚举类枚举实例名

    在Java编程语言中,反射(Reflection)是一种强大的工具,它允许程序在运行时检查和操作类、接口、字段和方法等对象。枚举(Enumeration)是Java中的一个特殊类类型,用于定义一组常量。本项目"test-enum-demo-...

    反射实例-JAVA反射机制

    在Java反射中,针对类的不同组成部分(构造函数、字段和方法),`java.lang.Class`类提供了多种反射调用方式来获取信息。以下是几种常用的反射调用: - **获取构造函数**:`Constructor getConstructor(Class[] ...

    java反射以及复制一个bean的值到另一个bean中。

    总之,Java反射提供了一种强大的机制,允许我们在运行时检查和操作类和对象。它在许多场景下非常有用,比如框架开发、动态代理、序列化、元数据操作等。同时,复制bean的值也是常见的需求,尤其在数据处理和对象模型...

    Java 1.5 反射机制

    反射机制的核心是`Class`类,它代表了Java中的每一个类型。通过`Class.forName()`方法,我们可以根据类名获取`Class`对象。一旦有了`Class`对象,我们就可以实例化对象,调用构造器,以及获取类的信息,如类名、...

    java反射,获取所有属性、方法以及List集合类

    Java反射是Java编程语言中的一个强大工具,它允许运行中的Java程序对自身进行检查并且可以直接操作程序的内部属性。在Java中,反射主要用于在运行时分析类和对象,包括访问私有成员、调用私有方法、创建对象、获取类...

    java反射机制.zip

    java反射机制java反射机制.zipjava反射机制.zipjava反射机制.zipjava反射机制.zipjava反射机制.zipjava反射机制.zipjava反射机制.zipjava反射机制.zipjava反射机制.zipjava反射机制.zipjava反射机制.zipjava反射机制...

    java反射-英文版反射规范

    虽然Java反射提供了一种非常灵活的方式来操作类和对象,但在实际应用中也需要注意几个关键点: ##### 1. **安全性** - 使用反射可能会导致安全漏洞,因为可以访问到私有成员。 - 在设计应用程序时应限制反射的...

    Java反射经典实例

    Java反射是Java编程语言中的一个强大特性,它允许运行时的程序访问并操作类、接口、字段和方法等信息,即使这些信息在编译时并未明确知晓。在Java中,反射通常通过`java.lang.Class`类和相关的API来实现。本实例将...

    Java反射性能测试分析

    Java反射机制是Java编程语言中一个强大的特性,它允许程序在运行时动态地访问、检测和修改类、接口、字段和方法等对象。然而,反射操作通常会引入额外的开销,这在性能敏感的应用场景下可能成为一个瓶颈。本文将深入...

    java反射例子,封装了一个反射帮助类

    Java反射是Java编程语言中的一个重要特性,它允许运行时访问和操作类、接口、字段和方法等信息。在Java中,反射提供了动态类型的能力,使我们可以在程序运行时检查类的信息,创建和调用对象的方法,甚至修改对象的...

    JAVA反射机制的入门代码

    Java反射机制是Java编程语言中的一个强大特性,它允许运行中的Java程序对自身进行检查并且可以直接操作程序的内部属性。这个特性使得Java具有了高度的灵活性和动态性,尤其是在处理元数据、创建对象、调用私有方法...

Global site tag (gtag.js) - Google Analytics