黑马程序员——反射


------Java培训、Android培训、iOS培训、.Net培训、期待与您交流! -------

反射:

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

 

     如果要使用反射,就要先获取到类的字节码文件对象(.class),那么获取字节码文件对象有三种方式:

       1、只要是数据类型,都可以通过类型名称,直接.class获取。

       2、创建类对象,调用Object类中的getClass()方法。

       3、知道类的路径名称,调用Class类中的forName(String className)方法。此方法会报出“ClassNotFoundException”编译期异常。

class Demo {
	private int age;

	public Demo() {
	}

	public int getAge() {
		return age;
	}

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

public class ReflectDemo {
	public static void main(String[] args) {
		// 第一种方式
		Class c1 = Demo.class;
        
		//第二种方式
		Demo d = new Demo();
		Class c2 = d.getClass();
		
		//第三种方式
		try{
			Class c3 = Class.forName("Test2.ReflectDemo");
		}catch(ClassNotFoundException ce){
			ce.printStackTrace();
		}
	}
}

 注意:在开发中通常使用第三种方式,所以要掌握。

      

     在获取了字节码文件对象后,我们就要通过此对象去查看类的信息,可以获取类的构造方法(Constructor)、成员变量(Field)、成员方法(Method)

     

     ConstructorFieldMethod都是继承自AccessibleObject类。该类提供将反射的对象标记为在使用时取消默认Java语言访问控制检查的能力。

     对此AccessibleOngect类提供了方法:public void setAccessible(boolean flag);

     如果值为true则指示反射的对象在使用是应该取消Java语言访问检查。为false则指示反射的对象应该实施Java语言访问检查。所以该方法也被称为暴力访问。    

 

一、获取构造方法:

Constructor类提供了关于类的单个构造方法的信息以及对它的访问权限。

 

1、获取指定的构造方法:

        A、只能获取指定公共构造方法

      public Constructor getConstructor(Class... argeName);

    B、获取任意指定构造方法(包括私有构造)

           Public Constructor getDeclaredConstructor(Class... argeName);

2、获取所有的构造方法:

    A、只能获取所有公共构造方法

       public Constructor[] getConstructors();

    B、获取所有任意构造方法(包括私有构造)

       Public Constructor[] getDeclaredConstructors();

 

 在获取了构造方法后,我们可以通过Constructor类的newInstance(Object...args)方法,创建此类的一个新实例:

import java.lang.reflect.Constructor;

class Person {
        //私有成员变量
	private String name;

	public Person() {
		System.out.println("public Person");
	}
	//私有带参构造
	private Person(String name){
		System.out.println("private Person");
		this.name = name;
	}
	@Override
	public String toString() {
		return "Person [name=" + name + "]";
	}
}

public class ReflectDemo {
	public static void main(String[] args) throws Exception {
        Class c = Class.forName("Test2.Person");
        
        //避免访问不到私有构造,尽量使用带Declared的方法和暴力访问方法
        Constructor con = c.getDeclaredConstructor(String.class);
        con.setAccessible(true);
        /**
         * 注意:访问的是什么数据类型的参数构造,
         *          因此创建的实例也要是对应的数据类型。
         *          无参则不需要
         */
        Object obj = con.newInstance("王一");
        System.out.println(obj);
	}
        /**
         *结果:
         *    private Person
         *    Person [name=王一]
         */
}

 

二、获取成员变量:

   Field类提供有关类或接口的单个字段的信息,以及对它的动态访问权限。

   1、获取指定的成员变量:

      A、只能获取指定公共成员变量

      public Field getField(String name);

  B、获取任意指定成员变量(包括私有变量)

           Public Field getDeclaredField(String name);

   2、获取所有的成员变量:

    A、只能获取所有公共成员变量

       public Field[] getFields();

    B、获取所有任意成员变量(包括私有变量)

       Public Field[] getDeclaredFields();

 

  在获取成员变量后,我们可以通过构造方法创建的新实例,以及Field类的get()和set()来获取和设置成员变量:

import java.lang.reflect.Constructor;
import java.lang.reflect.Field;

class Person {
	// 私有成员变量
	private String name;

	public Person() {
		System.out.println("public Person");
	}

	// 私有带参构造
	private Person(String name) {
		System.out.println("private Person");
		this.name = name;
	}

	@Override
	public String toString() {
		return "Person [name=" + name + "]";
	}
}

public class ReflectDemo {
	public static void main(String[] args) throws Exception {
		Class c = Class.forName("Test2.Person");

		// 避免访问不到私有构造,尽量使用带Declared的方法和暴力访问方法
		Constructor con = c.getDeclaredConstructor();
		con.setAccessible(true);
		Object obj = con.newInstance(); 

		// 避免访问不到私有成员变量,尽量使用带Declared的方法和暴力访问方法
		Field f = c.getDeclaredField("name");
		f.setAccessible(true);
        //获取变量,还没赋值,打印出默认值null
		f.get(obj);
		System.out.println(obj);
		//设置变量
		f.set(obj, "王一");
		System.out.println(obj);
	}
	/**
	 * 结果:
	 *       public Person
	 *       Person [name=null]
	 *       Person [name=王一]
	 */
}

   

三、获取成员方法:

   Method类提供关于类或接口上单独某个方法(以及如何访问该方法)的信息。

   1、获取指定的成员方法:

      A、只能获取指定公共成员方法

      public Field getMethod(String name,Class...argsName);

  B、获取任意指定成员方法(包括私有方法)

           Public Field getDeclaredMethod(String name,Class...argsName);

   2、获取所有的成员方法:

  A、只能获取所有公共成员方法

       public Method[] getMethod();

  B、获取所有任意成员方法(包括私有方法)

       Public Method[] getDeclaredMethods();

在获取成员方法后,我们可以通过构造方法创建的新实例,以及Method类的getName()和invoke(Object obj,Object...args)来获取和设置成员方法:

import java.lang.reflect.Constructor;
import java.lang.reflect.Method;

class Person {
	private String name;

	public Person() {
		System.out.println("public Person");
	}

	private String method(String str) {
		return str;
	}

	@Override
	public String toString() {
		return "Person [name=" + name + "]";
	}
}

public class ReflectDemo {
	public static void main(String[] args) throws Exception {
		Class c = Class.forName("Test2.Person");

		// 避免访问不到私有构造,尽量使用带Declared的方法和暴力访问方法
		Constructor con = c.getDeclaredConstructor();
		con.setAccessible(true);
		Object obj = con.newInstance();

		// 避免访问不到私有成员方法,尽量使用带Declared的方法和暴力访问方法
		Method m = c.getDeclaredMethod("method", String.class);
		m.setAccessible(true);
		// 获取方法名
		String name = m.getName();
		System.out.println(name);
		//设置方法参数
		String str = (String)m.invoke(obj, "Hello");
		System.out.println(str);
	}
	/**
	 * 结果: 
	 *          public Person
	 *          method
	 *          Hello 
	 */
}

 

总结:

1、通过Constructor、Field、Method三个类的方法可以发现,想要获取到私有的类成员,使用的方法必须是有带Declared关键字的。而当我们用反射访问一个类的时候,并不能知道有没有私有成员,所以为了避免无法获取私有成员,应该都要优先使用带Declared的方法。

2、使用带Declared的方法是我们可以获取到私有成员的外部信息,但并不代表就可以去访问内部资源。所以为了避免无法访问私有成员的内容,应该都要使用成员对象调用setAccessible(boolean flag)方法。

------ Java培训、Android培训、iOS培训、.Net培训、期待与您交流! -------
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值