Java效率提升神器jOOR怎么使用
今天小编给大家分享一下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;}
就实现一个对象的构造都需要写一长串代码,特别是异常处理非常污染视觉。
发现有一个第三方库:jOOR,通过链式DSL接口,简化了反射过程,
比如:
@Testvoid 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
@Testvoid 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!"); }}
使用方法区别:
传统方式:
@Testvoid test_proxy_normal() { StaticProxy staticProxy = new StaticProxy(new HelloWorldImpl()); staticProxy.print();}
jOOR实现方式
@Testvoid 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 objectscatch (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
编译代码,
如下所示:
@Testvoid 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!");}
以上就是“Java效率提升神器jOOR怎么使用”这篇文章的所有内容,感谢各位的阅读!相信大家阅读完这篇文章都有很大的收获,小编每天都会为大家更新不同的知识,如果还想学习更多的知识,请关注编程网行业资讯频道。
免责声明:
① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。
② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341