九、反射
- 反射内容主要在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提供的一种技术,能够对已有方法进行增强。
基于接口的动态代理,类加载器写的是接口的某个实现类的类加载器。要获得的是接口的动态代理对象,要传入的是接口的实现类的类加载器。最后把得到的代理对象向上转型为接口。
```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()); } ```