I generally oppose extension since it creates a very strong connection between classes, which is easy to accidentally break.
However, I finally thought I’d found a reasonable case for it – I want to optionally use a compressed version of a file type in an existing system. The compressed version would be almost as quick as the uncompressed, and would have exactly the same methods available (i.e. read and write) – the only difference would be in the representation on disk. Therefore, I had the compressed version extend the uncompressed version so that either kind of file could be used, just by optionally insantiating the other type.
public class CompressedSpecialFile extends SpecialFile(){ ... }
if (useCompression){
SpecialFile = new CompressedSpecialFile();
} else {
SpecialFile = new SpecialFile();
}
However, at a later point in the program, we use reflection:
Object[] values = new Object[]{SpecialFile sf, Integer param1, String param2, ...}
Class myclass = Class.forName(algorithmName);
Class[] classes = // created by calling .getClass on each object in values
constructor = myclass.getConstructor(classes);
Algorithm = (Algorithm) constructor.newInstance(values)
Which all worked fine, but now the myclass.getConstructor class throws a NoSuchMethodException since the run-time type of the SpecialFile is CompressedSpecialFile.
However, I thought that was how extension is supposed to work – since CompressedSpecialFile extends SpecialFile, any parameter accepting a SpecialFile should accept a CompressedSpecialFile. Is this an error in Java’s reflection, or a failure of my understanding?
Hmm, the response to this bug report seems to indicate that this is intentional.
https://bugs.java.com/bugdatabase/view_bug?bug_id=4301875
That bug report was closed as a duplicate of the following one, which provides a bit more implementation detail:
https://bugs.java.com/bugdatabase/view_bug;jsessionid=1b08c721077da9fffffffff1e9a6465911b4e?bug_id=4287725
You can keep digging into those referenced bugs and the actual links I provided (where there’s discussion as well as possible workarounds) but I think that gets at the reasoning (though why a new method reflecting java’s oop in reflection as well has not yet been implemented, I don’t know).
In terms of workarounds, I suppose that for the one-level-deep version of inheritance, you can just call getSuperclass() on each class whose name is that of the extending class, but that’s extremely inelegant and tied to you using it only on your classes implementing in the prescribed manner. Very kludgy. I’ll try and look for another option though.