首頁 > 軟體

Java效率提升神器jOOR

2022-07-07 14:01:22

前言

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內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!


IT145.com E-mail:sddin#qq.com