反射 复习


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();
}

}


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值