说起反射,总是让初学者觉得特别高端,无法理解,甚至有点与java的编程习惯有点背道而驰,或者有些人谈到这个词就会说一些JVM、动态加载之类的话,结果就是讲的人装X半天,懂的人知道他在装X,不懂的人还是一头雾水。那么有没有一种可以让初学者一听就懂的方式来描述反射呢?
其实反射从表面上来讲并没有什么了不起,只是把类、方法、属性都封装成了对象,然后就可以用面向对象的方式对其进行操作了。
这个描述可能还是不够简明,我喜欢这样描述反射:反射就是将字符串当代码来执行的一种java里的妥协方式。
这个描述乍一看好像也不好理解,但配合下面的例子就一目了然了。
定义两个成员变量:Integer a = 3; String b = "12";
定义一个方法:public void eval(String code){...};
为了简单,我们规定这个方法接收一个字符串参数,字符串是一个对象调用有返回值的无参方法,然后把字符串中的代码执行结果打印出来,比如我传入"b.length();",控制台会输出:2;再比如我传入"a.intValue();",控制台输出:3。
(参考实现思路:(不是重点,建议先往下读)我们知道字符串参数会有一个点,一对括号,这是我们规定的,而点前面是一个对象的变量名称,点和括号之间是方法名,我们可以对字符串进行解析,拿到我们需要的变量名和方法名,然后用反射将方法执行出来,将值输出到控制台。)
这种操作用java来实现可能比较繁琐,但是熟悉JS的朋友应该一眼就看出了我将方法命名为eval的用意。
var a = 3;
var b = {bFunc: function () {alert('bFunc is RUNNING')}};
eval('console.log(a)');//会在控制台输出:3
c = b.bFunc()';
eval(c);//会在控制台输出:bFunc is RUNNING
这段JS代码与上面我们要完成的java是异曲同工的,这就是java里的反射所要达成的目标。
很多解释型语言都有类似于JS的eval的这种方法,因为它们本身就是对字符串解析执行的,代码是字符串,内存里也有字符串,并没有本质区别,当然解释起来很方便,代码也变得非常灵活,但是java作为一门解释型语言为什么搞这么复杂呢?因为java解释的不是字符串,而是编译好的class文件,是字节码,但是人们又想像JS那样灵活,就只能用java的思想,将类、方法、属性都封装成对象,所以我说这是一种妥协的结果。
可能有的朋友会觉得,这并不是反射的全部,反射还能打破封装,调用对象的私有方法和访问私有属性。确实如此,不过在打破封装之前,你必须先知道你想要打破的这个方法或属性的名称,而不管你用什么办法获取这个名称,你会发现这个名称是一个字符串,最终还要回到上面所说的“把字符串当代码”。