Java效率提升神器jOOR
作者:? 发布时间:2022-06-07 20:46:16
前言
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替换。
链式调用方式更简洁明了
来源:https://juejin.cn/post/7112462881651687431


猜你喜欢
- 本文实例为大家分享了Android Studio实现弹窗设置的具体代码,供大家参考,具体内容如下弹窗能很好的显示当前处理事情的状态,那么这里
- Android页面嵌套了一个h5,H5页面内部有用户登陆页面,发现h5页面的登陆功能无法使用,一直登陆失败。和web那边商量一会,发现js写
- 1. 插入排序步骤:1.从第一个元素开始,该元素可以认为已经被排序2.取下一个元素tem,从已排序的元素序列从后往前扫描3.如果该元素大于t
- 上篇并发编程之Java内存模型volatile的内存语义介绍了volatile的内存语义,本文讲述的是final的内存语义,相比之下,fin
- 1、功能需求本实例将通过c# winform实现简单的分页功能,需要的基础知识有SQL语句,c#语言基础以及c# winform的一些简单知
- 今天把Android Studio 2.3 更新为了3.0 遇到一个蛋疼的问题如图:格式化完代码后发现不会自动换行了,看着真心不爽。后来发现
- 效果图白话分析:多线程:肯定是多个线程咯断点:线程停止下载的位置续传:线程从停止下载的位置上继续下载,直到完成任务为止。核心分析:断点:当前
- 对于随机数,大家都知道,计算机不 可能产生完全随机的数字,所谓的随机数发生器都是通过一定的算法对事先选定的随机种子做复杂的运算,用产生的结果
- 本文通过实例详细阐述了C++关于智能指针的概念及用法,有助于读者加深对智能指针的理解。详情如下:一、简介由于 C++ 语言没有自动内存回收机
- 这几年都在搞前后端分离、RESTful风格,我们项目中也在这样用。前几天有人遇到了解析JSON格式的请求数据的问题,然后说了一下解析的方式,
- 本文实例讲述了C#中static静态变量的用法。分享给大家供大家参考。具体如下:使用 static 修饰符声明属于类型本身而不是属于特定对象
- 开篇Mybatis有个实用的功能就是逆向工程,能根据表结构反向生成实体类,这样能避免手工生成出错。市面上的教程大多都很老了,大部分都是针对m
- 一、材料准备1、jdk1.8的安装包2、maven安装包3、idea工具二、配置jdk1.8环境变量1.jdk下载jdk下载网址(点击此链接
- 本文实例为大家分享了Java实现简单贪吃蛇游戏的具体代码,供大家参考,具体内容如下贪吃蛇小游戏制作方法首先需要的准备有:1、掌握Java基础
- 自定义starterSpringBoot中的starter是一种非常重要的机制,能够抛弃以前繁杂的配置,将其统一集成进 starter,应用
- 在android中的webview中,可以对文本内容进行对齐,具体方法如下 public class MainActivity
- 环境:SpringBoot 2.0.4.RELEASE需求:很多Controller方法,刚进来要先获取当前登录用户的信息,以便做后续的用户
- 对象嵌套关联查询一对多List集合查询mybatis嵌套关联查询如下由于我的是一对集合查询,所以我有两个类。@Data@TableName(
- 本文实例为大家分享了Android屏幕适配工具类的具体代码,供大家参考,具体内容如下DimenToolgithub地址Android 屏幕适
- java8 Stream大数据量List分批处理//按每3个一组分割private static final Integer MAX_NUM