简约不简单--JAVA Reflection(反射机制)

Java 反射机制

反射机制简介

反射机制应用示例

简单的Ioc实现

代理模式

Java动态代理

简单的Aop实现

 

 

“程序运行时,允许改变程序结构或变量类型,这种语言称为动态语言”。从这个观点看,Perl,Python,Ruby是动态语言,C++,Java,C#不是动态语言。

 

尽管在这样得定义与分类下Java不是动态语言,它却有着一个非常突出的动态相关的机制:反射机制(Reflection)

 

 

什么是反射?

 

反射的概念是由Smith在1982年首次提出的,主要是指程序可以访问、检测和修改它本身状态或者行为的一种能力。

 

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

 

 

JAVA反射机制(Reflection)

 

动态获取类的信息,以及动态调用对象的方法的功能。

主要提供了以下功能: 

在运行时判断任意一个对象所属的类; 

在运行时构造任意一个类的对象; 

在运行时判断任意一个类所具有的成员变量和方法; 

在运行时调用任意一个对象的方法; 

生成动态代理。 

 

JAVA反射机制包

在 JDK 中,主要由以下类来实现 Java  反射机制,这些类都位于 java.lang.reflect包中。

Class 类:代表一个类。 

Field 类:代表类的成员变量(成员变量也称为类的属性)。 

Method 类:代表类的方法。 

Constructor 类:代表类的构造方法。 

Array 类:提供了动态创建数组,以及访问数组元素的静态方法。

 

java.lang.Class 

static Class forName(String className)

返回描述类名为className的Class对象

Object newInstance()

返回一个类的一个新实例

Field[] getFields()

返回包含Field对象的数组,这些对象记录了这个类或者其超类的公共域

Field[] getDeclaredField()

返回包含Field对象的数组,这些对象记录了这个类的全部域

 

Method[] getMethods()                             返回这个类或者其超类所有的公有方法

Method[] getDeclareMethods()                 返回这个类或者接口的所有方法,不包括超类继承的方法

Constructor[] getConstructors()                返回所有包含了Class对象所描述类的公有构造器

Constructor[] getDeclaredConstructors()  返回包含了Class对象所描述的类的所有构造器

 

Java.lang.reflect.Constructor

Class[] getParameterTypes()

返回一个用于描述参数类型的Class对象数组

getReturnType()

返回一个用于描述返回类型的Class对象 

int getModifiers()

返回一个用于描述方法抛出的异常类型的Class对象数组

Class getDeclaringClass()

 返回一个用于描述类中定义的构造器、方法或域的Class对象

 

 

通过反射实例化参数

 

平常实例化对象通常使用new关键字。但是使用new关键字实例化的对象具有强耦合性。New对象无法胜任未知对象的实例化,这时候只能通过反射动态生成。例如Spring的DI。

实例化无参构造函数的对象

Class.newInstance()

Class.getConstructor(new Class[]{}).newInstance(new Object[]{})

实例化带参构造函数的对象

Clazz.getConstructor(Class<?> ...ParameterType).newInstance(Object ...initargs)

 

 

反射机制举例

 

Java代码   收藏代码
  1. import java.lang.reflect.*;   
  2. public class DumpMethods {   
  3.     public static void main(String args[]) throws Exception{   
  4.         //加载并初始化命令行参数指定的类     
  5.         Class classType = Class.forName(args[0]);   
  6.         //获得类的所有方法   
  7.         Method methods[] = classType.getDeclaredMethods();   
  8.         for(int i = 0; i < methods.length; i++)   
  9.             System.out.println(methods[i].toString());   
  10.     }   
  11. }   

 

 

 

 

Java代码   收藏代码
  1. 输入:java DumpMethods java.util.Stack  
  2.   
  3. public synchronized java.lang.Object java.util.Stack.pop()   
  4. public java.lang.Object java.util.Stack.push(java.lang.Object)   
  5. public boolean java.util.Stack.empty()   
  6. public synchronized java.lang.Object java.util.Stack.peek()   
  7. public synchronized int java.util.Stack.search(java.lang.Object)   

 

运用反射机制调用方法

getMethod和invoke方法的时序图

简约不简单--JAVA Reflection(反射机制)_第1张图片
 

获取反射对象

 

Java代码   收藏代码
  1. import java.lang.reflect.*;   
  2. public class ReflectTester {   
  3.     public Object  copy(Object object) throws Exception{   
  4.         //获得对象的类型   
  5.         Class classType=object.getClass();   
  6.         System.out.println("Class:"+classType.getName());   
  7.         //通过默认构造方法创建一个新的对象   
  8.         Object  objectCopy=classType.getConstructor(new Class[]{}).   
  9. newInstance(new Object[]{});   
  10.         //获得对象的所有属性   
  11.         Field fields[]=classType.getDeclaredFields();   
  12.         for(int i=0; i<fields.length;i++){   
  13.             Field field=fields[i];   
  14.             String fieldName=field.getName();   
  15.             String firstLetter=fieldName.substring(0,1).toUpperCase();   
  16.             //获得和属性对应的 getXXX()方法的名字   
  17.             String getMethodName="get"+firstLetter+fieldName.substring(1);   
  18.             //获得和属性对应的 setXXX()方法的名字   
  19.             String setMethodName="set"+firstLetter+fieldName.substring(1);   
  20.             //获得和属性对应的 getXXX()方法   
  21.             Method getMethod=classType.getMethod(getMethodName,new Class[]{});   
  22.             //获得和属性对应的 setXXX()方法   
  23.             Method setMethod=classType.getMethod(setMethodName,new Class[]{field.getType()});   
  24.             //调用原对象的 getXXX()方法   
  25.             Object value=getMethod.invoke(object,new Object[]{});   
  26.             System.out.println(fieldName+":"+value);   
  27.             //调用复制对象的 setXXX()方法   
  28.             setMethod.invoke(objectCopy,new Object[]{value});   
  29.         }  
  30.         return objectCopy;   
  31.     }   
  32.   
  33.     public static void main(String[] args) throws Exception{   
  34.         Customer customer=new Customer("Tom",21);   
  35.         customer.setId(new Long(1));   
  36.    
  37.         Customer customerCopy=(Customer)new ReflectTester().copy(customer);   
  38.         System.out.println("Copy information:"+customerCopy.getName()+  
  39.                                       “ "+  customerCopy.getAge());   
  40.     }   
  41. }   

 

 

运用反射机制调用方法

Java代码   收藏代码
  1. class Customer{                
  2. //Customer 类是一个 JavaBean   
  3.     private Long id;   
  4.     private String name;   
  5.     private int age;    
  6.     public Customer(){}   
  7.     public Customer(String name,int age){   
  8.         this.name=name;   
  9.         this.age=age;   
  10.     }       
  11.     public Long getId(){return id;}   
  12.     public void setId(Long id){this.id=id;}   
  13.     public String getName(){return name;}   
  14.     public void setName(String name){this.name=name;}   
  15.     public int getAge(){return age;}   
  16.     public void setAge(int age){this.age=age;}   
  17. }  

 

Java代码   收藏代码
  1. import java.lang.reflect.*;   
  2. public class InvokeTester {   
  3.   public int add(int param1,int param2){  return param1+param2;   }   
  4.     public String echo(String msg){   return "echo:"+msg; }   
  5.     public static void main(String[] args) throws Exception{   
  6.         Class classType=InvokeTester.class;   
  7.         Object invokeTester=classType.newInstance();   
  8.         //调用 InvokeTester 对象的 add()方法   
  9.         Method addMethod=classType.getMethod("add",new Class[]{int.class,int.class});   
  10.         Object result=addMethod.invoke(invokeTester,   
  11.                     new Object[]{new Integer(100),new Integer(200)});   
  12.         System.out.println((Integer)result);   
  13.         //调用 InvokeTester 对象的 echo()方法   
  14.         Method echoMethod=classType.getMethod("echo",new Class[]{String.class});   
  15.         result=echoMethod.invoke(invokeTester,new Object[]{"Hello"});   
  16.         System.out.println((String)result);   
  17.     }   
  18. }  

你可能感兴趣的:(reflection)