Java效率提升神器jOOR
前言
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;
}就實現(xiàn)一個對象的構(gòu)造都需要寫一長串代碼,特別是異常處理非常污染視覺。
發(fā)現(xiàn)有一個第三方庫:jOOR,通過鏈式DSL接口,簡化了反射過程,
比如:
@Test
void should_get_World() {
String result = Reflect.onClass("java.lang.String") // 類似Class.forName()
.create("Hello World") // 調(diào)用構(gòu)造器
.call("substring", 6) // 調(diào)用方法
.call("toString") // 調(diào)用方法
.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);
}已經(jīng)非常的簡潔了。
jOOR特點
- 提供on()操作符對類名、Class、Object進行統(tǒng)一實例化為Reflect對象,后續(xù)所有的反射操作基于該Reflect對象。
- 所有功能調(diào)用方式均被封裝成返回Reflect對象的鏈式結(jié)構(gòu),在使用上使得代碼更加簡潔。
- 對方法的簽名匹配封裝了更完善的匹配規(guī)則,包括精確匹配exactMethod()、近似匹配similarMethod()【對函數(shù)參數(shù)的近似匹配(int -> Integer)】和基類搜索等。
- 調(diào)用私有方法的不需要顯示調(diào)用setAccessible(),內(nèi)部動態(tài)讀取public標記自動適配。
- 更加簡潔的實現(xiàn)了對象構(gòu)造函數(shù)的反射調(diào)用create()方法。
- 函數(shù)的調(diào)用call()方法組合成了可以拼接在Reflect的對象后面的鏈式方法。
- 額外增加了高級操作符as(),它實現(xiàn)了類的代理訪問以及POJO對象的get/set/is方法實現(xiàn)。
常用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() {
// 【創(chuàng)建類】
Person person = Reflect.onClass(Person.class).create("steven").get();// 有參構(gòu)造器
//Person person = Reflect.onClass(Person.class).create().get(); // 無參構(gòu)造器
assertThat(person.toString()).isEqualTo("Person{name='steven', age=0}");
// 【調(diào)用方法】
Reflect.on(person).call("setName", "steven2");
String name = Reflect.on(person).call("getName").toString();
assertThat(name).isEqualTo("steven2");
// 【設(shè)置變量的值】
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代理是實現(xiàn)的靜態(tài)代理功能,首先創(chuàng)建靜態(tài)代理相關(guān)類
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!");
}
}使用方法區(qū)別:
傳統(tǒng)方式:
@Test
void test_proxy_normal() {
StaticProxy staticProxy = new StaticProxy(new HelloWorldImpl());
staticProxy.print();
}
jOOR實現(xiàn)方式
@Test
void test_proxy_jOOR() {
Reflect.onClass(StaticProxy.class)//反射調(diào)用StaticProxy
.create(new HelloWorldImpl())//調(diào)用構(gòu)造器
.as(HelloWorld.class)//作為HelloWorld接口的代理
.print();
}此時要求代理類和業(yè)務(wù)類具有相同的方法,才能通過調(diào)用代理的方法,負責(zé)拋出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)
特殊情況
當業(yè)務(wù)類為map類型,此時會把POJO的getter和setter轉(zhuǎn)換成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;
}
}動態(tài)編譯
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!");
}結(jié)論
通過以上案例可以看出,jOOR由于其鏈式編程的特性,對代碼的簡化和可擴展性要強Java自帶反射庫和其他第三方庫(apache、hutool等),且其包含了一些高級應(yīng)用,如代理等。
- 簡化了私有方法的反射調(diào)用
- 簡化了反射冗長的異常處理。
- 簡化了對Class、Method、Field、Constructor反射類的實例化,改為統(tǒng)一Reflect替換。
- 鏈式調(diào)用方式更簡潔明了
到此這篇關(guān)于Java效率提升神器jOOR的文章就介紹到這了,更多相關(guān)Java jOOR內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Spring @Valid @Validated實現(xiàn)驗證
這篇文章主要介紹了Spring @Valid @Validated實現(xiàn)驗證,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-01-01
詳解SpringBoot如何實現(xiàn)統(tǒng)一后端返回格式
在前后端分離的項目中后端返回的格式一定要友好,不然會對前端的開發(fā)人員帶來很多的工作量。那么SpringBoot如何做到統(tǒng)一的后端返回格式呢?本文將為大家詳細講講2022-04-04
springboot3集成mybatis-plus報sqlSession異常的問題解決
springboot3已經(jīng)發(fā)布正式版,但是在集成mybatis-plus最新版3.5.2的時候發(fā)現(xiàn)提示異常,本文就來介紹一下報sqlSession異常的問題解決,具有一定的參考價值,感興趣的可以了解一下2024-02-02
spring boot創(chuàng)建和數(shù)據(jù)庫關(guān)聯(lián)模塊詳解
這篇文章主要給大家介紹了關(guān)于spring boot創(chuàng)建和數(shù)據(jù)庫關(guān)聯(lián)模塊的相關(guān)資料,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-10-10
Java 圖解Spring啟動時的后置處理器工作流程是怎樣的
spring的后置處理器有兩類,bean后置處理器,bf(BeanFactory)后置處理器。bean后置處理器作用于bean的生命周期,bf的后置處理器作用于bean工廠的生命周期2021-10-10

