反射以及动态代理深入研究以及对现有销售平台项目的改进思考

  • 反射概念、作用 
  • 反射应用场景
  • 反射缺点
  • 反射机制
  • 反射常见类介绍
  • 反射使用实例
  • 动态代理概念以及应用
  • 动态代理实例 
  • 在公司实际项目中的应用与思考
 
 
 
反射概念、作用 
概念:java程序编译后,在jvm虚拟机进行记载,在程序运行时刻,可通过虚拟机找到对应的类,以及其下的所有属性、以及方法,并能够动态执行其方法。这种动态获取信息以及动态调用方法的功能就叫做反射
作用:通过反射机制,可以在运行时访问 Java 对象的属性,方法,构造方法等。
 
 
反射的应用场景
  • 开发通用框架-例如spring框架通过配置文件加载不同的对象或者类,调用不同的方法
  • 动态代理-在切面编程(AOP)中,就需要通过动态代理来拦截特定的方法
  • 注解-根据反射原理,利用注解标示器来注解对应执行动作,如@AutoWare
  • 可扩展性功能-通过特定写入的名称来创建对应的实例类或者方法,来实现需要特定的功能 

反射的缺点

  • 性能开销 - 由于反射涉及动态解析的类型,因此无法执行某些 Java 虚拟机优化。因此,反射操作的性能要比非反射操作的性能要差,应该在性能敏感的应用程序中频繁调用的代码段中避免。
  • 破坏封装性 - 反射调用方法时可以忽略权限检查,因此可能会破坏封装性而导致安全问题。
  • 内部曝光 - 由于反射允许代码执行在非反射代码中非法的操作,例如访问私有字段和方法,所以反射的使用可能会导致意想不到的副作用,这可能会导致代码功能失常并可能破坏可移植性。反射代码打破了抽象,因此可能会随着平台的升级而改变行为。
 

反射机制

反射以及动态代理深入研究以及对现有销售平台项目的改进思考
类加载的完整过程如下:
  1. 在编译时,Java 编译器编译好 .java 文件之后,在磁盘中产生 .class 文件。.class 文件是二进制文件,内容是只有 JVM 能够识别的机器码。
  2. JVM 中的类加载器读取字节码文件,取出二进制数据,加载到内存中,解析.class 文件内的信息。类加载器会根据类的全限定名来获取此类的二进制字节流;然后,将字节流所代表的静态存储结构转化为方法区的运行时数据结构;接着,在内存中生成代表这个类的 java.lang.Class 对象。
  3. 加载结束后,JVM 开始进行连接阶段(包含验证、准备、初始化)。经过这一系列操作,类的变量会被初始化。

Class 对象

要想使用反射,首先需要获得待操作的类所对应的 Class 对象。Java 中,无论生成某个类的多少个对象,这些对象都会对应于同一个 Class 对象。这个 Class 对象是由 JVM 生成的,通过它能够获悉整个类的结构。所以,java.lang.Class 可以视为所有反射 API 的入口点。
反射的本质就是:在运行时,把 Java 类中的各种成分映射成一个个的 Java 对象。
举例来说,假如定义了以下代码:
User user = new User();
步骤说明:
  1. JVM 加载方法的时候,遇到 new User(),JVM 会根据 User 的全限定名去加载 User.class 。
  2. JVM 会去本地磁盘查找 User.class 文件并加载 JVM 内存中。
  3. JVM 通过调用类加载器自动创建这个类对应的 Class 对象,并且存储在 JVM 的方法区。注意:一个类有且只有一个 Class 对象。
 

反射常见类介绍

java.lang.reflect 包
Java 中的 java.lang.reflect 包提供了反射功能。java.lang.reflect 包中的类都没有 public 构造方法。
java.lang.reflect 包的核心接口和类如下:
  • Member 接口 - 反映关于单个成员(字段或方法)或构造函数的标识信息。
  • Field 类 - 提供一个类的域的信息以及访问类的域的接口。
  • Method 类 - 提供一个类的方法的信息以及访问类的方法的接口。
  • Constructor 类 - 提供一个类的构造函数的信息以及访问类的构造函数的接口。
  • Array 类 - 该类提供动态地生成和访问 JAVA 数组的方法。
  • Modifier 类 - 提供了 static 方法和常量,对类和成员访问修饰符进行解码。
  • Proxy 类 - 提供动态地生成代理类和类实例的静态方法。
反射使用实例
 
class加载的三种方式
  • class类的forName静态方法
  • 直接获取某一个对象的class
  • 调用 Object 的 getClass 方法
 
代码示例:
public class ReflectClass {
 
 
    public static void main(String[] args) throws ClassNotFoundException {
        //第一种方法:forName
        System.out.println("=======获取对象名============");
        getClassForName();
        //第二种方法:直接获取某一个对象的class
        System.out.println("=======获取class============");
        getTheClass();
        //第三种方法:获取Object的getClass方法
        System.out.println("=======获取object的getClass============");
        getObjectClass();
    }
 
 
    private static void getObjectClass() {
        Person P=new Person();
        Class person1=P.getClass();
        System.out.println(person1.getCanonicalName());
    }
 
 
    private static void getTheClass() {
        Class person1=Person.class;
        System.out.println(person1.getCanonicalName());
        Class person2=String.class;
        System.out.println(person1.getCanonicalName());
        System.out.println(person2.getCanonicalName());
    }
 
 
    private static void getClassForName() throws ClassNotFoundException {
        Class person1=Class.forName("com.zhouyunjian.demo.Person");
        System.out.println(person1.getCanonicalName());
        Class person2=Class.forName("[D");
        System.out.println(person2.getCanonicalName());
    }
}
 
 
 
 
class  Person{
    private String name;
    private int age;
 
 
    public String getName() {
        return name;
    }
 
 
    public void setName(String name) {
        this.name = name;
    }
 
 
    public int getAge() {
        return age;
    }
 
 
    public void setAge(int age) {
        this.age = age;
    }
}
 
执行结果:
=======获取对象名============
com.zhouyunjian.demo.Person
double[]
=======获取class============
com.zhouyunjian.demo.Person
com.zhouyunjian.demo.Person
java.lang.String
=======获取object的getClass============
com.zhouyunjian.demo.Person
 
 
判别类是否为某个类实例
  • instanceof关键字
  • class的isInstance方法
代码
public class InstanceofDemo {
    public static void main(String[] args) {
        check01();
    }
    private static void check01() {
        ArrayList arrayList = new ArrayList();
     System.out.println("======"+(arrayList instanceof List));
        System.out.println("======"+(List.class.isInstance(arrayList)));
    }
}
 
 
 
 
输出
======true
======true
 
 
 
创建实例
通过反射来创建实例对象主要有两种方式:
  1. 用 Class 对象的 newInstance 方法。
  2. 用 Constructor 对象的 newInstance 方法。
 
public class CreateInstance {
    public static void main(String[] args) throws InstantiationException, IllegalAccessException, NoSuchMethodException, InvocationTargetException {
      //newInstance创建实例
        createInstance1();
       //构造器创建实例
       createInstanceByConstructor();
    }
 
 
    private static void createInstanceByConstructor() throws NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException {
        Class<?> person=Person1.class;
        Constructor con=person.getConstructor(String.class,int.class);
 
 
        Person1 person1=(Person1)con.newInstance("zhouyunjian",18);
        System.out.println(person1);
    }
 
 
    private static void createInstance1() throws IllegalAccessException, InstantiationException {
        Class<?> person=Person1.class;
        System.out.println("======="+person.newInstance());
    }
 
 
 
 
}
 
 
class  Person1 {
    private String name;
    private int age;
 
 
    public Person1() {
    }
    public Person1(String name, int age) {
        this.name = name;
        this.age = age;
    }
 
 
    public String getName() {
        return name;
    }
 
 
    public void setName(String name) {
        this.name = name;
    }
 
 
    public int getAge() {
        return age;
    }
 
 
    public void setAge(int age) {
        this.age = age;
    }
 
 
    @Override
    public String toString() {
        return "Person1{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }
}
 
结果
=======Person1{name='null', age=0}
Person1{name='zhouyunjian', age=18}
 
 
Field
Class 对象提供以下方法获取对象的成员(Field):
  1. getFiled - 根据名称获取公有的(public)类成员。
  2. getDeclaredField - 根据名称获取已声明的类成员。但不能得到其父类的类成员。
  3. getFields - 获取所有公有的(public)类成员。
  4. getDeclaredFields - 获取所有已声明的类成员。
代码
private static void getFieldInfo() throws NoSuchFieldException {
    Class<?> person=Person1.class;
    //获取声明的成员变量信息
    Field field=person.getDeclaredField("name");
    System.out.println(field.getName());
    //获取public成员变量信息
    Field fieldAdd=person.getField("address");
    System.out.println(fieldAdd.getName());
    Field[] allFilds=person.getFields();
    System.out.println(allFilds.toString());
}
 
 
测试
name
address
[Ljava.lang.reflect.Field;@74a14482
 
 
Method
Class 对象提供以下方法获取对象的方法(Method):
  • getMethod - 返回类或接口的特定方法。其中第一个参数为方法名称,后面的参数为方法参数对应 Class 的对象。
  • getDeclaredMethod - 返回类或接口的特定声明方法。其中第一个参数为方法名称,后面的参数为方法参数对应 Class 的对象。
  • getMethods - 返回类或接口的所有 public 方法,包括其父类的 public 方法。
  • getDeclaredMethods - 返回类或接口声明的所有方法,包括 public、protected、默认(包)访问和 private 方法,但不包括继承的方法。
 
代码
//验证方法
private static void getMethodInfo() throws NoSuchMethodException, InvocationTargetException, IllegalAccessException, InstantiationException {
    Class<?> person=Person1.class;
    Method method=person.getDeclaredMethod("countPersons",String.class);
    method.invoke(person.newInstance(),"zhouynjian");
}
 
//定义方法
public static void  countPersons(String name){
    System.out.println("=========="+name);
}
 
测试
==========zhouynjian
 
 
Constructor
Class 对象提供以下方法获取对象的构造方法(Constructor):
  • getConstructor - 返回类的特定 public 构造方法。参数为方法参数对应 Class 的对象。
  • getDeclaredConstructor - 返回类的特定构造方法。参数为方法参数对应 Class 的对象。
  • getConstructors - 返回类的所有 public 构造方法。
  • getDeclaredConstructors - 返回类的所有构造方法。
获取一个 Constructor 对象后,可以用 newInstance 方法来创建类实例。
 
Array
数组在 Java 里是比较特殊的一种类型,它可以赋值给一个对象引用。下面我们看一看利用反射创建数组的例子:
代码
 
Class<?> cls = Class.forName("java.lang.String");
//设置数组长度
Object array = Array.newInstance(cls, 25);
//往数组里添加内容
Array.set(array, 0, "Scala");
Array.set(array, 1, "Java");
Array.set(array, 2, "Groovy");
Array.set(array, 3, "Scala");
Array.set(array, 4, "Clojure");
//获取某一项的内容
System.out.println(Array.get(array, 3));
 
测试
Scala
 
 
 
 

动态代理

动态代理是反射的一个非常重要的应用场景。动态代理常被用于一些 Java 框架中。例如 Spring 的 AOP ,Dubbo 的 SPI 接口,就是基于 Java 动态代理实现的。

静态代理

静态代理其实就是指设计模式中的代理模式。
代理模式为其他对象提供一种代理以控制对这个对象的访问。
反射以及动态代理深入研究以及对现有销售平台项目的改进思考
Subject 定义了 RealSubject 和 Proxy 的公共接口,这样就在任何使用 RealSubject 的地方都可以使用 Proxy 。
abstract class Subject {
public abstract void Request();
}
RealSubject 定义 Proxy 所代表的真实实体。
class RealSubject extends Subject {
@Override
public void Request() {
System.out.println("真实的请求");
}
}
Proxy 保存一个引用使得代理可以访问实体,并提供一个与 Subject 的接口相同的接口,这样代理就可以用来替代实体。
class Proxy extends Subject {
private RealSubject real;
@Override
public void Request() {
if (null == real) {
real = new RealSubject();
}
real.Request();
}
}
说明:
静态代理模式固然在访问无法访问的资源,增强现有的接口业务功能方面有很大的优点,但是大量使用这种静态代理,会使我们系统内的类的规模增大,并且不易维护;并且由于 Proxy 和 RealSubject 的功能本质上是相同的,Proxy 只是起到了中介的作用,这种代理在系统中的存在,导致系统结构比较臃肿和松散。
 
 

动态代理

为了解决静态代理的问题,就有了创建动态代理的想法:
在运行状态中,需要代理的地方,根据 Subject 和 RealSubject,动态地创建一个 Proxy,用完之后,就会销毁,这样就可以避免了 Proxy 角色的 class 在系统中冗杂的问题了。
反射以及动态代理深入研究以及对现有销售平台项目的改进思考
Java 动态代理基于经典代理模式,引入了一个 InvocationHandler,InvocationHandler 负责统一管理所有的方法调用。
动态代理步骤:
  1. 获取 RealSubject 上的所有接口列表;
  2. 确定要生成的代理类的类名,默认为:com.sun.proxy.$ProxyXXXX;
  3. 根据需要实现的接口信息,在代码中动态创建 该 Proxy 类的字节码;
  4. 将对应的字节码转换为对应的 class 对象;
  5. 创建 InvocationHandler 实例 handler,用来处理 Proxy 所有方法调用;
  6. Proxy 的 class 对象 以创建的 handler 对象为参数,实例化一个 proxy 对象。
从上面可以看出,JDK 动态代理的实现是基于实现接口的方式,使得 Proxy 和 RealSubject 具有相同的功能。
但其实还有一种思路:通过继承。即:让 Proxy 继承 RealSubject,这样二者同样具有相同的功能,Proxy 还可以通过重写 RealSubject 中的方法,来实现多态。CGLIB 就是基于这种思路设计的。
在 Java 的动态代理机制中,有两个重要的类(接口),一个是 InvocationHandler 接口、另一个则是 Proxy 类,这一个类和一个接口是实现我们动态代理所必须用到的。

InvocationHandler 接口

InvocationHandler 接口定义:
public interface InvocationHandler {
public Object invoke(Object proxy, Method method, Object[] args)
throws Throwable;
}
每一个动态代理类都必须要实现 InvocationHandler 这个接口,并且每个代理类的实例都关联到了一个 Handler,当我们通过代理对象调用一个方法的时候,这个方法的调用就会被转发为由 InvocationHandler 这个接口的 invoke 方法来进行调用。
我们来看看 InvocationHandler 这个接口的唯一一个方法 invoke 方法:
Object invoke(Object proxy, Method method, Object[] args)throws Throwable
参数说明:
  • proxy - 代理的真实对象。
  • method - 所要调用真实对象的某个方法的 Method 对象
  • args - 所要调用真实对象某个方法时接受的参数
如果不是很明白,等下通过一个实例会对这几个参数进行更深的讲解。

Proxy 类

Proxy 这个类的作用就是用来动态创建一个代理对象的类,它提供了许多的方法,但是我们用的最多的就是 newProxyInstance 这个方法:
public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h) throws IllegalArgumentException
这个方法的作用就是得到一个动态的代理对象。
参数说明:
  • loader - 一个 ClassLoader 对象,定义了由哪个 ClassLoader 对象来对生成的代理对象进行加载。
  • interfaces - 一个 Interface 对象的数组,表示的是我将要给我需要代理的对象提供一组什么接口,如果我提供了一组接口给它,那么这个代理对象就宣称实现了该接口(多态),这样我就能调用这组接口中的方法了
  • h - 一个 InvocationHandler 对象,表示的是当我这个动态代理对象在调用方法的时候,会关联到哪一个 InvocationHandler 对象上

动态代理实例

上面的内容介绍完这两个接口(类)以后,我们来通过一个实例来看看我们的动态代理模式是什么样的:
首先我们定义了一个 Subject 类型的接口,为其声明了两个方法:
public interface Subject {
void hello(String str);
String bye();
}
接着,定义了一个类来实现这个接口,这个类就是我们的真实对象,RealSubject 类:
public class RealSubject implements Subject {
@Override
public void hello(String str) {
System.out.println("Hello " + str);
}
@Override
public String bye() {
System.out.println("Goodbye");
return "Over";
}
}
下一步,我们就要定义一个动态代理类了,前面说个,每一个动态代理类都必须要实现 InvocationHandler 这个接口,因此我们这个动态代理类也不例外:
public class InvocationHandlerDemo implements InvocationHandler {
// 这个就是我们要代理的真实对象
private Object subject;
// 构造方法,给我们要代理的真实对象赋初值
public InvocationHandlerDemo(Object subject) {
this.subject = subject;
}
@Override
public Object invoke(Object object, Method method, Object[] args)
throws Throwable {
// 在代理真实对象前我们可以添加一些自己的操作
System.out.println("Before method");
System.out.println("Call Method: " + method);
// 当代理对象调用真实对象的方法时,其会自动的跳转到代理对象关联的handler对象的invoke方法来进行调用
Object obj = method.invoke(subject, args);
// 在代理真实对象后我们也可以添加一些自己的操作
System.out.println("After method");
System.out.println();
return obj;
}
}
最后,来看看我们的 Client 类:
public class Client {
public static void main(String[] args) {
// 我们要代理的真实对象
Subject realSubject = new RealSubject();
// 我们要代理哪个真实对象,就将该对象传进去,最后是通过该真实对象来调用其方法的
InvocationHandler handler = new InvocationHandlerDemo(realSubject);
/*
* 通过Proxy的newProxyInstance方法来创建我们的代理对象,我们来看看其三个参数
* 第一个参数 handler.getClass().getClassLoader() ,我们这里使用handler这个类的ClassLoader对象来加载我们的代理对象
* 第二个参数realSubject.getClass().getInterfaces(),我们这里为代理对象提供的接口是真实对象所实行的接口,表示我要代理的是该真实对象,这样我就能调用这组接口中的方法了
* 第三个参数handler, 我们这里将这个代理对象关联到了上方的 InvocationHandler 这个对象上
*/
Subject subject = (Subject)Proxy.newProxyInstance(handler.getClass().getClassLoader(), realSubject
.getClass().getInterfaces(), handler);
System.out.println(subject.getClass().getName());
subject.hello("World");
String result = subject.bye();
System.out.println("Result is: " + result);
}
}
我们先来看看控制台的输出:
com.sun.proxy.$Proxy0
Before method
Call Method: public abstract void io.github.dunwu.javacore.reflect.InvocationHandlerDemo$Subject.hello(java.lang.String)
Hello World
After method
Before method
Call Method: public abstract java.lang.String io.github.dunwu.javacore.reflect.InvocationHandlerDemo$Subject.bye()
Goodbye
After method
Result is: Over
我们首先来看看 com.sun.proxy.$Proxy0 这东西,我们看到,这个东西是由 System.out.println(subject.getClass().getName()); 这条语句打印出来的,那么为什么我们返回的这个代理对象的类名是这样的呢?
Subject subject = (Subject)Proxy.newProxyInstance(handler.getClass().getClassLoader(), realSubject
.getClass().getInterfaces(), handler);
可能我以为返回的这个代理对象会是 Subject 类型的对象,或者是 InvocationHandler 的对象,结果却不是,首先我们解释一下为什么我们这里可以将其转化为 Subject 类型的对象
原因就是:在 newProxyInstance 这个方法的第二个参数上,我们给这个代理对象提供了一组什么接口,那么我这个代理对象就会实现了这组接口,这个时候我们当然可以将这个代理对象强制类型转化为这组接口中的任意一个,因为这里的接口是 Subject 类型,所以就可以将其转化为 Subject 类型了。
同时我们一定要记住,通过 Proxy.newProxyInstance 创建的代理对象是在 jvm 运行时动态生成的一个对象,它并不是我们的 InvocationHandler 类型,也不是我们定义的那组接口的类型,而是在运行是动态生成的一个对象,并且命名方式都是这样的形式,以$开头,proxy 为中,最后一个数字表示对象的标号。
接着我们来看看这两句
subject.hello("World");
String result = subject.bye();
这里是通过代理对象来调用实现的那种接口中的方法,这个时候程序就会跳转到由这个代理对象关联到的 handler 中的 invoke 方法去执行,而我们的这个 handler 对象又接受了一个 RealSubject 类型的参数,表示我要代理的就是这个真实对象,所以此时就会调用 handler 中的 invoke 方法去执行。
我们看到,在真正通过代理对象来调用真实对象的方法的时候,我们可以在该方法前后添加自己的一些操作,同时我们看到我们的这个 method 对象是这样的:
public abstract void io.github.dunwu.javacore.reflect.InvocationHandlerDemo$Subject.hello(java.lang.String)
public abstract java.lang.String io.github.dunwu.javacore.reflect.InvocationHandlerDemo$Subject.bye()
正好就是我们的 Subject 接口中的两个方法,这也就证明了当我通过代理对象来调用方法的时候,起实际就是委托由其关联到的 handler 对象的 invoke 方法中来调用,并不是自己来真实调用,而是通过代理的方式来调用的。
 
 
 
在公司实际项目中的应用与思考
 
基于风控系统规则引擎的设计优化
 
1、规则方式支持表格类型维护
  • 表格可以设置多选条件,多个结果
  • 表格支持横向扩展条件、结果值
  • 表格支持纵向扩展传值条件参数
 
风险等级比较
条件1
条件2
结果值(状态位)
客户等级A
产品等级A
可交易
客户等级B
产品等级B
不可交易
风险评测
条件1
结果值(等级)
100 分
E级
90分
D级
 
占额维护
条件1
条件2
结果值1(比例)
结果值2(比例)
对公客户
法人
保证金占用50%
授信占用50%
对公客户
非法人
保证金占用100%
保证金占用0%
 
2、规则引擎数据源接口,通过反射机制,扩展数据源功能,增强规则配置功能
 
2.1 、数据源中反射方法名支持传参: 原方法名(无参)->新方法名('参数')
 
2.2 、数据源数据结果扩展升级: 支持选择对应实体以及实体下的字段条件
 
客户实体类(客户类型、是否法人、客户等级等)
 
产品实体类(产品代码 、产品类型等)
 
2.3、数据源接口可扩展设计: 规则新增字段,可传入对应service接口。
 
原数据源接口dataSourceService:所有新增数据源均在此中新增方法,扩展性不够高,未来会越来越多
 
新数据源接口xxxSourceService: 按照各个业务类别进行扩展,类别越多,数据源越多  
 
2.4、数据源配置开放 
 
开放数据源配置包,通过管理端页面配置客户所需要的数据源信息。例如
 
此业务需要用到  客户风险等级信息,现在的数据源包中不支持,那么此时可通过新增 客户风险等级信息来维护 
 
总结:未来,规则扩展的关键点在数据源上,数据源可以做的越来越大,越来越广,适配任何可用场景。
 
3、大数据量规则场景配置
如果未来规则要越做越大,要考虑 具体客户、具体账户、具体交易等等适配不同个体下的场景。
 
以10万级别的规则量设计考虑。
 
  • 此规则必定与原规则不同路,防止影响原规则方式
  • 此规则需要适配各种各样不同客户化场景下的可定制化需求
  • 此规则必须达到高效、可用的情况
 
例子:  
多条件
多结果
客户id=1001,账户号=1002
期权权重=80%
商品权重=80%
 
3、向导式配置方式
 
规则配置向导说明,指导性演示如何配置
 
假设配置指导步骤:   数据源配置-> 新增规则 -> 测试规则 -> 规则接口调用说明  。
 
或者可能是一个流程引擎方式配置: 
 
新增流程引擎: 如上指导步骤
 
测试流程引擎: 如上规则接口调用说明
 
4、预定义计算表达式扩展 
 
支持基于预定义计算公式的扩展实现。开放可预定义接口RuleExpressExtendService方式,实现此接口即可扩展计算表达式 。
 
例如需要加总方式:   ∑   
1、实现RuleExpressExtendService接口 
2、实现 ∑对应的规则表达式:  engine.eval("def ∑(a,b){return a+b;}");
3、注入service注解 ,会被规则预定义公式自动扫描到即可直接使用 
 
 
5、规则默认配置扩展 
 
  • 配置规则时可以选择是否需要配置默认值
  • 如果配置默认值,则调用规则未生效的情况下,采用默认值数据
  • 可在3 配置向导中加此步骤