Java反射(一)

Java的反射机制可以在运行时再装配代码,使程序的编写更加灵活.(高大上,看起来就很厉害的样子,然而到底有什么用呢?)
 Java的反射机制可以访问一个未知对象的属性,方法,构造方法(有参or无参),
 实现Java的反射共有三种方法:
 
1.通过forName()方法;
2.x.class;
3.x.getClass()。
 我们主要掌握第一种. 

下面来个Demo:

public static void main(String[] args) throws Exception {
  reflectTest();
  reflectTest2("Tom","Cat"); 
  reflectTest3("Ben","Dog"); 
  reflectTest4();
 }
 public static void reflectTest() throws Exception {
   System.out.println("1");
   Class tclass2 = Class.forName("reflect.bean.Pet");
   Object obj = tclass2.newInstance();// 调用非静态new 一个对象,调用静态方法则不需要
 
  }

  public static void reflectTest2(String name, String kind) throws Exception {
   System.out.println("2");
   Class tclass = Class.forName("reflect.bean.Pet");
   Constructor con = tclass.getConstructor(String.class, String.class);
   con.newInstance(name, kind);

  }
  public static void reflectTest3(String name, String kind) throws Exception{
   System.out.println("3");
    Class tclass = Class.forName("reflect.bean.Pet");
    Object obj =tclass.newInstance();
    Method met=tclass.getMethod("PetInfo", String.class,String.class);
    met.invoke(obj, name,kind);
   }
public static void reflectTest4() throws Exception {
   System.out.println("4");
   Class tclass2 = Class.forName("reflect.bean.Pet");
   Object obj = tclass2.newInstance();
   Method met =tclass2.getMethod("PetInfo", null);
   met.invoke(obj, null);
  }
Pet类

package reflect.bean;

public class Pet {
 private String kind;
 private String name;

 public Pet(String kind, String name) {
  super();
  this.kind = kind;
  this.name = name;
  System.out.println("Pet---" + kind + "---" + name);
 }

 public Pet() {
  super();
  System.out.println("Pet");
 }

 public void PetInfo() {
  System.out.println("tom---Cat");
 }

 public void PetInfo(String kind, String name) {
  System.out.println(name + "---" + kind);
 }
}
输出结果:

1
Pet
2
Pet---Tom---Cat
3
Pet
Dog---Ben
4
Pet
tom---Cat

我们来看看Class.forName()方法:

public static Class<?> forName(String className) 
                throws ClassNotFoundException {
 
        return forName0(className, true, ClassLoader.getCallerClassLoader());
    }

Class.forName()的主要任务就是把指定路径的类装载到jvm中.

static ClassLoader getCallerClassLoader() {
        // NOTE use of more generic Reflection.getCallerClass()
        Class caller = Reflection.getCallerClass(3);
        // This can be null if the VM is requesting it
        if (caller == null) {
            return null;
        }
        // Circumvent security check since this is package-private
        return caller.getClassLoader0();
    }

这里判断了对象的访问权限,如果访问对象修饰符为 private,需要 禁止Field的访问控制检查,setAccessible 设置为true.

static void setAccessible(AccessibleObject[] array, boolean flag) 
          使用单一安全性检查(为了提高效率)为一组对象设置 accessible 标志的便捷方法。
然后,newInstance(),代码太长,可以自己去看看.要使用这个方法必须在对应类加载并连接的状态下,而前面的Class.forName()就保证了newInstance的使用满足条件.(正是在这里调用Class的方法,从而获得了更好的灵活性.)



  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值