反射在java中使用十分常见,可以说学java进阶不能不学反射,而反射的一大特点就是可以操作注解,现在很多框架都使用到注解进行简化开发,如果想能够看懂注解的实现原理,建议学习并能熟练运用反射机制。
1 首先了解反射
JAVA机制反射是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。
一个简单的反射程序
实体类
public class User {
private int id;
private String name;
public User() {
super();
// TODO Auto-generated constructor stub
}
public User(int id, String name) {
super();
this.id = id;
this.name = 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;
}
@Override
public String toString() {
return "User [id=" + id + ", name=" + name + "]";
}
}
测试程序
public static void main(String[] args) throws Exception {
//获得反射对象
//一个类的反射对象指向的是同一个反射对象,类似你照镜子,镜子显示的始终是你
Class clazz=Class.forName("com.atlin.pojo.User");//此种方法可以使该对象的反射开启
//创建反射实例,也就是创建对象,通过反射创建的对象通常比直接创建对象慢
User user=(User) clazz.newInstance();
Method method=clazz.getDeclaredMethod("setId", int.class);
method.invoke(user, 18);
Method methods=clazz.getDeclaredMethod("getId", null);
System.out.println((int)methods.invoke(user, null)==user.getId());
//相等
//修改属性值,可以对私有属性操作,一旦开启setAccessible
Field f=clazz.getDeclaredField("id");
f.setAccessible(true);
f.set(user, 5);
System.out.println(f.get(user));
//通过构造函数,创建对象
Constructor con=clazz.getDeclaredConstructor(int.class,String.class);
User ue=(User) con.newInstance(5,"LIN");
System.out.println(ue);
}
以上是一些对实体类的一些简单的反射操作,以下是利用反射操作注解的操作
实体类
@tb_num(tb="ss")
public class User {
@table(value="1",value2 = "123")
private int id;
@table(value="2",value2 = "123")
private String name;
public User() {
super();
// TODO Auto-generated constructor stub
}
public User(int id, String name) {
super();
this.id = id;
this.name = 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;
}
@Override
public String toString() {
return "User [id=" + id + ", name=" + name + "]";
}
}
此注解用来操作属性
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface table {
String value();
String value2();
}
此注解用来操作类名
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface tb_num {
public String tb();
}
操作方法类似,就不讨论了
反射操作注解操作
public static void main(String[] args) throws Exception {
Class clazz=Class.forName("com.atlin.pojo.User");
Annotation[] annotation=clazz.getAnnotations();
System.out.println(annotation.length);
for(Annotation a:annotation) {
System.out.println(a);
}
tb_num tb=(tb_num) clazz.getAnnotation(tb_num.class);
System.out.println(tb.tb());
Field f=clazz.getDeclaredField("id");
table ts=f.getAnnotation(table.class);
System.out.println(ts.value()+" "+ts.value2());
Field f1=clazz.getDeclaredField("name");
table ts1=f.getAnnotation(table.class);
System.out.println(ts1.value()+" "+ts1.value2());
}
仅供学习,如有问题可以提出,持续更新。