Test Driven Development (TDD), and Refactoring Legacy Code Using C#
Duration:  4 Days (Face-to-Face or Remote-Live), or 28 hours of coursework with personal facilitation over a four week span (On-Demand)
US Price: $2095 (Face-to-Face or Remote-Live), or $895 (On-Demand)
Delivery Options: Attend face-to-face in the classroom, remote-live or via on-demand training
Registration: Click here to view upcoming schedules and register for face-to-face sessions, click here to register for live remote attendance sessions, or click here to register for on-demand training with a start date that is convenient for you.
 View the course schedule
 View the course schedule
 
This course provides students with hands on experience learning Test Driven Development (TDD) using NUnit and Microsoft's Visual Studio. Students build unit tests using mocks, fakes, stubs and drivers, and address issues working with databases and other systems. Students learn how to create tests that assure that code will meet and exceed requirements. The course shows how regression testing assures that code that receives “test coverage” will not break existing systems, because tests are passed  before code is checked in. 
Students spend time working with the issues involved in refactoring legacy code, safely cutting into an already deployed system. Students are shown how to look for, or create “seams” to more safely improve code or add  features, and work on identifying “code smells” that need attention in a  productive system. Finally, students will explore dependency issues as well as techniques to  better understand and improve complex systems. 
Comprehensive labs using C# provide facilitated hands on practice that is crucial to the development of competence and confidence  with the new skills being learned.
 
C#  programming experience and an understanding of object-oriented design  principles. The Learning to Program with C# course or  equivalent knowledge provides a solid foundation.
  
    | Why TDD? Think Twice, Write 
      Production Code Once 
        Utilizing a Safety Net of Automated TestingAgile Development ConceptsEliminating Bugs EarlySmoothing Out Production RolloutsWriting Code Faster via Testing Reducing Technical DebtPracticing Emergent DesignMaking Changes More SafeThe Importance of Regression Testing | Basic Unit Testing 
        NUnit and Visual StudioAdding Complexity to Initial Simple Tests Making Tests Easy to RunThe TDD  Pattern: Red, Green RefactorUsing Methods of the Assert ClassBoundary TestingUnit Test Limitations | 
  
    | Comprehensive Unit Testing Concepts 
        Using Declarative-Style AttributesUsing Constraints for More Complex ScenariosUsing Test CategoriesException Handling in TestsNUnit Test Initialization and Clean Up MethodsWriting Clean and Dirty TestsTesting with Collections, Generics and ArraysNegative Testing | Mocks, Fakes, Stubs and Drivers 
        TDD Development PatternsNaming Conventions for Better CodeUsing Mock ObjectsUsing FakesUsing StubsTest DoublesManual MockingMocking with a Mock FrameworkSelf-Shunt Pattern | 
  
    | Database Unit Testing 
        Mocking the Data LayerIdentifying what Should Be Tested in DatabasesStored Procedure TestsSchema TestingUsing NDbUnit to Set Up the DB Test Environment | Refactoring Basics 
        Refactoring Existing Code RestructuringExtracting MethodsRemoving DuplicationReducing CouplingDivision of ResponsibilitiesImproving Clarity and MaintainabilityTest First - then RefactorMore Complex Refactoring Considerations | 
  
    | Patterns and Anti-Patterns in TDD 
        The SOLID PrinciplesFactory MethodsCoding to Interface ReferencesChecking Parameters for Validity TestOpen/Closed Principle: Open to Extension, Closed  to ChangeBreaking Out Method/ObjectExtract and Override CallExtract and Override Factory MethodSingleton PatternDecorator PatternFacade PatternState PatternMVP, MVC and MVVM PatternsFinding and Removing Code Smells/Antipatterns | Refactoring Legacy Code 
        Reducing Risk of Change
          Eliminating DependenciesCharacterization Tests as a Safety NetIntroducing Abstractions to Break Dependencies Analyzing Legacy Code
          Identifying Pinch Points with Effect AnalysisIdentifying Seams for Expansion and TestingListing Markup Minimizing Risk of Adding New Behavior
          Sprout MethodSprout ClassWrap MethodWrap Class Dealing with Code that's Difficult to Test
          Globals and Singletons in TestsInaccessible Methods and Fields Using Smells to Identify What to Refactor
          Dealing with Monster MethodsDealing with Excessively Complex, Large ClassesIdentifying and Eliminating DuplicationOther Smells Dealing with Large Legacy Systems | 
  
    | Code Coverage 
        White Box vs Black Box TestingPlanning to Increase Code Coverage Over Time
          Goal 80% or More Test CoverageStatement CoverageCondition CoveragePath Coverage | Risks Changing Legacy/Production Systems 
        RefactoringCoupling and Cohesion IssuesTaking Small Tested Steps | 
Related Training