一、java中生成对象的方式,本人所能想到的有以下几种(以 com.knight.Employer为例)
1、使用new方式,这每个javaer都会的
Employer e=new Employer();
2、使用克隆方式(clone),此方式要求Employer实现cloneable接口
Employer e2=(Employer)e1.clone();
3、序列化方式,此方式要求Employer实现Serializable接口
ObjectInputStream objIn=new ObjectInputStream(in);//in为InputStream实例
Employer e=(Employer)objIn.readObject();
4、反射
反射可以有很多变体,通常有以下几种
Class c=Class.forName("com.knight.Employer");
Employer e= (Employer)c.newInstance();
或者
ClassLoader loader=Huma.class.getClassLoader();
Class c=loader.loadClass("com.knight.Employer");
Employer e= (Employer)c.newInstance();
此两种方式都要求Employer有公有无参构造函数。
还有另外一种通过Constructor进行构造,以下详述
5、直接操作字节码,这种方式需要对class文件及虚拟机机制有深刻理解,不是一般的难
二、java反射api简介
以下与Huma类为例
package com.knight.test;
import java.io.ObjectInputStream;
import java.io.Serializable;
import java.lang.reflect.Constructor;
public class Huma {
public String name;
private int age;
private void log(){
System.out.println("log invoke");
}
public Huma() {
}
public Huma(String name,int age) {
this.name=name;
this.age=age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
}
java中反射机制主要使用以下几个类
Class,Constructor,Field,Method,AccessibleObject
1、Class的用法
Class可以获取关于类中所有方法,属性的信息
Field[] getFields()//获取Huma中声明的公有属性,包括从父类继承的
Field[] getDeclaredFields()//获取Huma中声明所有属性(包括私有属性),但不包含任何父类继承的属性
注意有无declared的区别,以下不在罗嗦
Field getField(String name)//获取名称为name的属性
以此类似,Class可以获取其对于的Constructor,Method信息
再举一个方法
Method getDeclaredMethod(String name, Class<?>... parameterTypes);此方法获取成员函数名字为name,函数的参数类型为Class<?>... parameterTypes的函数,其中parameterTypes为null或者为空数组时,表示该函数没有形参
public Object newInstance() //生成该类的一个对象,需有无参构造函数
public static Class forName(String className)//生成className指定的类的Class实例
2、AccessibleObject
AccessibleObject是一个接口,Constructor,Field,Method均实现了该接口
public void setAccessible(boolean flag);该方法设置可访问性,在类的成员为私有的情况下
设置flat为true将可以通过反射修改成员的值,否则会发生异常
3、Field类
public Object get(Object o);获取该Field实例对应对象o的Field的值
public void set(Object obj, Object value) ;设置该Field对应的对象obj的Field属性的值为value
3、Method类
Class<?>[] getParameterTypes() ;返回该Method的形参类型,按声明顺序
Class<?> getReturnType() ;返回该Method的返回类型
Class<?>[] getExceptionTypes() ;返回该Method声明抛出的异常
Object invoke(Object obj, Object... args) ;执行该Method,obj是该Method所属对象的一个实例
//args是一个对象数组,表示该Method参数数组实参
其中args为null或者为空数组时,表示该函数没形参,当obj为null时,表示该函数为静态方法
4、Constructor类
Class<T> getDeclaringClass();返回该Constructor所表示的Class对象
Class<?>[] getParameterTypes() ;返回该Constructor表示的形参类型数组
T newInstance(Object... initargs) ;生成该Constructor表示的对象的一个实例,initargs为形参数组
三、例子
// 生成Huma实例(带参数)
Constructor c=Huma.class.getDeclaredConstructor(new Class[]{String.class,int.class});
Huma huma=c.newInstance(new Object[]{"zhang san",18});
//获取Field,并改变其值
Field nameField=Huma.class.getField("zhangsan");
nameField.setsetAccessible(true);//注意name属性私有,所以在这里调用,参数为true
nameField.set("lisi");//此时name已经变为lisi了,上一句不调用的的话,此句将抛出异常
//获取方法并调用
Method methodSetName=Huma.class.getMethod("setName",new Class[]{String.class});
methodSetName.invoke(huma,new Object[]{"wangwu"});//相当于 huma.setName("wangwu");
最后需要注意的地方就是反射生成的类,修饰符应该是public的,否则,在其他地方调用时可能会出现问题
,例如 在一个 com.abc.Test的main方法中调用 Class.forName("com.knight.Huma"),并且Huma的修饰符不是public时,将抛出异常