Sometimes you come back to an application and can only weep at the lack of foresight which would have made implementing the required new feature a doddle! But you've no time to weep, just as you had no time back in the day to make the application dance to any tune imaginable. You also know that Agile methodology tells you to code what is needed and no more and yet, you're always asking yourself, wouldn't it be nice if you could predict the future and just squeeze in that little bit of extra abstraction? In my current predicament, I can honestly say that there was no way I could have imagined replacing the source of my real-time feed with a completely different API. And given that the new API is only intended for an internal audience, with perhaps the odd external client demonstration, was it really an oversight? Afterall, making the receiving code abstract enough to cope with a simple replacement of the input feed took two weeks, time I simply didn't have on the original project. And yet, was there anything I could have done that would have made my life simpler now without comprimising the original deadline? In fact, I'm going to propose the following guidelines (admittedly Java centric) to help compromise between over-engineering and delivering what is needed now:
- Always code to interfaces. This is extremely obvious but, yet again, the last API I worked on hadn't and the resulting experience was painful to say the least. However, there's no need to create interfaces for objects intended as data containers.
- Use dependency injection. That is, keep coupling between classes light and ask yourself if creating an object is necessary in all cases. Inject it if at all possible.
- Use factories. If an object has to be created, consider using a Factory object to create it instead. This can be as simple as a method on an Enum specifying the various types and associates a creation mechanism with each one.
There's nothing new in these suggestion and I'll keep adding to it as and when.