Definitely creating DRY code helps to some extent with maintainability, but it also often makes code harder to read, and tighter, which may hamper maintainability.
These day's I'm often seeing code that's not DRY specifically regarding loops , or loops of loops.
And almost always (I just can't seem to help myself) I'm pulling those code blocks into methods then passing an interface in to do the work. I don't think this neccisarily makes things any clearer, but then I don't seem to be able to help myself from pulling these loops apart.
http://blog.jayfields.com/2007/06/testing-inline-setup.html
Is advocating ignoring DRY for tests, and it can't help but make me wonder when else you shouldn't ignore DRY.
"Another question that always comes up: What about writing DRY code? For me, the value behind DRY has never been typing something once, but that when I want to make a change I only have to make it in one place. Since I'm advocating the duplication in a test file, you still only need to make the change in one place. You may need to do a find and replace instead of changing one helper method, but the other people who are stuck maintaining your tests will be very happy that you did."
for example I took some code that was generating matlab strings that looked like so:
I pulled the common loop out into a Code Generator Class that took a closure. It ended up okay. But I'm looking at the result. Which looked like:createStringForStuff(List<stuff> stuffs) { StringBuffer output = new StringBuffer("StuffName"); output.append(" = ["); for (Stuff stuff : stuffs) { String text = stuff.isItReallyStuff() ? "1" : "0"; output.append(" " + text); } output.append("]"); return output.toString(); } createStringForThing(List<thing> things) { StringBuffer output = new StringBuffer("ThingName"); output.append(" = ["); for (Thing thing : things) { String text = thing.isItReallyAThing() ? "2" : "-1"; output.append(" " + text); } output.append("]"); return output.toString(); } etc.
So yes it's less code. but reading that doesn't really give me any insight into what the code is doing. Where as before it wasn't DRY, but then again it also wasn't a lot of codecreateStringForStuff(List<stuff> stuffs) { MatlabCodeGenerator generator = new MatlabCodeGenerator("StuffName"); String result = generator.getCode(stuffs, new MatlabClosure()<stuff> { @Override public String get(Stuff stuff) { return stuff.isItReallyStuff() ? "1" : "0"; } }); return result; }