首先了解什么是java反射机制
java 反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。
java反射(放射)机制:“程序运行时,允许改变程序结构或变量类型,这种语言称为动态语言”。从这个观点看,Perl,Python,Ruby是动态语言,C++,Java,C#不是动态语言。但是JAVA有着一个非常突出的动态相关机制:Reflection,用在Java身上指的是我们可以于运行时加载、探知、使用编译期间完全未知的classes。换句话说,Java程序可以加载一个运行时才得知名称的class,获悉其完整构造(但不包括methods定义),并生成其对象实体、或对其fields设值、或唤起其methods。
接下来用代码来具体了解java反射
1,首先,定义一个接口
java代码:
package com.my.model;
public interface IUserDao {
public void add();
}
2,定义一些类实现这个接口。以及创建一个user类
User.java
java代码:
package com.my.model;
public class User {
private int id;
private String name;
public int getId() {
return id;
}
public void setId(int id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String show(String str) {
return str+" [id=" + id + ", name=" + name + "]";
}
public static void say(String hello,String world) {
System.out.println(hello+",huhu,"+world);
}
}
UserADao.java
java代码:
package com.my.model;
public class UserADao implements IUserDao {
@Override
public void add() {
System.out.println("a");
}
}
UserBDao.java
package com.my.model;
public class UserBDao implements IUserDao {
@Override
public void add() {
System.out.println("b");
}
}
UserCDao.java
package com.my.model;
public class UserCDao implements IUserDao {
@Override
public void add() {
System.out.println("c");
}
}
3,测试类 Test.java
java代码:
package com.my.test;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import com.my.model.IUserDao;
import com.my.model.User;
import com.my.model.UserADao;
public class Test {
//通过new生成对象
@org.junit.Test
public void test01() {
User u = new User();
u.setId(1);
u.setName("abc");
System.out.println(u.show("User"));
}
//通过字符串生成对象
@org.junit.Test
public void test02() {
try {
String objStr = "com.my.model.User";
Class clz = Class.forName(objStr);
User u = (User)clz.newInstance();
u.setId(1);
u.setName("abc");
System.out.println(u.show("ref"));
} catch (ClassNotFoundException e) {
e.printStackTrace();
} catch (InstantiationException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
}
}
//通过字符串生成对象并且调用函数方法
@org.junit.Test
public void test03() {
try {
String objStr = "com.my.model.User";
Class<?> clz = Class.forName(objStr);
User u = (User)clz.newInstance();
u.setId(1);
u.setName("abc");
String mm = "show";//通过字符串调用函数方法
/**
* 第一个参数是函数的名称,第二个参数是...(表示有无限个参数)表示函数的参数
*/
Method method = clz.getMethod(mm, String.class);
/**
* 第一个参数是调用函数的对象,第二个参数是函数执行时的参数
* u.show("abc");
*/
String str = (String) method.invoke(u, "abc");
System.out.println(str);
/**
* 同样可以调用static的方法
*/
Method m2 = clz.getMethod("say", String.class,String.class);
/**
* 调用Static的方法是使用class来调用
*/
m2.invoke(clz, "ok","world");
} catch (ClassNotFoundException e) {
e.printStackTrace();
} catch (InstantiationException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (SecurityException e) {
e.printStackTrace();
} catch (NoSuchMethodException e) {
e.printStackTrace();
} catch (IllegalArgumentException e) {
e.printStackTrace();
} catch (InvocationTargetException e) {
e.printStackTrace();
}
}
@org.junit.Test
public void testDao01() {
IUserDao u1 = new UserADao();
u1.add();
}
@org.junit.Test
public void testDao02() {
try {
String dao = "com.my.model.UserCDao";
IUserDao ud = (IUserDao)Class.forName(dao).newInstance();
ud.add();
} catch (InstantiationException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (ClassNotFoundException e) {
e.printStackTrace();
}
}
}
最后,总结
反射的好处主要有两个,一个是工厂模式,还有就是spring的aop.
通过反射机制我们可以动态的通过改变配置文件(以后是XML文件)的方式来加载类、调用类方法,以及使用类属性。