初见-Java的反射机制

概念

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

文章脉络

本篇将从以下几个方面讲述反射的知识:
- class的使用
- 方法的反射
- 构造函数的反射
- 成员变量的反射

什么是class类

在面向对象的世界里,万物皆对象。类是对象,类是java.lang.Class类的实例对象。另外class类只有java虚拟机才能new出来。任何一个类都是Class类的实例对象。这实例对象有三种表达方式:

public class User{   
}   

public class ClassTest{

User u=new User();   
//方式1(类名):   
Class c1=User.class;   
//方式2(对象名):   
Class c2=u.getClass();   
//方式3:
Class c3=Class.forName("com.forezp.User");   

//可以通过类的类型创建该类的实例对象   
User user=(User)c1.newInstance();   
}

class类的动态加载

Class.forName(类的全称);该方法不仅表示了类的类型,还代表了动态加载类。编译时刻加载类是静态加载、运行时刻加载类是动态加载类。

获取方法信息

基本的数据类型,void关键字都是Class 类的实例;可以通过getname();getSimpleName()获取类的名称。

//类名通过.class获得
Class c1=String.class;   
Class c2=int.class;   
Class c3=void.class;   
System.out.println(c1.getName());   
//输出:java.lang.String
System.out.println(c2.getSimpleName());
//输出:int
获取类的所有方法,并打印出来:
public static void printClassInfo(Object object){   
        Class c=object.getClass();   
        System.out.println("类的名称:"+c.getName());   

        /**
        * 一个成员方法就是一个method对象         *getMethods()所有的public方法,包括父类继承的public           
        * getDeclaredMethods()获取该类所有的方法,包括private ,但不包括继承的方法*/   
        Method[] methods=c.getMethods();//获取方法   
        //获取所以的方法,包括private ,c.getDeclaredMethods();   

        for(int i=0;i<methods.length;i++){   
            //得到方法的返回类型   
            Class returnType=methods[i].getReturnType();   
            System.out.print(returnType.getName());   
            //得到方法名:   
            System.out.print(methods[i].getName()+"(");   

            Class[] parameterTypes=methods[i].getParameterTypes();   
            for(Class class1:parameterTypes){   
                System.out.print(class1.getName()+",");   
            }   
            System.out.println(")");   
        }   
    }
public class ReflectTest {   

        public static void main(String[] args){   
                String s="ss";   
                ClassUtil.printClassInfo(s);   
        }   
}

运行:

类的名称:java.lang.String
booleanequals(java.lang.Object,)
java.lang.StringtoString()
inthashCode()

获取成员变量的信息

public static void printFiledInfo(Object o){

    Class c=o.getClass();   
    /**
    * getFileds()获取public           
    * getDeclaredFields()获取所有           */   
    Field[] fileds=c.getDeclaredFields();   

    for(Field f:fileds){   
        //获取成员变量的类型   
        Class filedType=f.getType();   
        System.out.println(filedType.getName()+" "+f.getName());   
    }   

}
            String s="ss";   
            //ClassUtil.printClassInfo(s);   
            ClassUtil.printFiledInfo(s);   
    }

运行:

[C value
int hash
long serialVersionUID
[Ljava.io.ObjectStreamField; serialPersistentFields
java.util.Comparator CASE_INSENSITIVE_ORDER
int HASHING_SEED
int hash32

获取构造函数的信息

     public static void printConstructInfo(Object o){   
        Class c=o.getClass();   

        Constructor[] constructors=c.getDeclaredConstructors();   
        for (Constructor con:constructors){   
            System.out.print(con.getName()+"(");   

            Class[] typeParas=con.getParameterTypes();   
            for (Class class1:typeParas){   
                System.out.print(class1.getName()+" ,");   
            }   
            System.out.println(")");   
        }   
    }
public static void main(String[] args){   
                String s="ss";   
                //ClassUtil.printClassInfo(s);   
                //ClassUtil.printFiledInfo(s);   
                ClassUtil.printConstructInfo(s);   
        }

运行:

java.lang.String([B ,)
java.lang.String([B ,int ,int ,)
java.lang.String([B ,java.nio.charset.Charset ,)
java.lang.String([B ,java.lang.String ,)
java.lang.String([B ,int ,int ,java.nio.charset.Charset ,)
java.lang.String(int ,int ,[C ,)
java.lang.String([C ,boolean ,)
java.lang.String(java.lang.StringBuilder ,)
java.lang.String(java.lang.StringBuffer ,)

六、方法反射的操作

获取一个方法:需要获取方法的名称和方法的参数才能决定一个方法。
方法的反射操作:
method.invoke(对象,参数列表);
举个例子:

class A{   

    public void add(int a,int b){   
        System.out.print(a+b);   
    }   

    public void toUpper(String a){   
        System.out.print(a.toUpperCase());   
    }   
}
class A{   
public static void main(String[] args) {   
        A a=new A();   
        Class c=a.getClass();   
        try {   
            Method method=c.getMethod("add",new Class[]{int.class,int.class});   
            //也可以 Method method=c.getMethod("add",int.class,int.class);   
            //方法的反射操作   
            method.invoke(a,10,10);   
        }catch (Exception e){   
            e.printStackTrace();   
        }   
    }

运行:

20

本篇文章已经讲解了java反射的基本用法, 它可以在运行时判断任意一个对象所属的类;在运行时构造任意一个类的对象;在运行时判断任意一个类所具有的成员变量和方法;在运行时调用任意一个对象的方法;生成动态代理。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

NobiGo

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值