Greedy Algorithms
In the two last examples of dynamic programming, we had to consider many sub-problems to solve a problem. For instance, in the rodding cutting problem we had to consider all the possible ways of making the first cut. Similarly, in the LCS problem there were two potential ways for obtaining the optimal solution when the last symbols in the sequence were not the same.
However, sometimes it is possible to greedily select just one case and thus reduce the problem into one smaller sub-problem. We next consider one such simple greedy algorithm. In the graphs rounds, we’ll see other greedy algorithms when we study the minimum spanning tree algorithms.
Example IV: Activity-selection problem
Assume that we are given a finite set of activities
a start time
, and a finish time
with .
An activity executes in the half-open interval
Example
Consider the set of activities
A maximal compatible subset is

A dynamic programming formulation
In order to relate dynamic programming and greedy algorithms, let us first see a dynamic programming approach for solving the activity selection problem.
Assume that the set of activities is
be the set of activities that start at
Each (sub-)problem on
Take any activity
Let
and be any maximal compatible subsets for the sets of activities in ending before and of activities in starting after . Then
is a maximal compatible subset for that contains .
Thus a maximal compatible subset for
The solution for the original problem is
enumerating all the subsets of
, checking the compatibility of each, and
remembering a largest compatible one
would require at least exponential time as there are exponetially many subsets.
In Scala, we can implement the dynamic programming algorithm as follows. Note that the implementation is not very well optimized.
def activitySelectorDynprog(acts: Set[(Int,Int)]) =
require(acts.forall(a => a._1 >= 0 && a._1 < a._2))
val mem = collection.mutable.Map[Set[(Int,Int)],Set[(Int,Int)]]()
def inner(current: Set[(Int,Int)]): Set[(Int,Int)] =
mem.getOrElseUpdate(current, {
var best = Set[(Int,Int)]()
for act <- current do
val earlier = current.filter(a => a._2 <= act._1)
val later = current.filter(a => act._2 <= a._1)
val earlierResult = inner(earlier)
val laterResult = inner(later)
if earlierResult.size + 1 + laterResult.size > best.size then
best = (earlierResult ++ laterResult) + act
best
})
inner(acts)
Running our example with it, we get:
scala> val acts = Set((1,4),(3,5),(0,6),(5,7),(3,9),(5,9),(6,10),
(8,11),(8,12),(2,14),(12,16))
acts: Set[(Int, Int)] = HashSet((8,12), (1,4), (8,11), (5,9), (3,9), (3,5), (12,16), (0,6), (5,7), (6,10), (2,14))
scala> activitySelectorDynprog(acts).toVector.sortBy(_._1)
res1: Vector[(Int, Int)] = Vector((1,4), (5,7), (8,12), (12,16))
The greedy approach
The activity selection problem has a property that we can use to obtain an even faster and simpler algorithm. Namely, we do not have to consider all the activities in a subset but can
just select one that finishes as early as possible, and
then, consider the sub-problem obtained by removing that activity and all the activities that are not compatible with it.
This optimization is based on the observation that
if we have a maximal compatible subset that does not contain
an activity
Theorem
Let
Proof
If
Otherwise, it must be that
Example
Consider again the activities
One maximal compatible subset is
Observe that

In Scala, we can implement the greedy algorithm as follows:
def activitySelectorGreedy(acts: Set[(Int,Int)]) =
require(acts.forall(a => a._1 >= 0 && a._1 < a._2))
val sorted = acts.toVector.sortBy(_._2)
val sol = collection.mutable.ArrayBuffer[(Int,Int)]()
var time = Int.MinValue
for act <- sorted do
if act._1 >= time then
sol += act
time = act._2
sol.toSet
Running our example with it, we get:
scala> val acts = Set((1,4),(3,5),(0,6),(5,7),(3,9),(5,9),(6,10),(8,11),(8,12),(2,14),(12,16))
acts: Set[(Int, Int)] = HashSet((8,12), (1,4), (8,11), (5,9), (3,9), (3,5), (12,16), (0,6), (5,7), (6,10), (2,14))
scala> activitySelectorGreedy(acts).toVector.sortBy(_._1)
res0: Vector[(Int, Int)] = Vector((1,4), (5,7), (8,11), (12,16))