| 1: Static Checking |                  TypesStatic Checking vs Dynamic CheckingArrays and CollectionsIteratingMethodsMutating Variables vs Reassigning VariablesDocumenting Assumptions | 
                      | 2: Basic Java |                  Snapshot DiagramsJava CollectionsJava API Documentation | 
                      | 3: Testing |                  ValidationTest-first ProgrammingChoosing Test Cases by PartitioningBlackbox and Whitebox TestingDocumenting Testing StrategiesCoverageUnit Testing and StubsAutomated Testing and Regression Testing | 
                      | 4: Code Review |                  Don't Repeat YourselfComments Where NeededFail FastAvoid Magic NumbersOne Purpose for Each VariableUse Good NamesUse Whitespace to Help the ReaderDon't Use Global VariablesMethods Should Return Results, Not Print Them | 
                      | 5: Version Control |                  Inventing Version ControlGit: Copy, Commit, Pull, Push, Merge | 
                      | 6: Specifications |                  Why Specifications?Behavioral EquivalenceSpecification StructureNull ReferencesWhat a Specification May Talk AboutTesting and SpecificationsSpecifications for Mutating MethodsExceptions for Signaling BugsExceptions for Special ResultsChecked and Unchecked ExceptionsThrowable HierarchyException Design ConsiderationsAbuse of Exceptions | 
                      | 7: Designing Specifications |                  Deterministic vs Undertermined SpecsDeclarative vs Operational SpecsStronger vs Weaker SpecsDiagramming SpecificationsDesigning Good SpecificationsPrecondition or Postcondition?About Access ControlAbout Static vs Instance Methods | 
                      | 8: Avoiding Debugging |                  First Defense: Making Bugs ImpossibleSecond Defense: Localizing BugsAssertionsWhat to AssertWhat Not to AssertIncremental DevelopmentModularity and Encapsulation | 
                      | 9: Mutability and Immutability |                  MutabilityRisks of MutationAliasing is What Makes Mutation RiskySpecifications for Mutating MethodsIterating Over Arrays and ListsMutation Undermines an IteratorMutation and ContractsUseful Implementation Types | 
                      | 10: Recursion |                  Choosing the Right Decomposition For a ProblemStructure of Recursive ImplementationsHelper MethodsChoosing the Right Recursive SubproblemRecursive Problem vs Recursive DataReentrant CodeWhen to Use Recursion Rather Than IterationCommon Mistakes in Recursive Implementations | 
                      | 11: Debugging |                  Reproduce the BugUnderstand the Location and Cause of the BugFix the Bug | 
                      | 12: Abstract Data Types |                  What Abstraction MeansClassifying Types and OperationsDesigning Abstract TypeRepresentation IndependenceRealizing ADT Concepts in JavaTesting and Abstract Data Type | 
                      | 13: Abstraction Functions and Rep Invariants |                  InvariantsRep Invariant and Abstraction FunctionDocumenting the AF, RI, and Safety from Rep ExposureADT Invariants Replace Preconditions | 
                      | 14: Interfaces |                  InterfacesSubtypesExample: MyStringExample: SetGeneric InterfacesWhy Interfaces?Realizing ADT Concepts in Java, Part II | 
                      | 15: Equality |                  Three Ways to Regard Equality== vs. equals()Equality of Immutable TypesThe Object ContractEquality of Mutable TypesThe Final Rule for Equals() and hashCode() | 
                      | 16: Recursive Data Types |                  Recursive FunctionsImmutable ListsRecursive Datatype DefinitionsFunctions Over Recursive DatatypesTuning the RepNull vs EmptyDeclared Type vs Actual TypeExample: Boolean FormulasWriting a Program with ADTsRecipes for Programming with ADTsExample: Matrix Multiplication | 
                      | 17: Regular Expressions and Grammars |                  GrammarsRegular Expressions | 
                      | 18: Parser Generators |                  Parser GeneratorsAn Antlr GrammarGenerating the ParserCalling the ParserTraversing the Parse TreeConstructing an Abstract Syntax TreeHandling Errors | 
                      | 19: Concurrency |                  Two Models for Concurrent ProgrammingProcesses, Threads, Time-SlicingExample: Shared MemoryInterleavingRace ConditionTweaking the Code Won't HelpReorderingExample: Message PassingConcurrency is Hard to Test and Debug | 
                      | 20: Thread Saftey |                  What Threadsafe MeansStrategy 1: ConfinementStrategy 2: ImmutabilityStrategy 3: Using Threadsafe Data TypesHow to Make A Safety Argument | 
                      | 21: Sockets and Networking |                  Client/Server Design PatternNetwork SocketsI/OBlockingUsing Network SocketsWire ProtocolsTesting Client/Server Code | 
                      | 22: Queues and Message-Passing |                  Two Models for ConcurrencyMessage Passing with ThreadsImplementing Message Passing with QueuesStoppingThread Safety Arguments with Message Passing | 
                      | 23: Locks and Synchronization |                  SynchronizationDeadlockDeveloping a ThreadSafe Abstract Data TypeLockingMonitor PatternThread Safety Argument with SynchronizationAtomic OperationsDesigning a Data Type for ConcurrencyDeadlock Rears its Ugly HeadGoals of Concurrent Program DesignConcurrency in Practice | 
                      | 24: Graphical User Interfaces |                  View TreeHow the View Tree is UsedInput HandlingSeparating Frontend from BackendBackground Processing in Graphical User Interfaces | 
                      | 25: Map, Filter, Reduce |                  Abstracting Out Control FlowMapFunctions as ValuesFilterReduceBenefits of Abstracting Out ControlFirst-class Functions in JavaMap/Filter/Reduce in JavaHigher-order Functions in Java | 
                      | 26: Little Languages |                  Representing Code as DataBuilding Languages to Solve ProblemsMusic Language | 
                      | 27: Team Version Control |                  Git WorkflowViewing Commit HistoryGraph of CommitsUsing Version Control as a Team |