If the Wheel Doesn't Work, Reinvent it
Chris Steinbeck has an interesting post on the CDK code review process that discusses a new VF implementation. In it, he notes:
I checked it out and looked at the code, which looked horrible because it was a 1:1 translation of a horrible looking C code. Clearly, a decent naming of the variables would greatly improve the code but I remember a statement that the translator himself could not make sense out of this, so the original author is to blame :-) . I do not get the impression that this problem can be rectified quickly. In fact, it took Mark a few days to debug this code by adding a rich collection of debug messages. I’m not sure that this is how it should be. The code is essentially unreadable.
For the unfamiliar, VF is a subgraph matching algorithm that has been shown to perform better than Ullmann for small graphs, and much better than Ullmann for large graphs.
Faced with essentially the same problem of implementing VF in Java for MX, I abandoned my early efforts to port the VFlib C++ implementation. The C++ implementation may make sense to a C++ programmer, but directly porting it to Java was judged as not being a good long-term move.
The problem was maintenance.
Although opinions on the subject vary, maintainable Java code to me has a few easily-identifiable characteristics. Among them are:
- Descriptive variable and method names.
- Limited use of deep nesting (> 3 levels) within methods.
- Stateful objects.
- Use of collections over primitives.
- Few methods over ten lines long.
The VF Java port that I created failed on just about every count - and failed consistently.
It turned out that a short description of the VF algorithm was remarkably clear, lending itself well to a Java-centric, object-oriented implementation that was successfully integrated into MX.
As a bonus, because test-driven development was used from the start for the MX implementation of VF, not only is the code maintainable, but it can be refactored and recasted with higher confidence due to the tests that are now present. This was used to great effect during a recent large-scale refactoring of the MX code to support arbitrary Query Atoms.
Would you consider bolting a bicycle wheel onto your new Kawasaki? Of course not. Why do the same with your software?
Reuse whenever it's consistent with your goals. When it's not, then reinvent.