您的位置:首页 > 编程语言 > Java开发

反射

2015-07-01 19:07 579 查看
摘要: 一、什么是反射机制
简单的来说,反射机制指的是程序在运行时能够获取自身的信息。在java中,只要给定类的名字,
那么就可以通过反射机制来获得类的所有信息。

一、什么是反射机制
简单的来说,反射机制指的是程序在运行时能够获取自身的信息。在java中,只要给定类的名字, 那么就可以通过反射机制来获得类的所有信息。
二、哪里用到反射机制
有些时候,我们用过一些知识,但是并不知道它的专业术语是什么,在刚刚学jdbc时用过一行代码,Class.forName("com.mysql.jdbc.Driver.class").newInstance();但是那时候只知道那行代码是生成驱动对象实例,并不知道它的具体含义。听了反射机制这节课后,才知道,原来这就是反射,现在很多开框架都用到反射机制,hibernate、struts都是用反射机制实现的.
三、反射机制的优点与缺点

为什么要用反射机制?直接创建对象不就可以了吗,这就涉及到了动态与静态的概念.
静态编译:就是编译器在编译可执行文件的时候,将可执行文件需要调用的对应动态链接库(.so或.lib)中的部分提取出来,链接到可执行文件中去,使可执行文件在运行的时候不依赖于动态链接库。
动态编译:动态编译的可执行文件需要附带一个的动态链接库。在执行时,需要调用其对应动态链接库中的命令。所以其优点一方面是缩小了执行文件本身的体积,另一方面是加快了编译速度,节省了系统资源。动态编译最大限度发挥了java的灵活性,体现了多态的应用,降低类之间的藕和性。
一句话,反射机制的优点就是可以实现动态创建对象和编译,体现出很大的灵活性,特别是在J2EE的开发中它的灵活性就表现的十分明显。比如,一个大型的软件,不可能一次就把把它设计的很完美,当这个程序编译后,发布了,当发现需要更新某些功能时,我们不可能要用户把以前的卸载,再重新安装新的版本,假如这样的话,这个软件肯定是没有多少人用的。采用静态的话,需要把整个程序重新编译一次才可以实现功能的更新,而采用反射机制的话,它就可以不用卸载,只需要在运行时才动态的创建和编译,就可以实现该功能。
&nbs
3ff0
p; 它的缺点是对性能有影响。使用反射基本上是一种解释操作,我们可以告诉JVM,我们希望做什么并且它满足我们的要求。这类操作总是慢于只直接执行相同的操作。
四、利用反射机制能获得什么信息
一句话,类中有什么信息,它就可以获得什么信息,不过前提是得知道类的名字,要不就没有后文了
首先得根据传入的类的全名来创建Class对象。
Class c=Class.forName("className");
注明:className必须为全名,也就是得包含包名,比如,cn.netjava.pojo.UserInfo;

Object obj=c.newInstance();//创建对象的实例
OK,有了对象就什么都好办了,想要什么信息就有什么信息了。
获得构造函数的方法
Constructor getConstructor(Class[] params)//根据指定参数获得public构造器
Constructor[] getConstructors()//获得public的所有构造器
Constructor getDeclaredConstructor(Class[] params)//根据指定参数获得public和非public的构造器
Constructor[] getDeclaredConstructors()//获得public的所有构造器
获得类方法的方法
Method getMethod(String name, Class[] params),根据方法名,参数类型获得方法
Method[] getMethods()//获得所有的public方法
Method getDeclaredMethod(String name, Class[] params)//根据方法名和参数类型,获得public和非public的方法
Method[] getDeclaredMethods()//获得所以的public和非public方法
获得类中属性的方法
Field getField(String name)//根据变量名得到相应的public变量
Field[] getFields()//获得类中所以public的方法
Field getDeclaredField(String name)//根据方法名获得public和非public变量
Field[] getDeclaredFields()//获得类中所有的public和非public方法
常用的就这些,知道这些,其他的都好办……
五、用反射机制能干什么事
刚开始在使用jdbc时侯,在编写访问数据库时写到想吐,有八个表,每个表都有增删改查中操作那时候还不知道有反射机制这个概念,所以就对不同的表创建不同的dao类,这样不仅开发速率地,而且代码
冗余的厉害,最要命的是看着差不多的,然后直接复制修改,由于容易犯各种低级的错误(大小写啊,多一个或少一个字母啊……),一个错误就可以让你找半天。
有了java反射机制,什么都好办了,只需要写一个dao类,四个方法,增删改查,传入不同的对象,就OK啦,无需为每一个表都创建dao类,反射机制会自动帮我们完成剩下的事情,这就是它的好处。说白了,反射机制就是专门帮我们做那些重复的有规则的事情,所以现在很多的自动生成代码的软件就是运用反射机制来完成的,只要你按照规则输入相关的参数,所以低级的程序员慢慢的就被抹杀了,为什么?因为代码都不用写了,随便一个人都会开发,还要程序员干什么啊?所以我们只有一条出路,那就是努力努力再努力,成为高级程序员,专门开发傻瓜软件,让其他程序员到 一边凉快去.
六、简单的例子
获取类的三种方式:
package com.reflect;

public class Demo {

}
class hello{
public static void main(String[] args) {
Demo demo=new Demo();
System.out.println("类或对象的包名:"+demo.getClass().getName());
System.out.println("类名是:"+demo.getClass().getSimpleName());
System.out.println("类的加载器:"+demo.getClass().getClassLoader());
}
}


【运行结果:】

类或对象的包名:com.reflect.Demo
类名是:Demo
类的加载器:sun.misc.Launcher$AppClassLoader@1a46e30


利用反射获取实现类:

package com.reflect;

public interface USA {
public static String language="English";

public void speakEnglish();
}


package com.reflect;

public interface China {
public static final String NAME="小明";

public static int age=26;

public void sayChina();

public void sayHello(String name,int age);
}

class Person implements China,USA{

private int age;

public int getAge() {
return age;
}

public void setAge(int age) {
this.age = age;
}

public Person() {
}

public Person(int age){
this.age=age;
}

@Override
public void sayChina() {
System.out.println("hello ,china");
}

@Override
public void sayHello(String name, int age) {
System.out.println(name +" "+ age );
}

@Override
public void speakEnglish() {
System.out.println("speak English");
}

}

class haha{
public static void main(String[] args){
Class<?> demo=null;

try {
demo=Class.forName("com.reflect.Person");
} catch (ClassNotFoundException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}

//保存所有的接口
Class<?> intes[]=demo.getInterfaces();
for (int i = 0; i < intes.length; i++) {
System.out.println("实现的接口   "+intes[i].getName());
}
}
}


【运行结果:】

实现的接口   com.reflect.China
实现的接口   com.reflect.USA


所有类的对象都是Class对象的实例

package com.reflect;

class Demo2 {

}

/**
* 所有类的对象都是Class对象的实例
*
*/
class xixi{

public static void main(String[] args) {
Class<?> c1=null;
Class<?> c2=null;
Class<?> c3=null;

try {
c1=Class.forName("com.reflect.Demo2");
} catch (ClassNotFoundException e) {
e.printStackTrace();
}

c2=new Demo2().getClass();

c3=Demo2.class;

System.out.println(c1==c2);

System.out.println(c2==c3);

System.out.println(c1==c3);

System.out.println(" c1的类名:"+c1.getName()+" c2的类名:"+c2.getName()+" c3的类名:"+c3.getName());
}

}


【运行结果:】

true
true
true
c1的类名:com.reflect.Demo2 c2的类名:com.reflect.Demo2 c3的类名:com.reflect.Demo2


class类的实例化比较

package com.reflect;

import org.apache.commons.lang.builder.ToStringBuilder;
import org.apache.commons.lang.builder.ToStringStyle;

/**
* Class实例化其他类的对象
*
*/
public class Demo3 {
private String name;
private int age;

public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}

@Override
public String toString() {
//return "Demo3 [name=" + name + ", age=" + age + "]";
return ToStringBuilder.reflectionToString(this, ToStringStyle.SHORT_PREFIX_STYLE);
}

}

class hehe{
public static void main(String[] args){
Class<?> demo=null;

try {
demo=Class.forName("com.reflect.Demo3");
} catch (ClassNotFoundException e) {
e.printStackTrace();
}

Demo3 d3=null;

try {
d3=(Demo3) demo.newInstance();
} catch (InstantiationException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
}

d3.setName("小明");
d3.setAge(23);

System.out.println(d3);
}
}


【运行结果:】

Demo3[name=小明,age=23]
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签:  java reflect