一些库经常会有传入一对键值对的方法,例如:
//es的QueryBuilder
public static TermQueryBuilder termQuery(String name, Object value);
//spring的ModelAndView
public ModelAndView addObject(String attributeName, Object attributeValue);
// and so on
但是呢,我们使用的参数大概率是某个dto的所有字段。
@Data
public class Model {
private String name;
private int age;
}
优雅一点的库会提供传入对象,然后内部直接把所有字段转成键值对的方法。
//spring的ModelAndView
public ModelAndView addObject(Object attributeValue);
不那么优雅的库,我们可以通过工具类帮它优雅。
//一行也能搞定
BeanMap.create(model).forEach(TermQueryBuilder::termQuery);
还有一种情况,我们只要dto上部分字段做参数,这样就难以避免出现魔法值。
termQuery("name",model.getName());
幸运的是,利用jdk8的特性还是可以优雅地解决问题。
Getter.use(model::getName,TermQueryBuilder::termQuery);
//实现
interface Getter extends Serializable {
String GETTER_PREFIX = "get";
Object get();
default SerializedLambda getSerializedLambda() throws Exception {
Method write = this.getClass().getDeclaredMethod("writeReplace");
write.setAccessible(true);
return (SerializedLambda) write.invoke(this);
}
default String getPropName() {
try {
String implMethodName = getSerializedLambda().getImplMethodName();
if (implMethodName.startsWith(GETTER_PREFIX)) {
implMethodName = implMethodName.replace(GETTER_PREFIX, "");
}
return implMethodName.substring(0, 1).toLowerCase() + implMethodName.substring(1);
} catch (Exception e) {
throw new RuntimeException(e);
}
}
static void use(Getter getter, BiConsumer<String, Object> consumer) {
consumer.accept(getter.getPropName(), getter.get());
}
}