title: 反射复习
date: 2020/8/17
1. 类的加载过程:
程序经过javac.exe命令以后,会生成一个或多个字节码文件(.class结尾)。
接着我们使用java.exe命令对某个字节码文件进行解释运行。相当于将某个字节码文件
加载到内存中。此过程就称为类的加载。加载到内存中的类,我们就称为运行时类,此
运行时类,就作为Class的一个实例。
2.换句话说,Class的实例就对应着一个运行时类。
3.加载到内存中的运行时类,会缓存一定的时间。在此时间之内,我们可以通过不同的方式
来获取此运行时类。
1.1 类加载器的作用
将class文件字节码内容加载到内存中,并将这些静态数据转换成方法区的运行时数据结构,然后在堆中生成一个代表这个类的java.lang.Class对象,作为方法区中类数据的访问入口
1.2 类加载器的分类
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-btoiougK-1631612471827)(C:\Users\ZZR\AppData\Roaming\Typora\typora-user-images\image-20210904230852357.png)]
2. 获取Class实例的几种方式:(前三种方式需要掌握)
//方式一:调用运行时类的属性:.class
Class clazz1 = Person.class;
System.out.println(clazz1);
//方式二:通过运行时类的对象,调用getClass()
Person p1 = new Person();
Class clazz2 = p1.getClass();
System.out.println(clazz2);
//方式三:调用Class的静态方法:forName(String classPath)
Class clazz3 = Class.forName("com.atguigu.java.Person");
// clazz3 = Class.forName("java.lang.String");
System.out.println(clazz3);
System.out.println(clazz1 == clazz2);
System.out.println(clazz1 == clazz3);
//方式四:使用类的加载器:ClassLoader (了解)
ClassLoader classLoader = ReflectionTest.class.getClassLoader();
Class clazz4 = classLoader.loadClass("com.atguigu.java.Person");
System.out.println(clazz4);
System.out.println(clazz1 == clazz4);
3. 创建类的对象的方式
方式一:new + 构造器
方式二:要创建Xxx类的对象,可以考虑:Xxx、Xxxs、XxxFactory、XxxBuilder类中查看是否有静态方法的存在。可以调用其静态方法,创建Xxx对象。
方式三:通过反射
@Test
public void test1() throws IllegalAccessException, InstantiationException {
//创建对象方式1
Person person = new Person();
//创建对象方式2,newInstance创建对应的运行时类的对象,内部调用了运行时类的空参构造器
Class<Person> clazz = Person.class;
Person person1 = clazz.newInstance();
System.out.println(person);
System.out.println(person1);
}
注:
newInstance():调用此方法,创建对应的运行时类的对象。内部调用了运行时类的空参的构造器。
要想此方法正常的创建运行时类的对象,要求:
1.运行时类必须提供空参的构造器
2.空参的构造器的访问权限得够。通常,设置为public。
在javabean中要求提供一个public的空参构造器。原因:
1.便于通过反射,创建运行时类的对象
2.便于子类继承此运行时类时,默认调用super()时,保证父类此构造器
4. 获取运行时类的指定属性值
@Test
public void test4() throws Exception{
//1.实例化Class对象
Class clazz = Person.class;
//2.创建运行时类
Person p = (Person) clazz.newInstance();
//3.获取指定变量名的属性
Field name = clazz.getDeclaredField("name");
//4.保证当前属性是可访问的
name.setAccessible(true);
/*
设置当前属性的值 set(指明设置哪个对象的属性,具体值)
*/
name.set(p, "zzr");
/*
get(执行对象)
*/
Object o = name.get(p);
System.out.println(o);
}
5. 获取运行时类的指定方法值
@Test
public void test5() throws Exception {
//1.实例化Class对象
Class clazz = Person.class;
//2.创建运行时类
Person p = (Person) clazz.newInstance();
/*
3.获取指定的某个方法
getDeclaredMethod(指明获取的方法的名称,指明获取的方法的形参列表)
*/
Method show = clazz.getDeclaredMethod("show", String.class);
//4.保证当前属性是可访问的
show.setAccessible(true);
/*
5.invoke(方法的调用者 给方法形参赋值的实参)
*/
Object invoke = show.invoke(p, "请开始你的表演");
System.out.println(invoke);
System.out.println("*****************调用静态方法*********************");
Method showDesc = clazz.getDeclaredMethod("showDesc");
showDesc.setAccessible(true);
//Object invoke1 = showDesc.invoke(p);
//静态
Object invoke1 = showDesc.invoke(null);
System.out.println(invoke1);
}
6. 静态代理
package com.zzzzzzsr;
/**
* @author ZZR
* @Description 静态代理
* @creat
*/
interface ClothFactory{
void produceCloth();
}
//代理类
class ProxyClothFactory implements ClothFactory{
//用被代理类对象进行实例化
private ClothFactory factory;
public ProxyClothFactory(ClothFactory factory) {
this.factory = factory;
}
@Override
public void produceCloth() {
System.out.println("代理工厂做一些准备工作");
factory.produceCloth();
System.out.println("代理工厂做一些后续收尾工作");
}
}
//被代理类
class NIkeClothFactory implements ClothFactory{
@Override
public void produceCloth() {
System.out.println("NIKE工厂生产一批运动服");
}
}
public class StaticProxyTest {
public static void main(String[] args) {
NIkeClothFactory nike = new NIkeClothFactory();
ProxyClothFactory proxyClothFactory = new ProxyClothFactory(nike);
proxyClothFactory.produceCloth();
}
}
7. 动态代理
package com.zzzzzzsr;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
/**
* @author ZZR
* @Description 接口、被代理类、代理类变成动态
* 问题1:如何根据加载到内存中的被代理类,动态的创建一个代理类及其对象
* 问题2:当通过代理类的对象调用方法时,如何动态的去调用被代理类中的同名方法
*
* @creat
*/
//公共接口
interface Human{
String getBelief();
void eat(String food);
}
//被代理类
class SuperMan implements Human{
@Override
public String getBelief() {
return "I believe I can fly!";
}
@Override
public void eat(String food) {
System.out.println("我喜欢吃"+food);
}
}
//生产代理类的工厂
class ProxyFactory{
//调用此静态方法,返回一个代理类的对象,解决问题1
public static Object getProxyInstance(Object obj){//obj:被代理类的对象,Object是代理类的类型
MyInvocationHandler handler = new MyInvocationHandler();
handler.bind(obj);
//利用被代理类的类加载器、接口动态的创建一个代理类的对象
return Proxy.newProxyInstance(obj.getClass().getClassLoader(),obj.getClass().getInterfaces(),handler); //参数1:类的加载器
}
}
//调用代理类对象的方法,动态的调用被代理类的方法
class MyInvocationHandler implements InvocationHandler{
//创建一个被代理类的对象,需要使用被代理类的对象进行赋值
private Object obj;
public void bind(Object obj){
this.obj = obj;
}
//当我们通过代理类的对象,调用方法a时,就会自动的调用如下的方法:invoke(),解决问题2
//将被代理类要执行的方法a的功能声明在invoke()方法中
//proxy:代理类的对象 method:代理类调的方法
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
//method:即为代理类对象调用的方法,此方法也就最作为了被代理类对象要调用的方法
//obj被代理类的对象
Object returnValue = method.invoke(obj, args);
//上述方法的返回值就作为当前类中的invoke()的返回值
return returnValue;
}
}
//main函数
public class ProxyTest {
public static void main(String[] args) {
//1.获取一个代理类的对象
SuperMan superMan = new SuperMan();
Human proxyInstance = (Human) ProxyFactory.getProxyInstance(superMan);
//2.调用代理类对象的方法,就动态的调用了被代理类中同名的方法
String belief = proxyInstance.getBelief();
System.out.println(belief);
proxyInstance.eat("四川麻辣烫");
System.out.println("*****换个被代理的类********");
NIkeClothFactory nIkeClothFactory = new NIkeClothFactory();
ClothFactory proxyInstance1 = (ClothFactory) ProxyFactory.getProxyInstance(nIkeClothFactory);
proxyInstance1.produceCloth();
}
}
e.eat(“四川麻辣烫”);
System.out.println("*****换个被代理的类********");
NIkeClothFactory nIkeClothFactory = new NIkeClothFactory();
ClothFactory proxyInstance1 = (ClothFactory) ProxyFactory.getProxyInstance(nIkeClothFactory);
proxyInstance1.produceCloth();
}
}