小编典典
没有可靠的方法来检测类是否不可变。这是因为可以通过多种方式更改类的属性,而您无法通过反射来检测所有这些属性。
达到此目的的唯一方法是:
仅允许不可变类型的最终属性(您知道的原始类型和类是不可变的),
要求课程本身是最终的
要求它们从您提供的基类继承(保证是不可变的)
然后,您可以使用以下代码检查您拥有的对象是否不可变:
static boolean isImmutable(Object obj) {
Class> objClass = obj.getClass();
// Class of the object must be a direct child class of the required class
Class> superClass = objClass.getSuperclass();
if (!Immutable.class.equals(superClass)) {
return false;
}
// Class must be final
if (!Modifier.isFinal(objClass.getModifiers())) {
return false;
}
// Check all fields defined in the class for type and if they are final
Field[] objFields = objClass.getDeclaredFields();
for (int i = 0; i < objFields.length; i++) {
if (!Modifier.isFinal(objFields[i].getModifiers())
|| !isValidFieldType(objFields[i].getType())) {
return false;
}
}
// Lets hope we didn't forget something
return true;
}
static boolean isValidFieldType(Class> type) {
// Check for all allowed property types...
return type.isPrimitive() || String.class.equals(type);
}
更新:
如注释中所建议,它可以扩展为在超类上递归,而不是检查某个类。还建议在isValidFieldType方法中递归使用isImmutable。这可能可行,并且我也做了一些测试。但是,这并非微不足道。您不能只通过调用isImmutable来检查所有字段类型,因为String已经通过了此测试(它的字段hash不是final!)。同样,您很容易陷入无尽的递归,导致
StackOverflowErrors ;)其他问题可能是由泛型引起的,在泛型中还必须检查其类型的不可变性。
我认为通过一些工作,这些潜在问题可能会得到解决。但是,然后,您必须首先问自己是否真的值得(也是明智的选择)。
2020-09-16