Java效率提升神器jOOR

 更新时间:2022年07月07日 11:27:51   作者:​ 你呀不牛   ​  
这篇文章主要介绍了Java效率提升神器jOOR,jOOR是一个第三方库,通过链式DSL接口,简化了反射过程,更多相关内容组要的朋友可以参考一下

前言

Java中的原生反射库虽然方法不多,但写起来却非常繁琐, 比如:

public static <T> T create(HttpRequest httpRequest) {
    Object httpRequestEntity = null;
    try {
        Class<T> httpRequestEntityCls = (Class<T>) Class.forName(HttpProcessor.PACKAGE_NAME + "." + HttpProcessor.CLASS_NAME);
        Constructor con = httpRequestEntityCls.getConstructor(HttpRequest.class);
        httpRequestEntity = con.newInstance(httpRequest);
    } catch (ClassNotFoundException e) {
        e.printStackTrace();
    } catch (InstantiationException e) {
        e.printStackTrace();
    } catch (IllegalAccessException e) {
        e.printStackTrace();
    } catch (NoSuchMethodException e) {
        e.printStackTrace();
    } catch (InvocationTargetException e) {
        e.printStackTrace();
    }
    return (T) httpRequestEntity;
}

就实现一个对象的构造都需要写一长串代码,特别是异常处理非常污染视觉

发现有一个第三方库:jOOR,通过链式DSL接口,简化了反射过程,

比如:

@Test
void should_get_World() {
    String result = Reflect.onClass("java.lang.String") // 类似Class.forName()
        .create("Hello World")     // 调用构造器
        .call("substring", 6)      // 调用方法
        .call("toString")          // 调用方法
        .get();                    // 获取最终包装类

    assertThat(result).isEqualTo("World");
}

再比如:原有java代码写法:

try {
  Method m1 = department.getClass().getMethod("getEmployees");
  Employee employees = (Employee[]) m1.invoke(department);
 
  for (Employee employee : employees) {
    Method m2 = employee.getClass().getMethod("getAddress");
    Address address = (Address) m2.invoke(employee);
 
    Method m3 = address.getClass().getMethod("getStreet");
    Street street = (Street) m3.invoke(address);
 
    System.out.println(street);
  }
}
 
// There are many checked exceptions that you are likely to ignore anyway 
catch (Exception ignore) {
 
  // ... or maybe just wrap in your preferred runtime exception:
  throw new RuntimeException(e);
}

采用jOOR后的写法:

Employee[] employees = on(department).call("getEmployees").get();
 
for (Employee employee : employees) {
  Street street = on(employee).call("getAddress").call("getStreet").get();
  System.out.println(street);
}

已经非常的简洁了。

jOOR特点

  • 提供on()操作符对类名、Class、Object进行统一实例化为Reflect对象,后续所有的反射操作基于该Reflect对象。
  • 所有功能调用方式均被封装成返回Reflect对象的链式结构,在使用上使得代码更加简洁。
  • 对方法的签名匹配封装了更完善的匹配规则,包括精确匹配exactMethod()、近似匹配similarMethod()【对函数参数的近似匹配(int -> Integer)】和基类搜索等。
  • 调用私有方法的不需要显示调用setAccessible(),内部动态读取public标记自动适配。
  • 更加简洁的实现了对象构造函数的反射调用create()方法。
  • 函数的调用call()方法组合成了可以拼接在Reflect的对象后面的链式方法。
  • 额外增加了高级操作符as(),它实现了类的代理访问以及POJO对象的get/set/is方法实现。

常用API测试

测试类:

class Person {
    private String name;
    private int age;

    public Person() {
    }
    public Person(String name) {
        this.name = name;
    }
    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 "Person{" +
            "name='" + name + '\'' +
            ", age=" + age +
            '}';
    }
}

测试APIS

@Test
void test_joor_apis() {
    // 【创建类】
    Person person = Reflect.onClass(Person.class).create("steven").get();// 有参构造器
    //Person person = Reflect.onClass(Person.class).create().get(); // 无参构造器
    assertThat(person.toString()).isEqualTo("Person{name='steven', age=0}");

    // 【调用方法】
    Reflect.on(person).call("setName", "steven2");
    String name = Reflect.on(person).call("getName").toString();
    assertThat(name).isEqualTo("steven2");

    // 【设置变量的值】
    int age = Reflect.on(person).set("age", 18).get("age");
    assertThat(age).isEqualTo(18);

    // 【得到变量】
    name = Reflect.on(person).field("name").get();// 方式一
    assertThat(name).isEqualTo("steven2");
    name = Reflect.on(person).get("name");// 方式二
    assertThat(name).isEqualTo("steven2");
}

代理功能

jOOR代理是实现的静态代理功能,首先创建静态代理相关类

interface HelloWorld {
    void print();
}
class HelloWorldImpl implements HelloWorld {
    public void print() {
        System.out.println("Hello World");
    }
}
class StaticProxy implements HelloWorld {
    private HelloWorld helloWorld;

    public StaticProxy(HelloWorld helloWorld) {
        this.helloWorld = helloWorld;
    }
    public void print() {
        System.out.println("Before Hello World!");
        helloWorld.print();
        System.out.println("After Hello World!");
    }
}

使用方法区别:

传统方式:

@Test
void test_proxy_normal() {
    StaticProxy staticProxy = new StaticProxy(new HelloWorldImpl());

    staticProxy.print();
}

jOOR实现方式

@Test
void test_proxy_jOOR() {
    Reflect.onClass(StaticProxy.class)//反射调用StaticProxy
            .create(new HelloWorldImpl())//调用构造器
            .as(HelloWorld.class)//作为HelloWorld接口的代理
            .print();
}

此时要求代理类和业务类具有相同的方法,才能通过调用代理的方法,负责抛出ReflectException异常

org.joor.ReflectException: java.lang.NoSuchMethodException: No similar method print with params [] could be found on type class StaticProxy.

	at org.joor.Reflect.call(Reflect.java:585)
	at org.joor.Reflect$1.invoke(Reflect.java:756)

特殊情况

当业务类为map类型,此时会把POJO的getter和setter转换成map的put和get

 // [#14] Emulate POJO behaviour on wrapped map objects
catch (ReflectException e) {
    if (isMap) {
        Map<String, Object> map = (Map<String, Object>) object;
        int length = (args == null ? 0 : args.length);

        if (length == 0 && name.startsWith("get")) {
            return map.get(property(name.substring(3)));
        }
        else if (length == 0 && name.startsWith("is")) {
            return map.get(property(name.substring(2)));
        }
        else if (length == 1 && name.startsWith("set")) {
            map.put(property(name.substring(3)), args[0]);
            return null;
        }
    }

动态编译

jOOR提供了可选的依赖java.compiler 可以简化 javax.tools.JavaCompiler 编译代码,

如下所示:

@Test
void test_compile_on_runtime() {
    Supplier<String> supplier = Reflect.compile(
        "com.example.HelloWorld",
        "package com.example;\n" +
        "class HelloWorld implements java.util.function.Supplier<String> {\n" +
        "    public String get() {\n" +
        "        return \"Hello World!\";\n" +
        "    }\n" +
        "}\n").create().get();


    String result = supplier.get();

    assertThat(result).isEqualTo("Hello World!");
}

结论

通过以上案例可以看出,jOOR由于其链式编程的特性,对代码的简化和可扩展性要强Java自带反射库和其他第三方库(apache、hutool等),且其包含了一些高级应用,如代理等。

  • 简化了私有方法的反射调用
  • 简化了反射冗长的异常处理。
  • 简化了对Class、Method、Field、Constructor反射类的实例化,改为统一Reflect替换。
  • 链式调用方式更简洁明了

到此这篇关于Java效率提升神器jOOR的文章就介绍到这了,更多相关Java jOOR内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Java编程伪共享与缓存行填充

    Java编程伪共享与缓存行填充

    这篇文章主要介绍了Java编程伪共享与缓存行填充,下面文章Disruptor提到的CPU缓存话题,做了一些尝试和研究,如Disruptor所说,CPU有缓存伪共享的问题,并且通过缓存行填充能完美的解决这个问题,需要的朋友可以参考一下
    2021-09-09
  • java生成指定范围的随机日期

    java生成指定范围的随机日期

    这篇文章主要为大家详细介绍了java生成指定范围的随机日期,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-12-12
  • 使用javassist动态生成类的配置代码

    使用javassist动态生成类的配置代码

    Javassist它是一个用 Java 编辑字节码的类库,它使 Java 程序能够在运行时定义新类,并在 JVM 加载时修改类文件,本文给大家介绍使用javassist动态生成类的实例代码,感兴趣的朋友一起看看吧
    2022-09-09
  • 关于泛型擦除问题的解决--Mybatis查询类型转换

    关于泛型擦除问题的解决--Mybatis查询类型转换

    这篇文章主要介绍了关于泛型擦除问题的解决--Mybatis查询类型转换方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-08-08
  • Java实现拖拽列表项的排序功能

    Java实现拖拽列表项的排序功能

    这篇文章主要介绍了Java实现拖拽列表项的排序功能,非常不错,具有参考借鉴价值,需要的朋友可以参考下
    2017-02-02
  • JAVA生成pdf文件的实操教程

    JAVA生成pdf文件的实操教程

    PDF是可移植文档格式,是一种电子文件格式,具有许多其他电子文档格式无法相比的优点,下面这篇文章主要给大家介绍了关于JAVA生成pdf文件的相关资料,文中通过图文介绍的非常详细,需要的朋友可以参考下
    2022-11-11
  • Java等待唤醒机制线程通信原理解析

    Java等待唤醒机制线程通信原理解析

    这篇文章主要介绍了Java等待唤醒机制线程通信原理解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2019-12-12
  • Java多线程下解决数据安全问题

    Java多线程下解决数据安全问题

    这篇文章主要介绍了Java多线程下解决数据安全问题,本文使用代码进行讲解,可供大家学习参考
    2021-08-08
  • java并发数据包Exchanger线程间的数据交换器

    java并发数据包Exchanger线程间的数据交换器

    这篇文章主要为大家介绍了java并发数据包使用数据交换器Exchanger来进行线程之间的数据交换。有需要的朋友可以借鉴参考下,希望能够有所帮助
    2022-03-03
  • Java黑科技之通过Google Java Style 文件配置IDEA和Ecplise代码风格

    Java黑科技之通过Google Java Style 文件配置IDEA和Ecplise代码风格

    在日常开发中,多人团队协作开发一个项目是很常见的,特别是大公司,这就会涉及到多人在一个工程上开发代码.无规矩不成方圆,一个好的代码风格,更加有利于团队协作,减少代码冲突,提高代码可阅读性,美观性.本文就带着大家仔细研究这个黑科技 ,需要的朋友可以参考下
    2021-05-05

最新评论