How to get Started with TDD
Tuesday, November 17, 2009
By Miško Hevery
Best way to learn TDD is to have someone show you while pairing with you. Short of that, I have set up an eclipse project for you where you can give it a try:
Your mission is to make the calculator work using TDD. This is the simplest form of TDD where you don't have to mock classes or create complex interactions, so it should be a good start for beginners.
TDD means:
I have already done all of the work of separating the behavior from the UI, so that the code is testable and properly Dependency Injected, so you don't have to worry about running into testability issues.
I have started you off with first 'testItShouldInitializeToZero' test. Here are some ideas for next tests you may want to write.
I would love to see what you will come up with and what your thoughts are, after you get the whole calculator working. I would also encourage you to post interesting corner case tests here for others to incorporate. If you want to share your code with others, I would be happy to post your solutions.
Good luck!
PS: I know it is trivial example, but you need to start someplace.
Best way to learn TDD is to have someone show you while pairing with you. Short of that, I have set up an eclipse project for you where you can give it a try:
- hg clone https://bitbucket.org/misko/misko-hevery-blog/
- Open project blog/tdd/01_Calculator in Eclipse.
- It should be set up to run all tests every time you modify a file.
- You may have to change the path to java if you are not an Mac OS.
- Project -> Properties -> Builders -> Test -> Edit
- Change location to your java
- Right-click on Calculator.java -> Run As -> Java Application to run the calculator
Your mission is to make the calculator work using TDD. This is the simplest form of TDD where you don't have to mock classes or create complex interactions, so it should be a good start for beginners.
TDD means:
- write a simple test, and assert something interesting in it
- implement just enough to make that tests green (nothing more, or you will get ahead of your tests)
- then write another test, rinse, and repeat.
I have already done all of the work of separating the behavior from the UI, so that the code is testable and properly Dependency Injected, so you don't have to worry about running into testability issues.
- Calculator.java: This is the main method and it is where all of the wiring is happening.
- CalculatorView.java: This is a view and we don't usually bother unit testing it has cyclomatic complexity of one, hence there is no logic. It either works or does not. Views are usually good candidates for end-to-end testing, which is not part of this exercise.
- CalculatorModel.java: is just a PoJo which marshals data from the Controller to the View, not much to test here.
- CalculatorController.java: is where all of your if statements will reside, and we need good tests for it.
I have started you off with first 'testItShouldInitializeToZero' test. Here are some ideas for next tests you may want to write.
- testItShouldConcatinateNumberPresses
- testItShouldSupportDecimalPoint
- testItShouldIgnoreSecondDecimalPoint
- testItShouldAddTwoIntegers
I would love to see what you will come up with and what your thoughts are, after you get the whole calculator working. I would also encourage you to post interesting corner case tests here for others to incorporate. If you want to share your code with others, I would be happy to post your solutions.
Good luck!
PS: I know it is trivial example, but you need to start someplace.
Interesting approach.. I usually test extensively the model and not the controller but I guess user-request handling constitutes the big part of this application.
ReplyDeleteNice initiative. Your TDD description is however missing the most important step: refactor. Without refactoring after making your tests pass, your code will quickly become unmaintainable.
ReplyDeleteOK, I've already started doing it.
ReplyDeleteA few questions:
What should the tick button do?
Do we have to support more than two operands (e.g 168+152+23)?
I like the test autorunning feature.
I also have created a tutorial for learning TDD. It's a bit larger than yours (a Tetris game) and has much more pre-written tests before it's time to write your own tests. Checkout the "tutorial" branch from http://github.com/orfjackal/tdd-tetris-tutorial and follow the instructions in the README file.
ReplyDeleteThis far I've used that as an exercise on one course (http://www.cs.helsinki.fi/u/luontola/tdd-2009/) and it appeared to get the students started in TDD quite nicely. It would be nice to get some feedback from others.
Hey, I think you already know this, but the Test-First Challenge (http://xp123.com/xplor/xp0201/) can also help getting started on TDD.
ReplyDeleteBTW a nice blog post !
Hi Misko,
ReplyDeleteYou're becoming a star - We do a webcast on testing called “This week in testing” and this week, we talk about this post about starting TDD.
You’re welcome to watch and comment (http://learn.typemock.com/this-week-in-test/2009/11/18/episode-4-animals-that-talk-back.html) and if you like what you see, please tell the whole world.
Thanks,
Gil Zilberfeld
Typemock
Good idea, but it's a shame you didn't write proper build scripts.
ReplyDeleteSome people don't use Eclipse.
"I know it is trivial example, but you need to start someplace."
ReplyDeleteI think that it will be a very good start point to those that are learning TDD, but with a previous knowledgement of JUnit.
:)
For windows developers you also need to change the ":" to a ";" in the arguments for the "Tests" builder. The line should be:
ReplyDelete-cp .;${workspace_loc:/01_Calculator/lib}/junit.jar AllTests
I don't usually do TDD (I write tests after) and I found it hard to get started with this one. I got a bit into trouble with writing a bunch of tests, then realising I'd rather change the model and ending up with a huge slew of tests that were now nonsensical.
ReplyDeleteBut maybe it's due to my resistance to using a String model for a numeric value. :-)