class User
{
String name;
int id;
public User()
{
}
public User(String name, int id)
{
this.name = name;
this.id = id;
}
public String getName()
{
return name;
}
public void setName(String name)
{
this.name = name;
}
public int getId()
{
return id;
}
public void setId(int id)
{
this.id = id;
}
}
public Class ReflectTest
{
pulic static void main(String[] args)
{
try
{
//测试
}
catch (Exception e)
{
e.printStackTrace();
}
}
}
一个 ClassLoader 下,一种类,只会有一个类对象存在。通常一个 JVM 下,只会有一个 ClassLoader, 所以在一个 JVM 中,一种类,只会有一个类对象存在。以上三种方式取出来的类对象,都是一样的。
//使用反射的方式创建对象
String className = "mypackage.User";
//类对象
Class uClass = Class.forName(className);
//构造器
Constructor con = uClass.getConstructor();
//通过构造器实例化
User u = (User) con.newInstance();
Class.newInstance()
和 Constructor.newInstance()
Class 对象调用 newInstance() 方法只能用于不带参数的构造方法;
Constructor 对象调用 newINstance() 方法可以指定参数
Constructor con =
uClass.getConstructor(new Class[]{String.class, int.class});
User u = (User) con.newInstance(new Object[]{"haha", 123});
//也可以不传参数数组,直接写参数列表
Constructor con = uClass.getConstructor(String.class, int.class);
User u = (User) con.newInstance("haha", 123);
User u = new User();
//获取类User的name字段
Field f = u.getClass().getDeclaredField("name");
//修改这个字段的值
f.set(u, "haha");
//获取被修改后的值
System.out.println(f.get(u));
2.getDeclaredField()
和getField()
getField()
只能获取public的,包括从父类继承来的字段。
getDeclaredField()
可以获取本类所有的字段,包括private的,但是不能获取继承来的字段。
注:这里只能获取到private的字段,但并不能访问该private字段的值,除非加上f.setAccessible(true)
。
常规方法
User u = new User();
Method m = u.getClass().getMethod("setName", String.class);
//等同于
Method m =
u.getClass().getMethod("setName", new Class[]{String.class});
//对u对象,调用这个方法
m.invoke(u, "Jack");
//等同于
m.invoke(u, new Object[]{"Jack"});
getMethod()
和 getDeclaredMethod()
getMethod()
:获得类的public类型的方法
getDeclaredMethod()
:获得类的所有方法
gtMethod(String name, Class[] parameterTypes)
:获得类的特定方法,name 参数指定方法的名字,parameterTypes 参数指定方法的参数类型
获取方法上的注解
User u = new User();
Method m = u.getClass().getMethod("setName", String.class);
// 类上的注解
MyClassAnnotation myClassAnnotation = u.getAnnotation(MyClassAnnotation.class);
// 方法上的注解
MyMethodAnnotation myMethodAnnotation = m.getAnnotation(MyMethodAnnotation.class);
// 获取注解上的信息,比如 value
// MyMethodAnnotation.value();
反射可以访问私有属性,private 还有什么意义?
private 的作用是在面向对象的封装上 ,并不是为了解决安全问题。
作用: 在运行时构造一个类的对象;判断一个类所具有的成员变量和方法;调用一个对象的方法;生成动态代理
应用场景:反射最大的应用就是框架
示例:通过配置文件,切换业务类和业务方法, 不需要修改一行代码,也不需要重新编译,只需要修改配置文件,再运行即可
package reflection;
public class Service1
{
public void doService1()
{
System.out.println("业务方法1");
}
}
package reflection;
public class Service2
{
public void doService2()
{
System.out.println("业务方法2");
}
}
class=reflection.Service1
method=doService1
public class Test
{
public static void main(String[] args) throws Exception
{
//从config.txt中获取类名称和方法名称
File configFile = new File("config.txt");
Properties config= new Properties();
config.load(new FileInputStream(configFile));
String className = (String) config.get("class");
String methodName = (String) config.get("method");
//根据类名称获取类对象
Class clazz = Class.forName(className);
//根据方法名称,获取方法对象
Method m = clazz.getMethod(methodName);
//获取构造器
Constructor c = clazz.getConstructor();
//根据构造器,实例化出对象
Object service = c.newInstance();
//调用对象的指定方法
m.invoke(service);
}
}