JavaSE基础知识梳理——反射

九、反射

  • 反射内容主要在java.lang.reflect包下,包括Method、Field、Constructor、Proxy、Class等类。
  • 反射的好处:解耦。new的方式在编码期间就增加了耦合度。每个文件必须有这个对象才能运行,而反射就不存在。反射可以通过配合IO流读取配置文件,根本不需要修改源码就可以新建类。
  • JVM先加载.class到内存中,并为它们创建一个Class类型的实例与之关联,Class的构造方法是private的,只有JVM可以创建它的实例,像.class,getClass等都是JVM创建的实例,不是new出来的。Class实例是唯一的,它代表这个类。
  • 反射的动态加载类指的是相对于不反射而言,类加载器会直接加载在代码中new出来的对象的类到内存中,而反射是运行时执行到了才加载,如果没有就不加载。

9.1 获取Class的三种方式

==三种方式获取字节码==

Class.forName(类的全路径) 或者获取本工程类路径下的类名(普通工程src下,maven的java下),而且会把该类加载的JVM中,可以和静态代码块联用。
实例变量.getClass()这种方式会把类加载到JVM中,因为实例变量已经被new出来了才能够调用.getClass
类.class 这种方式不会加载类到JVM中

9.2 常用方法

Object newInstance():反射构造对象是在运行时创建的,因为字节码是编译之后才有,无参构造对象
Object getConstructor().newInstance(Object... parameters)通过带参构造创建对象
    如:
    {
    Constructor<User> constructor = User.class.getConstructor(Integer.class,Double.class);
    User user = constructor.newInstance(4,2000d);}

String getSimpleName:获得纯类名
boolean isAnonymousClass:判断是否是基础类
Field getFiled():获取public的,而且可以获取父类的
Field getDeclaredField(),获取所有权限,只限当前类
boolean isAssignableFrom(Class),表示当前类能否由Class指派,也就是能否多态,如Number.class.isAssignableFrom(Integer.class)返回true
field.get(Object obj),获取指定实例的指定值,对于static类型可以填null或者.class因为它不属于任何实例
    获取目标的类路径的一种方法:(类路径是.class的路径)
    当前线程的类加载器默认从src(或者maven的java\resources)下去加载文件,获得的是绝对路径
    Thread.currentThread().getContextClassLoader().getResource("").getPath()
    -ResourceBundle:资源绑定器,是个抽象类(类路径下,且只能绑定.properties文件,参数为不带后缀名的字符串)
ResourceBundle r = ResourceBundle.getBundle("")
    int getModifies():获得权限修饰符,返回一个数值,解码的话用Modifier的静态方法
    Modifier.toString(int modifier)
    void setAccessable():开启或关闭安全检查(访问权限)

9.3 动态代理

  • 动态代理是jdk提供的一种技术,能够对已有方法进行增强。

  • 基于接口的动态代理,类加载器写的是接口的某个实现类的类加载器。要获得的是接口的动态代理对象,要传入的是接口的实现类的类加载器。最后把得到的代理对象向上转型为接口。

  • image-20200606002806430.png
```java
public static void main(String[] args) throws IOException {
        Customer customer = new Customer("Customer");
        Order o = (Order) Proxy.newProxyInstance(Customer.class.getClassLoader(), Customer.class.getInterfaces(),
                new InvocationHandler() {
                    @Override
                    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                        System.out.println("代理开始");
                        return method.invoke(customer,args);
                    }
                });
        o.buy(new Good());
    }
```

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。