反射机制

反射机制可通过在运行时加载类名而获取类,并对其进行操作。工厂模式,动态代理中较常用到。

在实际场景中:由于有好多类具有共同的接口样式,而他们又用的不是很频繁,如果在服务器中保有这些类会占用资源空间,如果通过接口指定的方式去加载,用完之后就销毁掉,可节省资源空间,且实现接口编程,扩展性好,代码量也少。打个比方:哺乳动物Animal为接口,里面包括获取乳头数,获取腿,获取头,获取尾巴等,像这种都具有一定共性,实现类为老虎,狮子,兔子,羊等等等。

1.获取className,常用的形式将需要反射的类放在属性文件*.properties或数据库字段中,通过key获取类名及路径。

 private static final String CLASS_NAME_PATH = "pattern/classname.properties";

     /**
* 获取类名称
* @return
*/
private static String getClassName(String name) {
Properties pro = new Properties();
try {
String url = ProxyFactory.class.getClassLoader().getResource("").getPath() + CLASS_NAME_PATH;
pro.load(new FileInputStream(url));
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
return (String)pro.get(name);
}

getClassName

 # animal
lion = pattern.creation.factory.animal.Lion
sheep = pattern.creation.factory.animal.Sheep
rabit = pattern.creation.factory.animal.Rabit
# plant
apple = pattern.creation.factory.plant.Apple
pear = pattern.creation.factory.plant.Pear
plum = pattern.creation.factory.plant.Plum

classPath.properties

>>> String strClass = getClassName("rabit");//获取兔子类

2.获取类,创建类实例

Class<?> clazz = Class.forName(strClass);  //获取类

Object obj = clazz.newInstance();            //构建实例对象

Class<?> superClazz = clazz.getSuperclass(); //获取父类
Class<?>[] interfaceArray = clazz.getInterfaces();//获取接口

     /**
* 根据classname获取类名
* @param strName
* @return
* @throws ClassNotFoundException
* @throws InstantiationException
* @throws IllegalAccessException
*/
@SuppressWarnings("unchecked")
public <T> T getClassByName(String strName) throws ClassNotFoundException, InstantiationException, IllegalAccessException {
String strClass = getClassName(strName);
Class<?> clazz = Class.forName(strClass);
Class<?> superClazz = clazz.getSuperclass();
Class<?>[] interfaceArray = clazz.getInterfaces();
System.out.println("className--: " + clazz.getName());//当前类名
System.out.println("superClassName--: " + superClazz.getName());//父类
for(Class<?> interfaces: interfaceArray){
System.out.println("interface--: " + interfaces.getName());//接口
}
return (T)clazz.newInstance();
}

getClass

>>> AnimalFactory animal = reflectFactory.getClassByName("rabit");

运行结果:

className--: pattern.creation.factory.animal.Rabit
superClassName--: pattern.creation.factory.animal.Animals
interface--: pattern.creation.factory.animal.AnimalFactory

3. 获取类属性及信息

Field[] fieldArray = clazz.getDeclaredFields(); // 显示所有的属性
Field[] publicFieldArray = clazz.getFields(); //显示public属性的元素

String name = field.getName();    //获取方法名

String modify = Modifier.toString(field.getModifiers());   // 修饰词
String classType = field.getType().getSimpleName();    // 类型
Annotation[] annoArray = field.getAnnotations();        // 注解

field.setAccessible(true);        //获取value要赋予权限

Object obj = clazz.newInstance();    //对象实例
Object value = field.get(obj);// 私有属性需要获取权限才可以

field.set(obj, value + "-----changed..");// 修改私有属性的值

     /**
* 通过反射机制获取类 的属性名及其内容
* @param strName
* @throws ClassNotFoundException
* @throws InstantiationException
* @throws IllegalAccessException
*/
public void getClassFieldByName(String strName) throws ClassNotFoundException, InstantiationException, IllegalAccessException {
String strClass = getClassName(strName);
Class<?> clazz = Class.forName(strClass);
Object obj = clazz.newInstance();
Field[] fieldArray = clazz.getDeclaredFields(); // 显示所有的属性
Field[] publicFieldArray = clazz.getFields(); //显示public属性的元素
Field[] newFieldArray = (Field[])ArrayUtils.addAll(fieldArray, publicFieldArray);
for(Field field : newFieldArray){
String name = field.getName();
String modify = Modifier.toString(field.getModifiers());
String classType = field.getType().getSimpleName();
Annotation[] annoArray = field.getAnnotations();
String strAnno = "";
for(Annotation anno : annoArray){
strAnno += "@"+anno.annotationType().getSimpleName();
}
field.setAccessible(true);
Object value = field.get(obj);// 私有属性需要获取权限才可以
System.out.println("propertyName: " + strAnno + " " + modify+ " " + classType + " " + name + "=" + value);
if(!StringUtils.contains(modify, "final")){
field.set(obj, value + "-----changed..");// 修改私有属性的值
System.out.println("propertyName: " + strAnno + " " + modify+ " " + classType + " " + name + "=" + value);
}
} Field[] fieldsArray = clazz.getSuperclass().getDeclaredFields();//父类的属性名
for(Field field : fieldsArray){
String name = field.getName();
String modify = Modifier.toString(field.getModifiers());
String classType = field.getType().getSimpleName();
field.setAccessible(true);
Object value = field.get(obj);// 私有属性需要获取权限才可以
System.out.println("superPropertyName: " + modify+ " " + classType + " " + name + "=" + value);
}
}

getClassFieldByName

>>> reflectFactory.getClassFieldByName("rabit");

运行结果:

propertyName: private String rabitName=I am rabit
propertyName: private String rabitName=I am rabit
propertyName: private String header=
propertyName: private String header=
propertyName: private static final int LEGS=4
propertyName: @Resource public String tails=1tails
propertyName: @Resource public String tails=1tails
propertyName: @Resource public String tails=1tails-----changed..
propertyName: @Resource public String tails=1tails-----changed..
superPropertyName: private static final long serialVersionUID=8482086762730882629
superPropertyName: private int slegs=0
superPropertyName: private int shands=0
superPropertyName: private int sheader=0

4. 获取方法,及方法调用

String modifier = Modifier.toString(method.getModifiers()); //修饰词
String returnType = method.getReturnType().getSimpleName();//返回类型
String name = method.getName();//方法名
Parameter[] parameters = method.getParameters();//参数列表
Annotation[] annoArray = method.getAnnotations();//注解列表

Method method = clazz.getMethod(methodName, classes); //根据方法名,获取方法

method.invoke(obj, orgs); //方法调用  注意invoke的第一个参数为方法对应的类的实例。

UserDTO dto = new UserDTO();
dto.setId("");
System.out.println(dto.getClass().getMethod("getId").invoke(dto));
     /**
* 获取方法并调用方法
* @param strName
* @param methodName
* @param orgs
* @param classes
* @throws ClassNotFoundException
* @throws InstantiationException
* @throws IllegalAccessException
* @throws NoSuchMethodException
* @throws SecurityException
* @throws IllegalArgumentException
* @throws InvocationTargetException
*/
public void getRunMethodByMethodName(String strName, String methodName, Object[] orgs, Class<?>[] classes) throws ClassNotFoundException, InstantiationException, IllegalAccessException, NoSuchMethodException, SecurityException, IllegalArgumentException, InvocationTargetException {
String strClass = getClassName(strName);
Class<?> clazz = Class.forName(strClass);
Class<?> superClazz = clazz.getSuperclass();
Object obj = clazz.newInstance();
Method[] methodArray = clazz.getDeclaredMethods();
Method[] supeMethodArray = superClazz.getDeclaredMethods();
Method[] sumMethodArray = (Method[])ArrayUtils.addAll(methodArray, supeMethodArray);
for(Method method : sumMethodArray){
String modifier = Modifier.toString(method.getModifiers());
String returnType = method.getReturnType().getSimpleName();
String name = method.getName();
Parameter[] parameters = method.getParameters();
Annotation[] annoArray = method.getAnnotations();
String strAnno = "";
for(Annotation anno : annoArray){
strAnno += "@"+anno.annotationType().getSimpleName();
}
String strPara = "";
for(Parameter para : parameters){
strPara+=para.getType().getSimpleName() + " " + para.getName() + ",";
}
System.out.println("method:----"+strAnno + " " + modifier + " " + returnType + " " + name + "(" + strPara +")");
}
Method method = clazz.getMethod(methodName, classes);
System.out.println("header: " + method.invoke(obj, orgs));
}

getRunMethodByMethodName

>>> Class<?>[] classArray = {String.class,int.class,String.class};
>>> Object[] paramsArray = {"兔子只有",1,"头"};
>>> reflectFactory.getRunMethodByMethodName("rabit", "getHeader",paramsArray, classArray);

运行结果:

method:---- public String getHeader()
method:---- public String getHeader(String arg0,int arg1,String arg2,)
method:---- public int getLegs()
method:---- public int getLegs()

header:  兔子只有$$$$1@@@@@@头

5. 构造体方法

clazz.getConstructors()  // 获取所有的构造体

----------------------------------------------------------------------------------------------------------

以上,为个人总结,如有写的不当之处,还望指正。

-----------DennyZhao

反射机制(java)的更多相关文章

  1. 反射机制(Java)

    反射机制 今天闲来无事,对反射机制http://www.cnblogs.com/jqyp/archive/2012/03/29/2423112.html阅读一番,整理了下这方面的知识以及自己的一些心得 ...

  2. Java:描述反射机制的作用?举几个反射的应用?

    比较全的解释了:JAVA反射机制 JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法:对于任意一个对象,都能够调用它的任意一个方法:这种动态获取的信息以及动态调用对象的方 ...

  3. Java 中的反射机制

    JAVA反射机制 JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法:对于任意一个对象,都能够调用它的任意一个方法:这种动态获取的信息以及动态调用对象的方法的功能称为ja ...

  4. java反射机制详解 及 Method.invoke解释

    JAVA反射机制 JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法:对于任意一个对象,都能够调用它的任意一个方法:这种动态获取的信息以及动态调用对象的方法的功能称为ja ...

  5. java反射机制 + Method.invoke解释 getMethod + 反射理解

    功能: 通过读取另一个Dll去创建一个控件(Form,Button,TextBox,DataGridView),然后对当中一些属性进行检查. 创建控件的大致流程是,Assembly->Modul ...

  6. 5. Java反射机制

    Java反射机制   问题: 在运行时,对一个JAVA类,能否知道属性和方法:能否调用它的任意方法? 答案是可以的,JAVA提供一种反射机制可以实现. 目录 什么是JAVA的反射机制 JDK中提供的R ...

  7. java反射机制(2)

    首先,我们在开始前提出一个问题: 1.在运行时,对于一个java类,能否知道属性和方法:能否去调用它的任意方法? 答案是肯定的. 本节所有目录如下: 什么是JAVA的反射机制 JDK中提供的Refle ...

  8. Java反射机制剖析(一)-定义和API

    1.     什么是Java反射机制 Java的反射机制是在程序运行时,能够完全知道任何一个类,及其它的属性和方法,并且能够任意调用一个对象的属性和方法.这种运行时的动态获取就是Java的反射机制.其 ...

  9. Java反射机制详解(1) -反射定义

    首先,我们在开始前提出一个问题: 1.在运行时,对于一个java类,能否知道属性和方法:能否去调用它的任意方法? 答案是肯定的. 本节所有目录如下: 什么是JAVA的反射机制 JDK中提供的Refle ...

  10. java反射机制(1)

      百度百科: java 反射机制:JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法:对于任意一个对象,都能够调用它的任意方法和属性:这种动态获取信息以及动态调用对象方 ...

随机推荐

  1. 学习笔记之C / C++

    面试总结之C/C++ - 浩然119 - 博客园 https://www.cnblogs.com/pegasus923/p/5558919.html 学习笔记之C++ How to Program(p ...

  2. 学习笔记之The Intelligent Investor, Rev. Ed

    The Intelligent Investor, Rev. Ed https://www.safaribooksonline.com/library/view/the-intelligent-inv ...

  3. 廖雪峰Java4反射与泛型-1反射-1Class类

    1.Class类与反射定义 Class类本身是一种数据类型(Type),class/interface的数据类型是Class,JVM为每个加载的class创建了唯一的Class实例. Class实例包 ...

  4. [UE4]虚幻4蓝图使用小技巧

    不得不说,虚幻的蓝图系统还是非常方便强大的,大大的提高了开发效率.蓝图是一个很成熟的系统,也就有很多隐藏的小技巧,这些技巧谈不上多高深,却可以使人们在使用蓝图时更加得心应手,更加喜爱这个“可视化编程“ ...

  5. c#day04从控制台获取一个字符

    class EnumStruct { //定义QQ的状态枚举 Qme QnLine OffLine Leave Busy //提示用户现在一个状态 然后接收 ,并将用户输入转换成枚举类 enum Qs ...

  6. jQuery动态创建DOM节点

    var SKU=$("#SKU"); // 通过ID查找到指定表格 var oTable=SKU[0]; // 获取第一个表格 // 获取正文tbody元素 var tBodies ...

  7. HibernateUtil工具类的使用

    为了简化代码的重复性,使用HibernateUtil工具类对Hibernate有关的代码进行整合 主要实现有,getSessionFactory(),getSession(),closeSession ...

  8. sublime 打开import require 模块文件的url 或路径的插件

    结果一番周折,终于发现sublime提供的一个插件(open url)可以实现打开import file 路径的文件,比如import demo from “../path” 的文件在新窗口或者新标签 ...

  9. Hadoop简介与分布式安装

    Hadoop的基本概念和分布式安装: Hadoop 简介 Hadoop 是Apache Lucene创始人道格·卡丁(Doug Cutting)创建的,Lucene是一个应用广泛的文本搜索库,Hado ...

  10. 基于拖放布局的 Twitter Bootstrap 网站生成器

    简单的几个拖放操作就能做出漂亮的 Twitter Bootstrap 网站?是的,LayoutIt 是一个 Twitter Bootstrap 界面生成器,能够帮助你快速制作出网站和界面模型,同时能够 ...