黑马程序员-反射

1. 类的加载

1.1 概述
当程序要使用某个类时,如果该类还未被加载到内存中,则系统会通过加载,连接,初始化三步来实现对这个类进行初始化。
    a:加载
    就是指将class文件读入内存,并为之创建一个Class对象。
    任何类被使用时系统都会建立一个Class对象。
    b:连接
验证 是否有正确的内部结构,并和其他类协调一致
准备 负责为类的静态成员分配内存,并设置默认初始化值
解析 将类的二进制数据中的符号引用替换为直接引用
      c:初始化 就是我们以前讲过的初始化步骤

        1.2  类在什么时候加载呢?

创建类的实例
访问类的静态变量,或者为静态变量赋值
调用类的静态方法
使用反射方式来强制创建某个类或接口对应的java.lang.Class对象
初始化某个类的子类
直接使用java.exe命令来运行某个主类

        1.3 类的加载器

负责将.class文件加载到内在中,并为之生成对应的Class对象。
虽然我们不需要关心类加载机制,但是了解这个机制我们就能更好的理解程序的运行。

2. 反射

2.1 概述

JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;
对于任意一个对象,都能够调用它的任意一个方法和属性;
这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。
要想解剖一个类,必须先要获取到该类的字节码文件对象。
而解剖使用的就是Class类中的方法.所以先要获取到每一个字节码文件对应的Class类型的对象.

2.2 Class类

其实反射就是通过class文件对象,去使用该文件中的成员变量,构造方法,成员方法。

要想这样使用,首先你必须得到class文件对象,其实也就是得到Class类的对象。
   Class类:
    成员变量Field
    构造方法Constructor
  成员方法Method
 
  2.3  获取class文件对象的方式:
  A:Object类的getClass()方法
  B:数据类型的静态属性class
  C:Class类中的静态方法
   public static Class forName(String className)

一般在开发中我们使用第三种方式,因为第三种是一个字符串,而不是一个具体的类名。这样我们就可以把这样的字符串配置到配置文件中。

2.4 通过class文件获取构造方法

获取构造方法
public Constructor[] getConstructors():所有公共构造方法
public Constructor[] getDeclaredConstructors():所有构造方法

获取单个构造方法
public Constructor<T> getConstructor(Class<?>... parameterTypes)
参数表示的是:你要获取的构造方法的构造参数个数及数据类型的class字节码文件对象

通过Constructor对象来创建实例

public T newInstance(Object... initargs)
使用此 Constructor 对象表示的构造方法来创建该构造方法的声明类的新实例,并用指定的初始化参数初始化该实例。

示例:通过反射获取带参数构造方法并使用

public class ReflectDemo2 {
	public static void main(String[] args) throws Exception {
		// 获取字节码文件对象
		Class c = Class.forName("cn.itcast_01.Person");

		// 获取带参构造方法对象
		// public Constructor<T> getConstructor(Class<?>... parameterTypes)
		Constructor con = c.getConstructor(String.class, int.class,
				String.class);

		// 通过带参构造方法对象创建对象
		// public T newInstance(Object... initargs)
		Object obj = con.newInstance("xx", 27, "北京");
		
		System.out.println(obj);
	}
}
示例:通过反射获取私有构造方法并使用

注意,在使用私有构造方法的时候,我们需要通过使用如下方法才能够对构造方法进行访问

public void setAccessible(boolean flag):值为true,就指示反射的对象在使用时应该取消Java语言访问检查

</pre><pre name="code" class="java">public class ReflectDemo3 {
	public static void main(String[] args) throws Exception {
		// 获取字节码文件对象
		Class c = Class.forName("cn.itcast_01.Person");

		// 获取私有构造方法对象
		Constructor con = c.getDeclaredConstructor(String.class);

		// 用该私有构造方法创建对象
		// 暴力访问
		con.setAccessible(true);// 值为true则指示反射的对象在使用时应该取消Java语言访问检查。
		Object obj = con.newInstance("xx");

		System.out.println(obj);
	}
}
2.5 通过反射获取成员变量并使用
   获取所有的成员变量
Field[] fields = c.getFields();所有非私有的成员变量
Field[] fields = c.getDeclaredFields();所有的成员变量

  获取单个的成员变量:Class类下的方法

public Field getField(String name);name是成员变量的名称

public Field getDeclaredField(String name);获取私有成员变量,如果是私有,访问的时候需要对其设置Java访问setAccessible(true)

设置成员变量的值:Field下的方法

public void set(Object obj,Object value):将指定对象变量上此 Field 对象表示的字段设置为指定的新值。

示例:

public class PersonDemo {
	public static void main(String[] args) throws Exception{
		//获取单个的成员变量、
		Class c = Class.forName("cn.itcast_01.Person");
		
		//获取无参构造方法
		Constructor con = c.getConstructor();
		
		//通过构造方法获取实例
		Object person = con.newInstance();
		
		//获取单个成员变量
		Field  address = c.getField("address");
		address.set(person, "beijing");
	}
}
2.6 通过反射获取成员方法并使用
 获取所有的方法
 Method[] methods = c.getMethods(); // 获取自己的包括父亲的公共方法
 Method[] methods = c.getDeclaredMethods(); // 获取自己的所有的方法

获取单个方法

public Method getMethod(String name,Class<?>... parameterTypes):第一个参数表示的方法名,第二个参数表示的是方法的参数的class类型

如果是私有方法,访问的时候需要对其设置Java访问setAccessible(true)

使用该方法

public Object invoke(Object obj,Object... args): 返回值是Object接收,第一个参数表示对象是谁,第二参数表示调用该方法的实际参数

示例私有方法的访问:

public class PersonDemo2 {
	public static void main(String[] args) throws Exception {
		// 获取字节码文件
		Class c = Class.forName("cn.itcast_01.Person");

		// 获取构造方法
		Constructor con = c.getConstructor(String.class, int.class,
				String.class);

		// 获取实例
		Object person = con.newInstance("xx", 12, "beijing");

		// 获取私有成员方法show(String name)
		Method show = c.getDeclaredMethod("show", String.class);

		// 设置对私有成员方法访问的权限
		show.setAccessible(true);

		// 给成员方法赋值
		show.invoke(person, "yyy");
	}
}
示例:

反射方法访问配置文件

/*
 * 有这么一个配置文件,class.txt
 * 里面保存了两个键,分别是className和methodName
 *  通过这两个键和其中的值,调用不同类和类的方法
 *  
 *  class.txt
 *  className=cn.itcast.test.Student
 *  methodName=love
 */
public class Test {
	public static void main(String[] args) throws Exception {

		// 加载键值对数据
		Properties p = new Properties();
		FileReader fr = new FileReader("class.txt");
		p.load(fr);

		// 获取键
		String className = p.getProperty("className");
		String methodName = p.getProperty("methodName");

		// 发射方法,使用反射,在调用配置文件的时候,若有需求变化,只需要改配置文件里面的值就可以了
		Class c = Class.forName(className);
		Constructor con = c.getConstructor();
		Object obj = con.newInstance();

		// 调用方法
		Method method = c.getMethod(methodName);
		method.invoke(obj);
	}
}

3. 动态代理简介

3.1 动态代理概述

代理:本来应该自己做的事情,却请了别人来做,被请的人就是代理对象。
举例:春季回家买票让人代买
动态代理:在程序运行过程中产生的这个对象
而程序运行过程中产生对象其实就是我们刚才反射讲解的内容,所以,动态代理其实就是通过反射来生成一个代理

动态代理类:

Proxy 提供用于创建动态代理类和实例的静态方法,它还是由这些方法创建的所有动态代理类的超类

Proxy类中的方法创建动态代理类对象
public static Object newProxyInstance(ClassLoader loader,Class<?>[] interfaces,InvocationHandler h)
最终会调用InvocationHandler的方法
InvocationHandler接口中的方法
Object invoke(Object proxy,Method method,Object[] args):在代理实例上处理方法调用并返回结果。


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

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值