什么是反射
反射,就是运行中的程序,自身可以动态的访问自身的一些属性和方法.
JAVA中为什么要用反射
说到这个问题,我们必须先了解一个概念:动态语言和静态语言.
动态语言:动态语言是在运行时确定数据类型的语言。变量使用之前不需要类型声明,通常变量的类型是被赋值的那个值的类型。
例如PHP/ASP/Ruby/Python/Perl/ABAP/SQL/JavaScript/Unix Shell等等。
比如js中的一个代码片段:
var a = "Hello Word"; a = new Date(); alert(a);
可见,定义了一个a,被赋予不同类型的值.
静态语言:静态语言是在编译时变量的数据类型即可确定的语言,多数静态类型语言要求在使用变量之前必须声明数据类型。
例如:C++、Java、Delphi、C#等。
优缺点:静态语言在编译的时候就可以发现错误,编译如果错误,则不能运行,动态语言一边编译,一边执行,所以在运行时才能看出错误.
综上所述,Java如果想在程序运行中获取当前程序中的方法或者属性,则需要反射来完成.
Java中怎么使用反射
先上图来了解一下Java中的加载类的简要图,可能画的不是太完整,大家凑活着看
使用反射的步骤
产生字节码对象,有三种方式
(1)Class clazz = Class.forName("目标类的完整路径"),比如Class.forName("com.mysql.jdbc.Driver")
这种方式一般用于要获取用户自定义的字节码文件.
(2)Class clazz = String.class;类名.class;
知道目标类,可以直接用类名.class获取
(3)String str = "a";
Class clazz = str.getClass(); ps:getClass()方法由Object提供
第三中感觉实用中最多的,用类的实例,或者对象.getClass();获取字节码对象
用字节码对象获取类中的信息
package com.tangbaobao.reflect;
/**
* @author 唐学俊
* @version 创建时间:2017年11月3日 下午3:08:08
*
*/
public class Test1 {
public static void main(String[] args) {
Class<?> clzz = null;
try {
clzz = Class.forName("java.lang.String");
} catch (ClassNotFoundException e) {
e.printStackTrace();
}
System.out.println("全类名:"+clzz.getName());
System.out.println("只获取类名:"+clzz.getSimpleName());
Class<?>[] interfaces = clzz.getInterfaces();//获取字节码对象的实现的接口
for (Class<?> class1 : interfaces) {
System.out.println(class1);
}
System.out.println("父类对象的全类名"+clzz.getSuperclass().getName());
System.out.println("获取当前类加载器:"+Test1.class.getClassLoader());
System.out.println("获得包名:"+clzz.getPackageName());
System.out.println("获得修饰符:"+clzz.getModifiers());
}
}
还有很多API,就不一一列举了,如果有想深入了解的大家可以参照官方的API文档
获取运行中类(当前字节码对象的属性)
java为我们提供了Field类
假设获取Class clazz = User.class;
获取某一个知道名字的公共的属性:Field field = clazz.getField("username")
获取clazz中所有公有的属性 Field[] fields = clazz.getFields()
field 方法,我用一个动态生成sql语句的demo来展现
package com.tangbaobao.reflect;
import java.lang.reflect.Field;
import java.lang.reflect.Modifier;
import com.tangbaobao.domain.User;
/**
* @author 唐学俊
* @version 创建时间:2017年11月3日 下午4:46:40
*
*/
public class Test2 {
public static void main(String[] args) {
User user = new User();
generateSql(user);
}
public static void generateSql(Object obj) {
Class<?> clazz = obj.getClass();
//insert into 表名 values(.....)
//String sql = "insert into "+clazz.getSimpleName()+"(";
//用StringBuffer
StringBuffer sql = new StringBuffer("insert into "+clazz.getSimpleName()+"(");
StringBuffer sql_v = new StringBuffer(" values(");
//String sql_v =")";
//获得属性名
Field[] fields = clazz.getDeclaredFields();
for(int i=0; i<fields.length; i++) {
if(i!=fields.length-1) {
sql.append(fields[i].getName()+",") ;
sql_v.append("?,");
}else {
sql.append( fields[i].getName()+")");
sql_v.append("?)");
}
}
System.out.println(sql.toString()+sql_v.toString());
}
}