ISBN: 978-0201616224 - Published On: 1999
Programming is a craft. It comes down to getting a computer to do what you want it to do. As a programmer, you are part listener, part advisor, part interpreter, and part dictator.
There are many people offering you help. Tool vendors out with their miracles their products. Gurus promise that their techniques guarantee results. Everyone claims that their programming language is the best, and every operating system is the answer to all conceivable ills.
None of this is true. There are no easy answers. There is no such thing as a best solution. There can only be systems that are more appropriate in a particular set of circumstances.
Theory and practice combine to make you strong.
The most basic characteristics all Pragmatic Programmers share basic enough to state as tips:
TIP 1: Care About Your Craft
TIP 2: Think! About Your Work
This isn’t a one-time audit of current practices—it’s an ongoing critical appraisal of every decision you make, every day, and on every development.
The construction of software should be an engineering discipline. However, this doesn’t preclude individual craftsmanship.
A tourist visiting England’s Eton College asked the gardener how he got the lawns so perfect. “That’s easy,” he replied, “You just brush off the dew every morning, mow them every other day, and roll them once a week.”
“Is that all?” asked the tourist.
“Absolutely,” replied the gardener. “Do that for 500 years and you’ll have a nice lawn, too.”
Being responsible, Pragmatic Programmers won’t sit idly by and watch their projects fall apart through neglect.
The greatest of all weaknesses is the fear of appearing weak. - J. B. Bossuet, Politics from Holy Writ, 1709
A Pragmatic Programmer takes charge of his or her own career, and isn’t afraid to admit ignorance or error. Despite thorough testing, good documentation, and solid automation, things go wrong. We can be proud of our abilities, but we must be honest about our shortcomings—our ignorance as well as our mistakes.
Don’t blame all the problems on a vendor, a language, management, or your coworkers. Any and all of these may play a role, but it is up to you to provide solutions, not excuses.
TIP 3: Provide Options, Don’t Make Lame Excuses
Entropy is a term from physics that refers to the amount of “disorder” in a system. Unfortunately, the laws of thermodynamics guarantee that the entropy in the universe tends toward a maximum. When disorder increases in software, programmers call it “software rot.”
Researchers in the field of crime and urban decay discovered a fascinating trigger mechanism, one that very quickly turns a clean, intact, inhabited building into a smashed and abandoned derelict. One broken window, left unrepaired for any substantial length of time, instills in the inhabitants of the building a sense of abandonment a sense that the powers that be don’t care about the building.
TIP 4 Don’t Live with Broken Windows
Don’t leave “broken windows” (bad designs, wrong decisions, or poor code) unrepaired. Fix each one as soon as it is discovered. If there is insufficient time to fix it properly, then board it up. Perhaps you can comment out the offending code, or display a “Not Implemented” message, or substitute dummy data instead.
If you find yourself on a team and a project where the code is pristinely beautiful—cleanly written, well designed, and elegant you will likely take extra special care not to mess it up.
You may be in a situation where you know exactly what needs doing and how to do it. The entire system just appears before your eyes you know it’s right. But ask permission to tackle the whole thing and you’ll be met with delays and blank stares. People will form committees, budgets will need approval, and things will get complicated. Everyone will guard their own resources. Sometimes this is called “start-up fatigue.”
Work out what you can reasonably ask for. Develop it well. Once you’ve got it, show people, and let them marvel. Then say “of course, it would be better if we added .” Pretend it’s not important. Sit back and wait for them to start asking you to add the functionality you originally wanted. People find it easier to join an ongoing success.
TIP 5 Be a Catalyst for Change
Most software disasters start out too small to notice, and most project overruns happen a day at a time. Systems drift from their specifications feature by feature, while patch after patch gets added to a piece of code until there’s nothing of the original left. It’s often the accumulation of small things that breaks morale and teams.
TIP 6 Remember the Big Picture
The real world just won’t let us produce much that’s truly perfect, particularly not bug-free software. Time, technology, and temperament all conspire against us. Discipline yourself to write software that’s good enough—good enough for your users, for future maintainers, for your own peace of mind.
The phrase “good enough” does not imply sloppy or poorly produced code. All systems must meet their users’ requirements to be successful. We are simply advocating that users be given an opportunity to participate in the process of deciding when what you’ve produced is good enough.
Normally you’re writing software for other people. How often do you ask them how good they want their software to be? Sometimes there’ll be no choice. If you’re working on pacemakers. However, if you’re working on a brand new product, you’ll have different constraints. The marketing people will have promises to keep, the eventual end users may have made plans based on a delivery schedule, and your company will certainly have cash-flow constraints. It would be unprofessional to ignore these users’ requirements simply to add new features to the program, or to polish up the code just one more time.
TIP 7 Make Quality a Requirements Issue
Programming is like painting. You start with a blank canvas and materials. You use science, art, and craft to determine what to do with them. You sketch out an overall shape, paint the underlying environment, then fill in the details. You constantly step back with a critical eye to view what you’ve done. Artists will tell you that all the hard work is ruined if you don’t know when to stop. If you add layer upon layer, detail over detail, the painting becomes lost in the paint.
Your knowledge becomes out of date as new techniques, languages, and environments are developed. Your knowledge is a they’re expiring asset. Managing a knowledge portfolio is very similar to managing a financial portfolio:
TIP 8 Invest Regularly in Your Knowledge Portfolio
You need to ensure that the knowledge in your portfolio is accurate and unswayed by either vendor or media hype. Just because a search engine lists a hit first doesn’t mean that it’s the best match.
TIP 9 Critically Analyze What You Read and Hear
Plan what you want to say. Write an outline. Then ask yourself, “Does this get across whatever I’m trying to say?” Refine it until it does.
This approach is not just applicable to writing documents. When you’re faced with an important meeting or a phone call with a major client, jot down the ideas you want to communicate, and plan a couple of strategies for getting them across.
We’ve all sat in meetings where a development geek glazes over the eyes of the vice president of marketing with a long monologue on the merits of some arcane technology. This isn’t communicating: it’s just talking, and it’s annoying.
# The WISDOM acrostic—understanding an audience
W-hat do you want them to learn?
What is their I-nterest in what you’ve got to say?
How S-ophisticated are they?
How much D-etail do they want?
Whom do you want to O-wn the information?
How can you M-otivate them to listen to you?
What
Interest
Sophisticated
Detail
Own
Motivate
It’s six o’clock on Friday afternoon, following a week when the auditors have been in, and your boss’s youngest is in the hospital. probably isn’t a good time to ask her for a memory upgrade.
Adjust the style of your delivery to suit your audience.
Your ideas are important. They deserve a good-looking vehicle to convey them to your audience. Any chef will tell you that you can slave in the kitchen for hours only to ruin your efforts with poor presentation.
Involve your readers with early drafts of your document. Get their feedback, and pick their brains.
There’s one technique that you must use if you want people to listen to you: listen to them.
Even if this is a situation where you have all the information, even if this is a formal meeting with you standing in front of suits—if you don’t listen to them, they won’t listen to you.
Always respond to e-mails and voice mails, even if the response is simply “I’ll get back to you later.” Keeping people informed makes them far more forgiving of the occasional slip, and makes them feel that you haven’t forgotten them.
TIP 10 It’s Both What You Say and the Way You Say It
Most people assume that maintenance begins when an application is released, that maintenance means fixing bugs and enhancing features. We think these people are wrong. Programmers are constantly in maintenance mode. Our understanding changes day by day.
We feel that the only way to develop software reliably, and to make our developments easier to understand and maintain, is to follow what we call the DRY principle:
EVERY PIECE OF KNOWLEDGE MUST HAVE A SINGLE, UNAMBIGUOUS, AUTHORITATIVE REPRESENTATION WITHIN A SYSTEM,
TIP 11 DRY — Don’t Repeat Yourself
TIP 12 Make It Easy to Reuse
“Orthogonality” is a term borrowed from geometry. Two lines are orthogonal if they meet at right angles, such as the axes on a graph.
In computing, the term has come to signify a kind of independence or decoupling. Two or more things are orthogonal if changes in one do not affect any of the others. In a well-designed system, the database code will be orthogonal to the user interface: you can change the interface without affecting the database, and swap databases without changing the interface.
Helicopter controls are decidedly not orthogonal. Your hands and feet are constantly moving, trying to balance all the interacting forces.
Nonorthogonal systems are inherently more complex to change and control.
TIP 13 Eliminate Effects Between Unrelated Things
You get two major benefits if you write orthogonal systems: increased productivity and reduced risk.
Most developers are familiar with the need to design orthogonal systems, although they may use words such as modular, component-based, and layered to describe the process. Systems should be composed of a set of cooperating modules, each of which implements functionality independent of the others.
There is an easy test for orthogonal design. Once you have your components mapped out, ask yourself: If I dramatically change the requirements behind a particular function, how many modules are affected? In an orthogonal system, the answer should be “one.” In reality, this is naive. Unless you are remarkably lucky, most real-world requirements changes will affect multiple functions in the system. However, if you analyze the change in terms of functions, each functional change should still ideally affect just one module.
Be careful to preserve the orthogonality of your system as you introduce third-party toolkits and libraries. Choose your technologies wisely. Ask yourself whether it imposes changes on your code that shouldn’t be there.
There are several techniques you can use to maintain orthogonality:
Engineers prefer simple, single solutions to problems. Management tends to agree with the engineers: single, easy answers fit nicely on spreadsheets and project plans. If only the real world would cooperate! There is always more than one way. If you go into a project hampered by the myopic notion that there is only one way to do it, you may be in for an unpleasant surprise.
Suppose you decide, early in the project, to use a relational database from vendor A. Much later, during performance testing, you discover that the database is simply too slow, but that the object database from vendor B is faster. With most conventional projects, you’d be out of luck. Most of the time, calls to third-party products are entangled throughout the code. But if you really abstracted the idea of a database out to the point where it simply provides persistence as a service then you have the flexibility to change horses in midstream.
TIP 14 There Are No Final Decisions
Ready, fire, aim…
Tracer bullets are loaded at intervals on the ammo belt alongside regular ammunition. When they’re fired, their phosphorus ignites and leaves a pyrotechnic trail from the gun to whatever they hit. If the tracers are hitting the target, then so are the regular bullets.
TIP 15 Use Tracer Bullets to Find the Target
To get the same effect in code, we’re looking for something that gets us from a requirement to some aspect of the final system quickly, visibly, and repeatably.
Tracer code is not disposable: you write it for keeps. It contains all the error checking, structuring, documentation, and self-checking that any piece of production code has. It simply is not fully functional. However, once you have achieved an end-to-end connection among the components of your system, you can check how close to the target you are, adjusting if necessary. Once you’re on target, adding functionality is easy.
The tracer code approach has many advantages:
You use the technique in situations where you’re not 100% certain of where you’re going. You shouldn’t be surprised if your first couple of attempts miss: the user says “that’s not what I meant,” or data you need isn’t available when you need it, or performance problems seem likely. Work out how to change what you’ve got to bring it nearer the target, and be thankful that you’ve used a lean development methodology.
With a prototype, you’re aiming to explore specific aspects of the final system.
With a true prototype, you will throw away whatever you lashed together when trying out the concept, and recode it properly using the lessons you’ve learned.
The tracer code approach addresses a different problem. You need to know how the application as a whole hangs together. You want to show your users how the interactions will work in practice, and you want to give your developers an architectural skeleton on which to hang code.
We build software prototypes to analyze and expose risk, and to offer chances for correction at a greatly reduced cost. Like the car makers, we can target a prototype to test one or more specific aspects of a project.
Prototypes are designed to answer just a few questions, so they are much cheaper and faster to develop than applications that go into production. The code can ignore unimportant details—unimportant to you at the moment, but probably very important to the user later on. If you are prototyping a GUI, for instance, you can get away with incorrect results or data.
TIP 16 Prototype to Learn
Before you embark on any code-based prototyping, make sure that everyone understands that you are writing disposable code.Prototypes can be deceptively attractive to people who don’t know that they are just prototypes. You must make it very clear that this code is disposable, incomplete, and unable to be completed.
Languages influence how you think about a problem, and how you think about communicating.
TIP 17 Program Close to the Problem Domain
TIP 18 Estimate to Avoid Surprises
One of the interesting things about estimating is that the units you use make a difference in the interpretation of the result. If you say that something will take about 130 working days, then people will be expecting it to come in pretty close. If you say “about six months” then they know to look for it any time between five and seven months from now.
Duration | Quote |
---|---|
1–15 days | days |
3–8 weeks | weeks |
8-30 weeks | months |
30+ weeks | think hard before giving an estimate |
We think it’s a great idea to record your estimates so you can see how close you were. If an overall estimate involved calculating sub estimates, keep track of these as well. Often you’ll find your estimates are pretty good—in fact, after a while, you’ll come to expect this.
The normal rules of estimating can break down in the face of the complexities and vagaries of a sizable application development.
TIP 19 Iterate the Schedule with the Code
This may not be popular with management, who typically want a single, hard-and-fast number before the project even starts. You’ll have to help them understand that the team, their productivity, and the environment will determine the schedule. By formalizing this, and refining the schedule as part of each iteration, you’ll be giving them the most accurate scheduling estimates you can.
You say “I’ll get back to you.”
Every craftsman starts his or her journey with a basic set of good quality tools. Each tool will have its own personality and quirks, and will need its own special handling. Tools amplify your talent. Expect to add to your toolbox regularly. Always be on the lookout for better ways of doing things.
Plain text is made up of printable characters in a form that can be read and understood directly by people. Plain text doesn’t mean that the text is unstructured; XML, SGML, and HTML are great examples of plain text that has a well-defined structure.
TIP 20 Keep Knowledge in Plain Text
Human-readable forms of data, and self-describing data, will outlive all other forms of data and the applications that created them. Period. Virtually every tool in the computing universe can operate on plain text.
Every woodworker needs a good, solid, reliable workbench, for a programmer manipulating files of text, that workbench is the command shell. From the shell prompt, you can invoke your full repertoire of tools, using pipes to combine them in ways never dreamt of by their original developers.
TIP 21 Use the Power of Command Shells
You need to be able to manipulate text as effortlessly as possible, because text is the basic raw material of programming.
We think it is better to know one editor very well, and use it for all editing tasks code, documentation, memos, system administration, and so on.
TIP 22 Use a Single Editor Well
TIP 23 Always Use Source Code Control
Always. Even if you are a single-person team on a one-week project. Even if it’s a “throw-away” prototype. Make sure that everything is under source code control—documentation, phone number lists, memos to vendors, makefiles, build and release procedures, that little shell script that burns the CD master—everything.
There is a tremendous hidden benefit in having an entire project under the umbrella of a source code control system: you can have product builds that are automatic and repeatable.
No one writes perfect software, so it’s a given that debugging will take up a major portion of your day.
Debugging itself is a sensitive, emotional subject for many developers. Instead of attacking it as a puzzle to be solved.
TIP 24 Fix the Problem, Not the Blame
You need to turn off many of the defenses you use each day to protect your ego, tune out any project pressures you may be under, and get yourself comfortable. Above all, remember the first rule of debugging: Don’t Panic!
TIP 25 Don’t Panic
Before you start to look at the bug, make sure that you are working on code that compiled cleanly—without warnings. You may need to interview the user who reported the bug in order to gather more data than you were initially given.
Artificial tests don’t exercise enough of an application. You must brutally test both boundary conditions and realistic end-user usage patterns. The best way to start fixing a bug is to make it reproducible. After all, if you can’t reproduce it, how will you know if it is ever fixed?
Debuggers generally focus on the state of the program now. Sometimes you need more—you need to watch the state of a program or a data structure over time.
Tracing statements are those little diagnostic messages you print to the screen or to a file that say things such as “got here” and “value of x = 2.” It’s a primitive technique compared with IDE-style debuggers, but it is peculiarly effective at diagnosing several classes of errors that debuggers can’t.
A very simple but particularly useful technique for finding the cause of a problem is simply to explain it to someone else. The other person should look over your shoulder at the screen, and nod his or her head constantly (like a rubber duck bobbing up and down in a bathtub). They do not need to say a word; the simple act of explaining, step by step, what the code is supposed to do often causes the problem to leap off the screen and announce itself.
TIP 26 “select” Isn’t Broken
When you find yourself surprised by a bug, you must reevaluate truths you hold dear.
TIP 27 Don’t Assume It - Prove IT
Pragmatic Programmers manipulate text the same way woodworkers shape wood. Text manipulation languages are to programming what routers. Here router means the tool that spins cutting blades very, very fast. They are noisy, messy, and somewhat brute force. Make mistakes with them, and entire pieces can be ruined
TIP 28 Learn a Text Manipulation Language
TIP 29 Write Code That Writes Code
Passive code generators save typing. They are basically parameterized templates, generating a given output from a set of inputs. Once the result is produced, it becomes a full-fledged source file in the project; it will be edited, compiled, and placed under source control just like any other file. Its origins will be forgotten.
With an active code generator, you can take a single representation of some piece of knowledge and convert it into all the forms your application needs. This is not duplication, because the derived forms are disposable, and are generated as needed by the code generator (hence the word active).
TIP 30 You Can’t Write Perfect Software
We are constantly interfacing with other people’s code code that might not live up to our high standards and dealing with inputs that may or may not be valid. So we are taught to code defensively.
Pragmatic Programmers take this a step further. They don’t trust themselves, either. Knowing that no one writes perfect code, including themselves, Pragmatic Programmers code in defenses against their own mistakes.
A contract defines your rights and responsibilities, as well as those of the other party. In addition, there is an agreement concerning repercussions if either party fails to abide by the contract.
Design by Contract (DBC, for short).
Every function and method in a software system does something. Before it starts that something, the routine may have some expectation of the state of the world, and it may be able to make a statement about the state of the world when it concludes. Meyer describes these expectations and claims as follows:
If all the routine’s preconditions are met by the caller, the routine shall guarantee that all postconditions and invariants will be true when it completes.
TIP 31 Design with Contracts
It’s easy to fall into the “it can’t happen” mentality. Most of us have written code that didn’t check that a file closed successfully, or that a trace statement got written as we expected. And all things being equal, it’s likely that we didn’t need to—the code in question wouldn’t fail under any normal conditions. But we’re coding defensively.
All errors give you information. You could convince yourself that the error can’t happen, and choose to ignore it. Instead, Pragmatic Programmers tell themselves that if there is an error, something very, very bad has happened.
TIP 32 Crash Early
Many times, crashing your program is the best thing you can do. The alternative may be to continue, writing corrupted data to some vital database or commanding the washing machine into its twentieth consecutive spin cycle. A dead program normally does a lot less damage than a crippled one.
It seems that there’s a mantra that every programmer must memorize early in his or her career. “THIS CAN NEVER HAPPEN…” “This code won’t be used 30 years from now, so two-digit dates are fine.” “This application will never be used abroad, so why internationalize it?” “count can’t be negative.” “This printf can’t fail.”
TIP 33 – If It Can’t Happen, Use Assertions to Ensure That It Won’t
Don’t use assertions in place of real error handling. Assertions check for things that should never happen.
There is a common misunderstanding about assertions, promulgated by the people who write compilers and language environments: “Assertions add some overhead to code. Because they check for things that should never happen, they’ll get triggered only by a bug in the code. Once the code has been tested and shipped, they are no longer needed, and should be turned off to make the code run faster. Assertions are a debugging facility.”
There are two patently wrong assumptions here.
Turning off assertions when you deliver a program to production is like crossing a high wire without a net because you once made it across in practice. There’s dramatic value, but it’s hard to get life insurance.
We believe that exceptions should rarely be used as part of a program’s normal flow; exceptions should be reserved for unexpected events.
Assume that an uncaught exception will terminate your program and ask yourself, “Will this code still run if I remove all the exception handlers?” If the answer is “no,” then maybe exceptions are being used in non-exceptional circumstances.
For example, if your code tries to open a file for reading and that file does not exist, should an exception be raised?
Our answer is, “It depends.”
TIP 34 – Use Exceptions for Exceptional Problems
Programs that use exceptions as part of their normal processing suffer from all the readability and maintainability problems of classic spaghetti code.
An error handler is a routine that is called when an error is detected. You can register a routine to handle a specific category of errors. There are times when you may want to use error handlers, either in-stead of or alongside exceptions.
We all manage resources whenever we code: memory, transactions, threads, files, timers—all kinds of things with limited availability.
TIP 35 – Finish What You Start
the routine or object that allocates a resource should be responsible for deallocating it.
The basic pattern for resource allocation can be extended for routines that need more than one resource at a time. There are just two more suggestions:
If you are programming in an object-oriented language, you may find it useful to encapsulate resources in classes. Each time you need a particular resource type, you instantiate an object of that class. When the object goes out of scope, or is reclaimed by the garbage collector, the object’s destructor then deallocates the wrapped resource.
If an exception is thrown, how do you guarantee that everything allocated prior to the exception is tidied up? The answer depends to some extent on the language.
In order to keep up with today’s near-frantic pace of change, we need to make every effort to write code that’s as loose—as flexible—as possible. In this chapter, we’ll tell you how to make reversible decisions, so your code can stay flexible and adaptable in the face of an uncertain world.
Organize your code into modules and limit the interaction between them.
What’s wrong with having modules that know about each other? Nothing in principle — we don’t need to be as paranoid as spies or dissidents. However, you do need to be careful about how many other modules you interact with and, more importantly, how you came to interact with them.
Traversing relationships between objects directly can quickly lead to a combinatorial explosion of dependency relationships. If n
objects all know about each other, then a change to just one object can result in the other n -1
objects needing changes
You can see symptoms of this phenomenon in a number of ways:
Systems with many unnecessary dependencies are very hard and expensive to maintain, and tend to be highly unstable.
TIP 36 – Minimize Coupling Between Modules
The Law of Demeter for functions states that any method of an object should call only methods belonging to:
class Demeter {
private:
A *a;
int func();
public:
//...
void example(B& b);
}
void Demeter::example(B& b) { //
C c;
int f = func(); // <= 1. itself
b.invert(); // <= 2. any parameters that were passed in to the method
a = new A();
a->setActive(); // <= 3. any objects it created
c.print(); // <= 4. any directly held component objects
}
Every time we have to go in and change the code to accommodate some change in business logic, or in the law, or in management’s personal tastes of the day, we run the risk of breaking the system—of introducing a new bug. So we say “out with the details!” Get them out of the code.
TIP 37 – Configure, Don’t Integrate
First, we want to make our systems highly configurable. Not just things such as screen colors, but deeply ingrained items such as the choice of algorithms, database products, middleware technology, and user-interface style. These items should be implemented as configuration options, not through integration or engineering. Use metadata to describe configuration options for an application: tuning parameters, user preferences, the installation directory, and so on.
We use the term in its broadest sense. Metadata is any data that describes the application—how it should run, what resources it should use, and so on. Typically, metadata is accessed and used at runtime, not at compile time.
TIP 38 – Put Abstractions in Code, Details in Metadata
There are several benefits to this approach:
Because business policy and rules are more likely to change than any other aspect of the project, it makes sense to maintain them in a very flexible format.
We are talking about the role of time as a design element of the software itself. There are two aspects of time that are important to us:
When people first sit down to design an architecture or write a program, things tend to be linear. That’s the way most people think do this and then always do that.
But thinking this way leads to temporal coupling: coupling in time.
We need to allow for concurrency and to think about decoupling any time or order dependencies
TIP 39 – Analyze Workflow to Improve Concurrency
One way to do this is to capture their description of workflow using a notation such as the UML activity diagram. You can use diagrams to maximize parallelism by identifying items that could be performed in parallel.
TIP 40 – Design Using Services
Instead of components, we have really created services—independent, concurrent objects behind well-defined, consistent interfaces.
With linear code, it’s easy to make assumptions that lead to sloppy programming. But concurrency forces you to think through things a bit more carefully—you’re not alone at the party anymore. Because things can now happen at the “same time,” you may suddenly see some time based dependencies.
To begin with, any global or static variables must be protected from concurrent access.
TIP 41 – Always Design for Concurrency
We don’t want to have three separate copies of the data. So we create a model—the data itself, with common operations to manipulate it. Then we can create separate views that display the data in different ways: as a spreadsheet, as a graph, or in a totals box. Each of these views may have its own controller. This is the key concept behind the Model-View-Controller (MVC) idiom: separating the model from both the GUI that represents it and the controls that manage the view.
The view and controller are tightly coupled, and in some implementations of MVC the view and controller are a single component.
TIP 42 – Separate Views from Models
Consider how detectives might use a blackboard to coordinate and solve a murder investigation. Each detective may make contributions to this potential murder mystery by adding facts, statements from witnesses, any forensic evidence that might arise, and soon. As the data accumulates, a detective might notice a connection and post that observation or speculation as well.
None of the detectives needs to know of the existence of any other detective—they watch the board for new information, and add their findings. The detectives may be trained in different disciplines, may have different levels of education and expertise, and may not even work in the same precinct. They share a desire to solve the case, but that’s all.
Different detectives may come and go during the course of the process, and may work different shifts. There are no restrictions on what may be placed on the blackboard. It may be pictures, sentences, physical evidence, and so on.
A blackboard system lets us decouple our objects from each other completely, providing a forum where knowledge consumers and producers can exchange data anonymously and asynchronously. As you might guess, it also cuts down on the amount of code we have to write.
TIP 43 – Use Blackboards to Coordinate Workflow
Pragmatic Programmers think critically about all code, including our own. We constantly see room for improvement in our programs and our designs.
Accidents of implementation are things that happen simply because that’s the way the code is currently written. You end up relying on undocumented error or boundary conditions. “It works now, better leave well enough alone. . . .”
It’s easy to be fooled by this line of thought. Why should you take the risk of messing with something that’s working?
Coincidences can mislead at all levels from generating requirements through to testing. At all levels, people operate with many assumptions in mind but these assumptions are rarely documented and are often in conflict between different developers. Assumptions that aren’t based on well-established facts are the bane of all projects.
TIP 44 – Don’t Program by Coincidence
The O() notation is a mathematical way of dealing with approximation.
Whenever you find yourself writing a simple loop, you know that you have an O(n) algorithm. If that loop contains an inner loop, then you’re looking at O(m x n). You should be asking yourself how largethese values can get. If the numbers are bounded, then you’ll know how long the code will take to run. If the numbers depend on external factors, then you might want to stop and consider the effect that large values may have on your running time or memory consumption.
TIP 45 – Estimate the Order of Your Algorithms
TIP 46 – Test Your Estimates
If it’s tricky getting accurate timings, use code profilers to count the number of times the different steps in your algorithm get executed, and plot these figures against the size of the input.
Code needs to evolve; it’s not a static thing.
Unfortunately, the most common metaphor for software development is building construction. Well, software doesn’t quite work that way. Rather than construction, software is more like gardening—it is more organic than concrete. You plant many things in a garden according to an initial plan and conditions. Some thrive, others are destined to end up as compost. You may move plantings relative to each other to take advantage of the inter- play of light and shadow, wind and rain. Overgrown plants get split or pruned, and colors that clash may get moved to more aesthetically pleasing locations. You pull weeds, and you fertilize plantings that are in need of some extra help. You constantly monitor the health of the garden, and make adjustments as needed.
TIP 47 – Refactor Early, Refactor Often
Martin Fowler offers the following simple tips on how to refactor without doing more harm than good:
A software unit test is code that exercises a module. Typically, the unit test will establish some kind of artificial environment, then invoke rou- tines in the module being tested. It then checks the results that are returned, either against known values or against the results from previous runs of the same test (regression testing).
TIP 48 – Design to Test
When you design a module, or even a single routine, you should design both its contract and the code to test that contract. By designing code to pass a test and fulfill its contract, you may well consider boundary conditions and other issues that wouldn’t occur to you otherwise. There’s no better way to fix errors than by avoiding them in the first place. In fact, by building the tests before you implement the code, you get to try out the interface before you commit to it.
Regardless of the technology you decide to use, test harnesses should include the following capabilities
All software you write will be tested—if not by you and your team, then by the eventual users—so you might as well plan on testing it thoroughly.
TIP 49 – Test Your Software, or Your Users Will
Tool makers and infrastructure vendors have come up with a magic bullet, the wizard. Wizards are great. Just click a single button, answer a couple of simple questions, and the wizard will automatically generate skeleton code for you.
But using a wizard designed by a guru does not automatically make Joe developer equally expert. Joe can feel pretty good—he’s just produced a mass of code and a pretty spiffy-looking program. He just adds in the specific application functionality and it’s ready to ship. But unless Joe actually understands the code that has been produced on his behalf, he’s fooling himself. He’s programming by coincidence.
Wizards are a one-way street—they cut the code for you, and then move on. If the code they produce isn’t quite right, or if circumstances change and you need to adapt the code, you’re on your own.
TIP 50 – Don’t Use Wizard Code You Don’t Understand
Requirements rarely lie on the surface. Normally, they’re buried deep beneath layers of assumptions, misconceptions, and politics.
TIP 51 – Don’t Gather Requirements—Dig for Them
A requirement is a statement of something that needs to be accomplished. Good requirements might include the following:
However, very few requirements are as clear-cut, and that’s what makes requirements analysis complex.
It’s important to discover the underlying reason why users do a particular thing, rather than just the way they currently do it. At the end of the day, your development has to solve their business problem, not just meet their stated requirements. Documenting the reasons behind requirements will give your team invaluable information when making daily implementation decisions.
TIP 52 – Work with a User to Think Like a User
A big danger in producing a requirements document is being too specific. Good requirements documents remain abstract. Where requirements are concerned, the simplest statement that accurately reflects the business need is best.
This doesn’t mean you can be vague—you must capture the underlying semantic invariants as requirements, and document the specific or current work practices as policy.
Requirements are not architecture. Requirements are not design, nor are they the user interface. Requirements are need.
TIP 53 – Abstractions Live Longer than Details
Many projects failures are blamed on an increase in scope—also known as feature bloat, creeping featurism, or requirements creep. (Scope Creep)
Unfortunately, not many projects seem to track requirements actively. This means that they have no way to report on changes of scope— who requested a feature, who approved it, total number of requests approved, and so on. By tracking requirements you can get a clearer picture that “just one more feature” is really the fifteenth new feature added this month.
Create and maintain a project glossary—one place that defines all the specific terms and vocabulary used in a project. All participants in the project, from end users to support staff, should use the glossary to ensure consistency.
TIP 54 – Use a Project Glossary
Some constraints are absolute; others are merely preconceived notions.Absolute constraints must be honored, however distasteful or stupid they may appear to be. On the other hand, some apparent constraints may not be real constraints at all.
For example, there’s that old bar trick where you take a brand new, unopened champagne bottle and bet that you can drink beer out of it. The trick is to turn the bottle upside down, and pour a small quantity of beer in the hollow in the bottom of the bottle. Many software problems can be just as sneaky.
It’s not whether you think inside the box or outside the box. The problem lies in finding the box—identifying the real constraints.
TIP 55 – Don’t Think Outside the Box
When faced with an intractable problem, enumerate all the possible avenues you have before you. Don’t dismiss anything, no matter how unusable or stupid it sounds. Now go through the list and explain why a certain path cannot be taken. Are you sure? Can you prove it?
You too need to listen to the voice that whispers “wait.” If you sit down to start typing and there’s some nagging doubt in your mind, heed it.
TIP 56 – Listen to Nagging Doubts—Start When You’re Ready
When you feel a nagging doubt, or experience some reluctance when faced with a task, heed it. You may not be able to put your finger on exactly what’s wrong, but give it time and your doubts will probably crystallize into something more solid, something you can address. Software development is still not a science. Let your instincts contribute to your performance.
Choose an area that you feel will be difficult and begin producing some kind of proof of concept. One of two things will typically happen. Shortly after starting, you may feel that you’re wasting your time. Give up on the prototype, and hack into the real development.
On the other hand, as the prototype progresses you may have one of those moments of revelation when you suddenly realize that some basic premise was wrong. Not only that, but you’ll see clearly how you can put it right. You’ll feel comfortable abandoning the prototype and launching into the project proper. Your instincts were right, and you’ve just saved yourself and your team a considerable amount of wasted effort.
When you make the decision to prototype as a way of investigating your unease, be sure to remember why you’re doing it. The last thing you want is to find yourself several weeks into serious development before remembering that you started out writing a prototype.
The specification is also an agreement with the user a codification of their needs and an implicit contract that the final system will be in line with that requirement.
The problem is that many designers find it difficult to stop. They feel that unless every little detail is pinned down in excruciating detail they haven’t earned their daily dollar.
This is a mistake for several reasons.
TIP 57 – Some Things Are Better Done than Described
A design that leaves the coder no room for interpretation robs the programming effort of any skill and art. Some would say this is for the best, but they’re wrong. Often, it is only during coding that certain options become apparent.
As a Pragmatic Programmer, you should tend to view requirements gathering, design, and implementation as different facets of the same mprocess—the delivery of a quality system.
Just to be clear, we are not against generating specifications. Indeed, we recognize that there are times where incredibly detailed specifications are demanded—for contractual reasons, because of the environment where you work, or because of the nature of the product you are developing. Just be aware that you reach a point of diminishing, or even negative, returns as the specifications get more and more detailed. Also be careful about building specifications layered on top of specifications, without any supporting implementation or prototyping; it’s all too easy to specify something that can’t be built.
Detailed specifications are clearly appropriate for life-critical systems. We feel they should also be produced for interfaces and libraries used by others. When your entire output is seen as a set of routine calls, you’d better make sure those calls are well specified.
Computing has never been short of methods intended to make programming more like engineering. Each method gathers its disciples, and each enjoys a period of popularity. Then each is replaced by the next.
Some developers, adrift in a sea of sinking projects, keep clinging to the latest fad. At the end of the day, though, it doesn’t matter how good the flot is, the developers are still aimlessly adrift.
We like (some) formal techniques and methods. But we believe that blindly adopting any technique without putting it into the context of your development practices and capabilities is a recipe for disappointment.
TIP 58 – Don’t Be a Slave to Formal Methods
Formal methods have some serious shortcomings.
Absolutely. But always remember that formal development methods are just one more tool in the toolbox. If, after careful analysis, you feel you need to use a formal method, then embrace it—but remember who is in charge. Never become a slave to a methodology.
Look at methodologies critically, then extract the best from each and meld them into a set of working practices that gets better each month. This is crucial. You should work constantly to refine and improve your processes.
TIP 59 – Expensive Tools Do Not Produce Better Designs
As soon as you have more than one person working on a project, you need to establish some ground rules and delegate parts of the project accordingly. The single most important factor in making project-level activities work consistently and reliably is to automate your procedures.
No Broken Windows - Teams as a whole should not tolerate broken windows—those small imperfections that no one fixes.
Boiled Frogs - It’s easy for teams to get boiled. People assume that someone else is handling an issue. Even the best-intentioned teams can be oblivious to significant changes in their projects. Fight this. Make sure everyone actively monitors for changes.
Communicate - Great project teams have a distinct personality. People look forward to meetings with them, because it will be well-prepared.The documentation they produce is crisp, accurate, and consistent. The team speaks with one voice.
Don’t Repeat Yourself - This duplication leads to wasted effort, and can result in a maintenance nightmare. Some teams appoint a member as the project librarian, responsible for coordinating documentation and code repositories. Other team members can use this person as the first port of call when they’re looking for something.
Orthogonality - It is a mistake to think that the activities of a project—analysis, design, coding, and testing—can happen in isolation. They can’t. These are different views of the same problem, and artificially separating them can cause a boatload of trouble. We favor splitting teams functionally. Divide your people into small teams, each responsible for a particular functional aspect of the final system.
TIP 60 – Organize Around Functionality, Not Job Functions
Automation - A great way to ensure both consistency and accuracy is to automate everything the team does. To ensure that things get automated, appoint one or more team members as tool builders to construct and deploy the tools that automate the project drudgery. Have them produce makefiles, shell scripts, editor templates, utility programs, and the like.
TIP 61 – Don’t Use Manual Procedures
People just aren’t as repeatable as computers are. Nor should we expect them to be. A shell script or batch file will execute the same instructions, in the same order, time after time. It can be put under source control, so you can examine changes to the procedure over time as well (“but it used to work. . . ”).
Because memory is the second thing you lose as you age, we don’t want to rely on it too heavily. We can run scripts to perform procedures for us automatically, based on the content of source code and documents. Our goal is to maintain an automatic, unattended, content- driven workflow.
TIP 62 – Test Early. Test Often. Test Automatically.
We want to start testing as soon as we have code. Those tiny minnows have a nasty habit of becoming giant, man-eating sharks pretty fast, and catching a shark is quite a bit harder.
TIP 63 – Coding Ain’t Done ’Til All the Tests Run
Unit testing - A unit test is code that exercises a module.
Integration testing - shows that the major subsystems that make up the project work and play well with each other.
Validation and verification - As soon as you have an executable user interface or prototype, you need to answer an all-important question: the users told you what they wanted, but is it what they need? This, too, needs to be tested.
Resource exhaustion, errors, and recovery - In the real world, your programs don’t have limitless resources; they run out of things.
Performance testing - Performance testing, stress testing, or testing under load may be an important aspect of the project as well. Ask yourself if the software meets the performance requirements under real-world conditions—with the expected number of users, or connections, or transactions per second. Is it scalable?
Usability testing - It is performed with real users, under real environmental conditions. Look at usability in terms of human factors. Not only do we want our own tools to fit our hands, but we want the tools we create for users to fit their hands as well.
Regression testing - A regression test compares the output of the current test with previous (or known) values. We can ensure that bugs we fixed today didn’t break things that were working yesterday.
Test data - Where do we get the data to run all these tests? There are only two kinds of data: real-world data and synthetic data. We actually need to use both, because the different natures of these kinds of data will expose different bugs in our software. You need a lot of data, possibly more than any real-world sample could provide. You need data to stress the boundary conditions. You need data that exhibits certain statistical properties.
Exercising GUI systems - Testing GUI-intensive systems often requires specialized testing tools. These tools may be based on a simple event capture/playback model, or they may require specially written scripts to drive the GUI.
Testing the tests - Because we can’t write perfect software, we can’t write perfect tests. After you have written a test to detect a particular bug, cause the bug deliberately and make sure the test complains. If you are really serious about testing, you might want to appoint a project saboteur. The saboteur’s role is to take a separate copy of the source tree, introduce bugs on purpose, and verify that the tests will catch them.
TIP 64 – Use Saboteurs to Test Your Testing
Testing thoroughly - These coverage analysis tools watch your code during testing and keep track of which lines of code have been executed and which haven’t. These tools help give you a general feel for how comprehensive your testing is, but don’t expect to see 100% coverage.Even if you do happen to hit every line of code, that’s not the whole picture. What is important is the number of states that your program may have. States are not equivalent to lines of code
TIP 65 – Test State Coverage, Not Code Coverage
As soon as any code that is indented to be used in production exists.
Most testing should be done automatically. It’s important to note that by “automatically” we mean that the test results are interpreted automatically as well.
Usually, it isn’t a problem to run regressions on all of the individual unit tests and integration tests as often as needed. But some tests may not be easily run on a such a frequent basis. Stress tests, for instance, may require special setup or equipment, and some hand holding. These tests may be run less often—weekly or monthly, perhaps. But it is important that they be run on a regular, scheduled basis.
If a bug slips through the net of existing tests, you need to add a new test to trap it next time.
TIP 66 – Find Bugs Once
Once a human tester finds a bug, it should be the last time a human tester finds that bug. The automated tests should be modified to check for that particular bug
Pragmatic Programmers embrace documentation as an integral part of the overall development process. Writing documentation can be made easier by not duplicating effort or wasting time, and by keeping documentation close at hand—in the code itself, if possible.
TIP 67 – Treat English as Just Another Programming Language
TIP 68 – Build Documentation In, Don’t Bolt It On
Comments in Code - comments should discuss why something is done, its purpose and its goal. The code already shows how it is done, so commenting on this is redundant. Commenting gives you the perfect opportunity to document those elusive bits of a project that can’t be documented anywhere else: engineering trade-offs, why decisions were made, what other alternatives were discarded, and so on.
In reality, the success of a project is measured by how well it meets the expectations of its users. A project that falls below their expectations is deemed a failure, no matter how good the deliverable is in absolute terms.
TIP 69 – Gently Exceed Your Users’ Expectations
Some consultants call this process “managing expectations”—actively controlling what users should hope to get from their systems. We think this is a somewhat elitist position. Our role is not to control the hopes of our users. Instead, we need to work with them to come to a common understanding of the development process and the final deliverable, along with those expectations they have not yet verbalized. If the team is communicating fluently with the outside world, this process is almost automatic; everyone should understand what’s expected and how it will be built.
There are some important techniques that can be used to facilitate this process. Of these, Tracer Bullets, page 48, and Prototypes and Post it Notes, page 53, are the most important.
TIP 70 – Sign Your Work
Craftsmen of an earlier age were proud to sign their work. You should be, too.
On some projects, the idea of code ownership can cause cooperation problems. People may become territorial, or unwilling to work on common foundation elements. You become prejudiced in favor of your code and against your coworkers.
That’s not what we want. You shouldn’t jealously defend your code against interlopers; by the same token, you should treat other people’s code with respect. The Golden Rule (“Do unto others as you would have them do unto you”) and a foundation of mutual respect among the developers is critical to make this tip work.
Anonymity, especially on large projects, can provide a breeding ground for sloppiness, mistakes, and bad code. It becomes too easy to see yourself as just a cog in the wheel, producing lame excuses in endless status reports instead of good code.
While code must be owned, it doesn’t have to be owned by an individual.
We want to see pride of ownership. “I wrote this, and I stand behind my work.” Your signature should come to be recognized as an indicator of quality. People should see your name on a piece of code and expect it to be solid, well written, tested, and documented. A really professional job. Written by a real professional.
A Pragmatic Programmer.