简介

        本文用示例介绍Java反射的常用方法、反射实例化对象、取得类结构、应用。

Class常用方法


方法



说明



getName()



返回String形式的该类的名称。



newInstance()



根据某个Class对象产生其对应类的实例,它调用的是此类的默认构造方法(没有默认无参构造器会报错)



getClassLoader()



返回该Class对象对应的类的类加载器。



getSuperClass()



返回某子类所对应的直接父类所对应的Class对象



getComponentType()



如果当前类表示一个数组,则返回表示该数组组件的 Class 对象,否则返回 null。



getConstructor(Class[])



返回当前 Class 对象表示的类的指定的公有构造子对象。



getConstructors()



返回当前 Class 对象表示的类的所有公有构造子对象数组。



getDeclaredConstructor(Class[])



返回当前 Class 对象表示的类的指定已说明的一个构造子对象。



getDeclaredConstructors()



返回当前 Class 对象表示的类的所有已说明的构造子对象数组。



getDeclaredField(String)



返回当前 Class 对象表示的类或接口的指定已说明的一个域对象。



getDeclaredFields()



返回当前 Class 对象表示的类或接口的所有已说明的域对象数组。



getDeclaredMethod(String, Class[])



返回当前 Class 对象表示的类或接口的指定已说明的一个方法对象。



getDeclaredMethods()



返回 Class 对象表示的类或接口的所有已说明的方法数组。



getField(String)



返回当前 Class 对象表示的类或接口的指定的公有成员域对象。



getFields()



返回当前 Class 对象表示的类或接口的所有可访问的公有域对象数组。



getInterfaces()



返回当前对象表示的类或接口实现的接口。



getMethod(String, Class[])



返回当前 Class 对象表示的类或接口的指定的公有成员方法对象。



getMethods()



返回当前 Class 对象表示的类或接口的所有公有成员方法对象数组,包括已声明的和从父类继承的方法。



isArray()



判定此Class对象所对应的是否是一个数组对象



isInstance(Object)



此方法是 Java 语言 instanceof 操作的动态等价方法。



isInterface()



判定指定的 Class 对象是否表示一个接口类型



isPrimitive()



判定指定的 Class 对象是否表示一个 Java 的基类型。


构造实例化对象

Class.forName和ClassLoader




Class.forName



ClassLoader



关系



Class.forName()方法也是调用的 ClassLoader 来实现的。



灵活度



灵活度低。

例如:加载的类只能是classpath下的



灵活度高。

例如:可以自己编写加载类的方法:比如通过读取类文件的二进制数据,这个时候文件可以不存在ClassPath中。



是否进行初始化



是。

forName方法最后调用 forName0(本地方法),第二个参数设置为了 true,代表对加载的类进行初始化(执行类中的静态代码块、对静态变量赋值等)



否。


forName

@CallerSensitive
public static Class<?> forName(String className)
throws ClassNotFoundException {
Class<?> caller = Reflection.getCallerClass();
return forName0(className, true, ClassLoader.getClassLoader(caller), caller);
}

它有个重载(可以手动选择在加载类时是否要对其类进行初始化。)

public static Class<?> forName(String name, boolean initialize,
ClassLoader loader){}

实例(forName)

package org.example.a;

class Test {
static {
System.out.println("静态代码块");
}
public static final String title = getTitle();

private static String getTitle(){
System.out.println("执行getTitle()");
return "TITLE";
}
}

public class Demo {
public static void main(String[] args) {
Class<?> c = null;
try {
c = Class.forName("org.example.a.Test");
}catch (Exception e){
e.printStackTrace();
}

Test test = null;
try {
test = (Test) c.newInstance();
} catch (Exception e) {
e.printStackTrace();
}
}
}

执行结果

静态代码块
执行getTitle()

实例(ClassLoader)

package org.example.a;

class Test {
static {
System.out.println("静态代码块");
}
public static final String title = getTitle();

private static String getTitle(){
System.out.println("执行getTitle()");
return "TITLE";
}
}

public class Demo {
public static void main(String[] args) {
Class<?> c = null;
try {
c = ClassLoader.getSystemClassLoader().loadClass("org.example.a.Test");
}catch (Exception e){
e.printStackTrace();
}

Test test = null;
try {
test = (Test) c.newInstance();
} catch (Exception e) {
e.printStackTrace();
}
}
}

执行结果

无任何输出。

JDBC为什么用Class.forName

        JDBC 规范中明确要求 Driver(数据库驱动)类必须向 DriverManager 注册。

        以 MySQL 的驱动为例,我们看到 Driver 注册到 DriverManager 中的操作写在了静态代码块中,这就是为什么在写 JDBC 时使用 Class.forName() 的原因了。

public class Driver extends NonRegisteringDriver implements java.sql.Driver {
//
// Register ourselves with the DriverManager
//
static {
try {
java.sql.DriverManager.registerDriver(new Driver());
} catch (SQLException E) {
throw new RuntimeException("Can't register driver!");
}
}
/**
* Construct a new driver and register it with DriverManager
*
* @throws SQLException
* if a database error occurs.
*/
public Driver() throws SQLException {
// Required for Class.forName().newInstance()
}
}

无参与有参构造

 前边示例是无参构造,也可以有参构造:

package org.example.a;

import java.lang.reflect.Constructor;

class Test {
String name;
public Test(String name) {
this.name = name;
}

static {
System.out.println("静态代码块");
}
public static final String title = getTitle();

private static String getTitle(){
System.out.println("执行getTitle()");
return "TITLE";
}
}

public class Demo {
public static void main(String[] args) {
Class<?> c = null;
try {
c = Class.forName("org.example.a.Test");
}catch (Exception e){
e.printStackTrace();
}

Constructor<?> cons[] = c.getConstructors();

Test test = null;
try {
test = (Test) cons[0].newInstance("Tony");
} catch (Exception e) {
e.printStackTrace();
}
}
}

执行结果

静态代码块
执行getTitle()

取得类结构

取得接口/父类/构造方法

方法

说明

Class<?>[] c1 = c.getInterfaces();

取得实现的接口

Class<?> c2 = c.getSuperclass();

取得父类

Constructor<?> con[] = c.getConstructors();

取得构造函数

取得方法


方法



说明



Method getMethod(String name, Class<?>... parameterTypes)



通过方法名和参数列表获得方法



Method[] getMethods()



获得所有方法。

反映此 Class 对象所表示的类或接口(包括那些由该类或接口声明的以及从超类和超接口继承的那些的类或接口)的公共(public) member 方法。



Method getDeclaredMethod(String name, Class<?>... parameterTypes)






Method[] getDeclaredMethods()



反映此 Class 对象表示的类或接口声明的所有方法,包括公共、保护、默认(包)访问和私有方法,但不包括继承的方法。 


详细方法


方法



说明



int getModifiers()



取得本方法的访问修饰符。

取得后通过Modefier.toString(xxx)还原



String getName()



取得方法的名称



Class<?>[] getParameterTypes()



得到方法的全部参数类型



Class<?> getReturnType()



得到方法的返回值类型



CIass<?>[] getExceptionType()



得到一个方法的全部抛出异常


实例

package org.example.a;

import java.lang.reflect.Method;
import java.lang.reflect.Modifier;

interface ICategory {
public static final String CATEGORY_NAME = "Human";

public void sayCategory();

public String sayHello(String name, Integer age);
}

class Human implements ICategory {
private String name;
private Integer age;

public Human() {
}

public String getName() {
return name;
}

public void setName(String name) {
this.name = name;
}

public Integer getAge() {
return age;
}

public void setAge(Integer age) {
this.age = age;
}


@Override
public void sayCategory() {
System.out.println(CATEGORY_NAME);
}

@Override
public String sayHello(String name, Integer age) {
return "名字:" + name + "年龄:" + age;
}
}

public class Demo {
public static void main(String[] args) {
Class<?> c = null;
try {
c = Class.forName("org.example.a.Human");
} catch (Exception e) {
e.printStackTrace();
}

Method m[] = c.getMethods();
for (int i = 0; i < m.length; i++) {
Class<?> r = m[i].getReturnType();
Class<?> p[] = m[i].getParameterTypes();
int xx = m[i].getModifiers();
System.out.print(Modifier.toString(xx) + " ");
System.out.print(r.getName() + " ");
System.out.print(m[i].getName());
System.out.print("(");
for (int x = 0; x < p.length; x++) {
System.out.print(p[x].getName() + " " + "arg" + x);
if (x < p.length - 1) {
System.out.print(", ");
}
}
System.out.print(")");

Class<?> ex[] = m[i].getExceptionTypes();
if (ex.length > 0) {
System.out.print(" throws ");
for (int j = 0; j < ex.length; j++) {
System.out.print(ex[j].getName());
if (j < ex.length - 1) {
System.out.print(",");
}
}
}
System.out.println();
}
}
}

执行结果

public java.lang.String getName()
public void setName(java.lang.String arg0)
public java.lang.String sayHello(java.lang.String arg0, java.lang.Integer arg1)
public java.lang.Integer getAge()
public void setAge(java.lang.Integer arg0)
public void sayCategory()
public final void wait() throws java.lang.InterruptedException
public final void wait(long arg0, int arg1) throws java.lang.InterruptedException
public final native void wait(long arg0) throws java.lang.InterruptedException
public boolean equals(java.lang.Object arg0)
public java.lang.String toString()
public native int hashCode()
public final native java.lang.Class getClass()
public final native void notify()
public final native void notifyAll()

取得属性


方法



说明



Field getField(String name)



根据属性名称获取public属性



Field[] getFields()



获得该类所有的public属性



Field getDeclaredField(String name)



根据属性名称获取属性。private属性也可获取



Field[] getDeclaredFields()



获得该类所有的属性。private属性也可获取


详细方法


方法



说明



Object get(Object obj)



获得一个对象中属性的具体内容



void set(Object obj,Object value)



设置指定对象中属性的具体内容



int RetModifiers()



获得属性的修饰符



String getName()



获得此属性的名称



boolean isAccessible()



判断此属性是否可被外部访问



void setAccessible(boolean flag)



设置一个属性是否可被外部访问



static void setAccessibie(AccessibIeObject[] array,boolean flag)



将一组属性设置是否可被外部访问



String toString()



返回此Field类的信息


示例

package org.example.a;

import java.lang.reflect.Field;
import java.lang.reflect.Modifier;

interface ICategory {
public static final String CATEGORY_NAME = "Human";

public void sayCategory();

public String sayHello(String name, Integer age);
}

class Human implements ICategory {
private String name;
private Integer age;

public Human() {
}

public String getName() {
return name;
}

public void setName(String name) {
this.name = name;
}

public Integer getAge() {
return age;
}

public void setAge(Integer age) {
this.age = age;
}


@Override
public void sayCategory() {
System.out.println(CATEGORY_NAME);
}

@Override
public String sayHello(String name, Integer age) {
return "名字:" + name + "年龄:" + age;
}
}

public class Demo {
public static void main(String[] args) {
Class<?> c = null;
try {
c = Class.forName("org.example.a.Human");
} catch (Exception e) {
e.printStackTrace();
}

Field f[] = c.getDeclaredFields();
System.out.println("本类属性: ");
for (int i = 0; i < f.length; i++) {
Class<?> r = f[i].getType();
int mo = f[i].getModifiers();
String priv = Modifier.toString(mo);
System.out.print(priv + " ");
System.out.print(r.getName() + " ");
System.out.print(f[i].getName());
System.out.println(";");
}

System.out.println("-------------------------------------------");

Field f1[] = c.getDeclaredFields();
System.out.println("公共属性: ");
for (int i = 0; i < f1.length; i++) {
Class<?> r = f1[i].getType();
int mo = f1[i].getModifiers();
String priv = Modifier.toString(mo);
System.out.print(priv + "");
System.out.print(r.getName() + " ");
System.out.print(f1[i].getName());
System.out.println(";");
}
}
}

执行结果

本类属性: 
private java.lang.String name;
private java.lang.Integer age;
-------------------------------------------
公共属性:
privatejava.lang.String name;
privatejava.lang.Integer age;

取得注解

        Method, Constructor, Field这三个类对象都可以判断是否有特定的注解类,方法是:boolean isAnnotationPresent(Class<? extends Annotation> annotationClass)

示例

本处使用枚举类示例,普通类类似的。

package org.example.a;

import java.lang.reflect.Field;

enum Color{
@Deprecated
RED,

GREEN,
BLUE
}

public class Demo {
public static void main(String[] args) {
for (Color value : Color.values()) {
Field field = null;
try {
field = Color.class.getField(value.name());
} catch (NoSuchFieldException e) {
e.printStackTrace();
}
if (field.isAnnotationPresent(Deprecated.class)) {
System.out.println(value.name() + "上有注解类:" + Deprecated.class.getName());
}
}
}
}

执行结果

RED上有注解类:java.lang.Deprecated

应用

调用类的方法

package org.example.a;

import java.lang.reflect.Method;

class Human {
private String name;
private Integer age;

public Human() {
}

public String getName() {
return name;
}

public void setName(String name) {
this.name = name;
}

public Integer getAge() {
return age;
}

public void setAge(Integer age) {
this.age = age;
}


public void sayCategory() {
System.out.println("HUMAN");
}

public String sayHello(String name, Integer age) {
return "名字:" + name + " " + "年龄:" + age;
}
}


public class Demo {
public static void main(String[] args) {
Class<?> c = null;
try {
c = Class.forName("org.example.a.Human");
} catch (ClassNotFoundException e) {
e.printStackTrace();
}

try {
Method method = c.getMethod("sayHello", String.class, Integer.class);
System.out.println(method.invoke(c.newInstance(), "Tony", 20));
} catch (Exception e) {
e.printStackTrace();
}
}
}

执行结果

名字:Tony 年龄:20

修改属性

package org.example.a;

import java.lang.reflect.Field;

class Human {
private String name;
private Integer age;

public Human() {
}

public String getName() {
return name;
}

public void setName(String name) {
this.name = name;
}

public Integer getAge() {
return age;
}

public void setAge(Integer age) {
this.age = age;
}


public void sayCategory() {
System.out.println("HUMAN");
}

public String sayHello(String name, Integer age) {
return "名字:" + name + " " + "年龄:" + age;
}
}


public class Demo {
public static void main(String[] args) {
Class<?> c = null;
try {
c = Class.forName("org.example.a.Human");
} catch (ClassNotFoundException e) {
e.printStackTrace();
}

try {
Object obj = c.newInstance();
Field nameField = c.getDeclaredField("name");
nameField.setAccessible(true); //将name属性设置成可被外部访问
nameField.set(obj, "Tony"); //设置name属性内容
// System.out.println(nameField.get(obj));
System.out.println(((Human) obj).getName());
} catch (Exception e) {
e.printStackTrace();
}
}
}

执行结果

Tony

操作数组

方法

说明

Class<?> getComponentType()

取得一个数组的Class对象

详细方法

方法

说明

static Object get(Object array,int index)

根据下标取得数组内容

static Object newlnstance(Class<?> componentType, int length)

根据己有的数组类型开辟新的数组对象

static void set(Object array,int index,Object value)

修改指定位置的内容

示例:取得数组信息并修改数组内容

package org.example.a;

import java.lang.reflect.Array;

public class Demo {
public static void main(String[] args) {
int temp[] = {1, 2, 3};
Class<?> c = temp.getClass().getComponentType();
System.out.println("类型:" + c.getName());
System.out.println("长度:" + Array.getLength(temp));
System.out.println("第一个内容:" + Array.get(temp, 0));
Array.set(temp, 0, 6);
System.out.println("第一个内容:" + Array.get(temp, 0));
}
}

执行结果

类型:int
长度:3
第一个内容:1
第一个内容:6

示例:修改数组的大小

package org.example.a;

import java.lang.reflect.Array;


public class Demo {
public static void main(String[] args) {
int temp[] = {1, 2, 3};
int newTemp[] = (int[]) arrayInc(temp, 5);
print(newTemp);
System.out.println("---------------------------");
String str[] = {"abc", "def", "gh"};
String newStr[] = (String[]) arrayInc(str, 8);
print(newStr);
}

public static Object arrayInc(Object obj, int len) {
Class<?> c = obj.getClass();
Class<?> arr = c.getComponentType();
Object newO = Array.newInstance(arr, len);
int co = Array.getLength(obj);
System.arraycopy(obj, 0, newO, 0, co);
return newO;
}

public static void print(Object obj) {
Class<?> c = obj.getClass();
if (!c.isArray()) {
return;
}
Class<?> arr = c.getComponentType();
System.out.println(arr.getName() + "数组长度是:" + Array.getLength(obj));
for (int i = 0; i < Array.getLength(obj); i++) {
System.out.println(Array.get(obj, i) + "、");
}
}
}

执行结果

int数组长度是:5
1、
2、
3、
0、
0、
---------------------------
java.lang.String数组长度是:8
abc、
def、
gh、
null、
null、
null、
null、
null、

其他网址

《Java开发实战经典》=>  第15章 Java反射机制