我的计划是编写一个基于注释的缓存框架,用于缓存方法的返回值。当第一次使用特定参数调用方法时,缓存应该存储方法的返回值。
当使用相同的参数第二次调用相同的方法时,该方法应该从缓存中返回先前计算的结果,并且不再执行其代码。
我的注释如下所示:
@Cached(cacheProvider = HashMapCacheProvider.class)
public Product getProduct(String productId){
// Scraping the product from a website ...
return product;
}
目前我的小框架已经运行良好。我正在使用 Javassist 创建包含带注释方法的类的代理对象。
为了创建一个新的缓存对象,我使用以下代码:
public static <T> T newCachedInstance(Class<T> clazz)
throws InstantiationException, IllegalAccessException {
ProxyFactory factory = new ProxyFactory();
factory.setSuperclass(clazz);
factory.setFilter(new MethodFilter() {
public boolean isHandled(Method m) {
// ignore finalize()
return !m.getName().equals("finalize");
}
});
Class<T> c = factory.createClass();
T proxy = c.newInstance();
((ProxyObject) proxy).setHandler(new CachedMethodHandler());
return proxy;
}
问题是,我只能通过此方法创建新的缓存对象,而不是通过其类的构造函数。
因此,我正在寻找一种解决方案,将现有的对象附加到缓存机制。
这是我的问题:
是否可以将代理附加到现有对象?
根据我的理解,如果不更新对此对象的所有现有引用,这应该是不可能的。
我的另一种方法是使用字节码操作来操作带注释的方法的代码,如下所示:
public class Hello {
public void say() {
System.out.println("Hello");
}
}
public class Test {
public static void main(String[] args) throws Exception {
ClassPool cp = ClassPool.getDefault();
CtClass cc = cp.get("Hello");
CtMethod m = cc.getDeclaredMethod("say");
m.insertBefore("{ System.out.println(\"Hello.say():\"); }");
Class c = cc.toClass();
Hello h = (Hello)c.newInstance();
h.say();
}
}
你还有其他想法吗?操作现有对象的方法的最佳实践是什么?