Dynamic programming is used to solve optimization problems by breaking them down into overlapping subproblems. It solves subproblems only once, storing the results in a table to lookup when the same subproblem occurs again, avoiding recomputing solutions. Key steps are characterizing optimal substructures, defining solutions recursively, computing solutions bottom-up, and constructing the overall optimal solution. Examples provided are matrix chain multiplication and longest common subsequence.