是否有标准或简单的方法来冻结JavaBean?

Mic*_*uff 5 java javabeans

我发现自己处于使用大量JavaBeans的不幸位置.我说不幸,因为我不喜欢他们是可变的.因此,在我创建bean之后(我使用Jackson从JSON转换它们),我想在指定的时刻"冻结"bean,以确保它们不会在线下变异.

我没有能力编写每个bean的Mutable/Immutable版本,也没有自定义Jackson工厂的版本,因此我正在寻找一个类似于Collections.unmodifiableCollection(...);bean 的库或函数调用.

(是的,我知道如果原始的豆子被泄漏,它并没有真正冻结它.但是我会在创建后立即包装或冻结它们,而不再使用原件.)

有任何想法吗?

Mat*_*att 3

你是基于bean接口还是基于pojos?如果它们是基于接口的,您可以简单地编写一个实用方法:

public class ImmutableBean {
    public static <T> T makeImmutable(T t) { ... }
}
Run Code Online (Sandbox Code Playgroud)

然后,您可以创建一个特定于您的 bean 的 java.lang.Proxy 实例,或者查找以“set”开头的方法之类的模式,并抛出一些特定的异常。这是一个关于创建代理实例的简短教程: http ://www.javalobby.org/java/forums/t18631.html

如果它们是 pojo 的,您应该能够使用 javassist 或 cglib 之类的库来创建增强版本,其中您的 setter 会被存根以引发异常。

这是一个 javassist 示例:

 public static <T> T makeImmutable(final T t) throws NotFoundException, InstantiationException, IllegalAccessException {
    MethodHandler mh = new MethodHandler() {
        @Override
        public Object invoke(java.lang.Object self, java.lang.reflect.Method thisMethod, java.lang.reflect.Method proceed, java.lang.Object[] args) throws Throwable {
            if (thisMethod.getName().startsWith("set")) {
                throw new UnsupportedOperationException();
            }
            return proceed.invoke(t, args);
        }

    };

    ProxyFactory proxyFactory = new ProxyFactory();
    proxyFactory.setSuperclass(t.getClass());
    Class<T> clazz = (Class<T>) proxyFactory.createClass();
    T instance = clazz.newInstance();
    ((ProxyObject)instance).setHandler(mh);

    return instance;

}
Run Code Online (Sandbox Code Playgroud)

但是,这将要求您的原始类具有无参数构造函数,因此如果您没有无参数构造函数,您可能必须为您的类执行特定方法。