Dynamic Programming Lecture 8 February 15, 2011 Sariel (UIUC) - - PowerPoint PPT Presentation

dynamic programming
SMART_READER_LITE
LIVE PREVIEW

Dynamic Programming Lecture 8 February 15, 2011 Sariel (UIUC) - - PowerPoint PPT Presentation

CS 473: Fundamental Algorithms, Spring 2011 Dynamic Programming Lecture 8 February 15, 2011 Sariel (UIUC) CS473 1 Spring 2011 1 / 38 Part I Longest Increasing Subsequence Sariel (UIUC) CS473 2 Spring 2011 2 / 38 Sequences Definition


slide-1
SLIDE 1

CS 473: Fundamental Algorithms, Spring 2011

Dynamic Programming

Lecture 8

February 15, 2011

Sariel (UIUC) CS473 1 Spring 2011 1 / 38

slide-2
SLIDE 2

Part I Longest Increasing Subsequence

Sariel (UIUC) CS473 2 Spring 2011 2 / 38

slide-3
SLIDE 3

Sequences

Definition

Sequence: an ordered list a1, a2, . . . , an. Length of a sequence is number of elements in the list.

Definition

ai1, . . . , aik is a subsequence of a1, . . . , an if 1 ≤ i1 < . . . < ik ≤ n.

Definition

A sequence is increasing if a1 < a2 < . . . < an. It is non-decreasing if a1 ≤ a2 ≤ . . . ≤ an. Similarly decreasing and non-increasing.

Sariel (UIUC) CS473 3 Spring 2011 3 / 38

slide-4
SLIDE 4

Sequences

Example...

Example

Sequence: 6, 3, 5, 2, 7, 8, 1 Subsequence: 5, 2, 1 Increasing sequence: 3, 5, 9 Increasing subsequence: 2, 7, 8

Sariel (UIUC) CS473 4 Spring 2011 4 / 38

slide-5
SLIDE 5

Longest Increasing Subsequence Problem

Input A sequence of numbers a1, a2, . . . , an Goal Find an increasing subsequence ai1, ai2, . . . , aik of maximum length

Example

Sequence: 6, 3, 5, 2, 7, 8, 1 Increasing subsequences: 6, 7, 8 and 3, 5, 7, 8 and 2, 7 etc Longest increasing subsequence: 3, 5, 7, 8

Sariel (UIUC) CS473 5 Spring 2011 5 / 38

slide-6
SLIDE 6

Longest Increasing Subsequence Problem

Input A sequence of numbers a1, a2, . . . , an Goal Find an increasing subsequence ai1, ai2, . . . , aik of maximum length

Example

Sequence: 6, 3, 5, 2, 7, 8, 1 Increasing subsequences: 6, 7, 8 and 3, 5, 7, 8 and 2, 7 etc Longest increasing subsequence: 3, 5, 7, 8

Sariel (UIUC) CS473 5 Spring 2011 5 / 38

slide-7
SLIDE 7

Na¨ ıve Enumeration

Assume a1, a2, . . . , an is contained in an array A

algLISNaive(A[1..n]): max = 0

for each subsequence B of A do if B is increasing and |B| > max then

max = |B| Output max

Running time: O(n2n). 2n subsequences of a sequence of length n and O(n) time to check if a given sequence is increasing.

Sariel (UIUC) CS473 6 Spring 2011 6 / 38

slide-8
SLIDE 8

Na¨ ıve Enumeration

Assume a1, a2, . . . , an is contained in an array A

algLISNaive(A[1..n]): max = 0

for each subsequence B of A do if B is increasing and |B| > max then

max = |B| Output max

Running time: O(n2n). 2n subsequences of a sequence of length n and O(n) time to check if a given sequence is increasing.

Sariel (UIUC) CS473 6 Spring 2011 6 / 38

slide-9
SLIDE 9

Na¨ ıve Enumeration

Assume a1, a2, . . . , an is contained in an array A

algLISNaive(A[1..n]): max = 0

for each subsequence B of A do if B is increasing and |B| > max then

max = |B| Output max

Running time: O(n2n). 2n subsequences of a sequence of length n and O(n) time to check if a given sequence is increasing.

Sariel (UIUC) CS473 6 Spring 2011 6 / 38

slide-10
SLIDE 10

Recursive Approach: Take 1

LIS: Longest increasing subsequence

Can we find a recursive algorithm for LIS? LIS (A[1..n]): Case 1: does not contain an in which case LIS (A[1..n]) = LIS (A[1..(n − 1)]) Case 2: contains an in which case LIS (A[1..n]) is not so clear. Observation: if an is in the longest increasing subsequence then all the elements before it must be smaller.

Sariel (UIUC) CS473 7 Spring 2011 7 / 38

slide-11
SLIDE 11

Recursive Approach: Take 1

LIS: Longest increasing subsequence

Can we find a recursive algorithm for LIS? LIS (A[1..n]): Case 1: does not contain an in which case LIS (A[1..n]) = LIS (A[1..(n − 1)]) Case 2: contains an in which case LIS (A[1..n]) is not so clear. Observation: if an is in the longest increasing subsequence then all the elements before it must be smaller.

Sariel (UIUC) CS473 7 Spring 2011 7 / 38

slide-12
SLIDE 12

Recursive Approach: Take 1

LIS: Longest increasing subsequence

Can we find a recursive algorithm for LIS? LIS (A[1..n]): Case 1: does not contain an in which case LIS (A[1..n]) = LIS (A[1..(n − 1)]) Case 2: contains an in which case LIS (A[1..n]) is not so clear. Observation: if an is in the longest increasing subsequence then all the elements before it must be smaller.

Sariel (UIUC) CS473 7 Spring 2011 7 / 38

slide-13
SLIDE 13

Recursive Approach: Take 1

LIS: Longest increasing subsequence

Can we find a recursive algorithm for LIS? LIS (A[1..n]): Case 1: does not contain an in which case LIS (A[1..n]) = LIS (A[1..(n − 1)]) Case 2: contains an in which case LIS (A[1..n]) is not so clear. Observation: if an is in the longest increasing subsequence then all the elements before it must be smaller.

Sariel (UIUC) CS473 7 Spring 2011 7 / 38

slide-14
SLIDE 14

Recursive Approach: Take 1

algLIS(A[1..n]):

if (n = 0) then return 0

m = algLIS(A[1..(n − 1)]) B is subsequence of A[1..(n − 1)] with

  • nly elements less than an

(* let h be size of B, h ≤ n-1 *) m = max(m, 1 + algLIS(B[1..h])) Output m

Recursion for running time: T(n) ≤ 2T(n − 1) + O(n). Easy to see that T(n) is O(n2n).

Sariel (UIUC) CS473 8 Spring 2011 8 / 38

slide-15
SLIDE 15

Recursive Approach: Take 1

algLIS(A[1..n]):

if (n = 0) then return 0

m = algLIS(A[1..(n − 1)]) B is subsequence of A[1..(n − 1)] with

  • nly elements less than an

(* let h be size of B, h ≤ n-1 *) m = max(m, 1 + algLIS(B[1..h])) Output m

Recursion for running time: T(n) ≤ 2T(n − 1) + O(n). Easy to see that T(n) is O(n2n).

Sariel (UIUC) CS473 8 Spring 2011 8 / 38

slide-16
SLIDE 16

Recursive Approach: Take 1

algLIS(A[1..n]):

if (n = 0) then return 0

m = algLIS(A[1..(n − 1)]) B is subsequence of A[1..(n − 1)] with

  • nly elements less than an

(* let h be size of B, h ≤ n-1 *) m = max(m, 1 + algLIS(B[1..h])) Output m

Recursion for running time: T(n) ≤ 2T(n − 1) + O(n). Easy to see that T(n) is O(n2n).

Sariel (UIUC) CS473 8 Spring 2011 8 / 38

slide-17
SLIDE 17

Recursive Approach: Take 2

LIS(A[1..n]): Case 1: does not contain an in which case LIS(A[1..n]) = LIS(A[1..(n − 1)]) Case 2: contains an in which case LIS(A[1..n]) is not so clear.

Observation

For second case we want to find a subsequence in A[1..(n − 1)] that is restricted to numbers less than an. This suggests that a more general problem is LIS smaller(A[1..n], x) which gives the longest increasing subsequence in A where each number in the sequence is less than x.

Sariel (UIUC) CS473 9 Spring 2011 9 / 38

slide-18
SLIDE 18

Recursive Approach: Take 2

LIS smaller(A[1..n], x) : length of longest increasing subsequence in A[1..n] with all numbers in subsequence less than x

LIS smaller(A[1..n], x):

if (n = 0) then return 0

m = LIS smaller(A[1..(n − 1)], x)

if (A[n] < x) then

m = max(m, 1 + LIS smaller(A[1..(n − 1)], A[n])) Output m LIS(A[1..n]): return LIS smaller(A[1..n], ∞)

Recursion for running time: T(n) ≤ 2T(n − 1) + O(1). Question: Is there any advantage?

Sariel (UIUC) CS473 10 Spring 2011 10 / 38

slide-19
SLIDE 19

Recursive Approach: Take 2

LIS smaller(A[1..n], x) : length of longest increasing subsequence in A[1..n] with all numbers in subsequence less than x

LIS smaller(A[1..n], x):

if (n = 0) then return 0

m = LIS smaller(A[1..(n − 1)], x)

if (A[n] < x) then

m = max(m, 1 + LIS smaller(A[1..(n − 1)], A[n])) Output m LIS(A[1..n]): return LIS smaller(A[1..n], ∞)

Recursion for running time: T(n) ≤ 2T(n − 1) + O(1). Question: Is there any advantage?

Sariel (UIUC) CS473 10 Spring 2011 10 / 38

slide-20
SLIDE 20

Recursive Approach: Take 2

LIS smaller(A[1..n], x) : length of longest increasing subsequence in A[1..n] with all numbers in subsequence less than x

LIS smaller(A[1..n], x):

if (n = 0) then return 0

m = LIS smaller(A[1..(n − 1)], x)

if (A[n] < x) then

m = max(m, 1 + LIS smaller(A[1..(n − 1)], A[n])) Output m LIS(A[1..n]): return LIS smaller(A[1..n], ∞)

Recursion for running time: T(n) ≤ 2T(n − 1) + O(1). Question: Is there any advantage?

Sariel (UIUC) CS473 10 Spring 2011 10 / 38

slide-21
SLIDE 21

Recursive Algorithm: Take 2

Observation: The number of different subproblems generated by LIS smaller(A[1..n], x) is O(n2). Memoization the recursive algorithm leads to an O(n2) running time! Question: What are the recursive subproblem generated by LIS smaller(A[1..n], x)? For 0 ≤ i < n LIS smaller(A[1..i], y) where y is either x or

  • ne of A[i + 1], . . . , A[n].

Observation: previous recursion also generates only O(n2)

  • subproblems. Slightly harder to see.

Sariel (UIUC) CS473 11 Spring 2011 11 / 38

slide-22
SLIDE 22

Recursive Algorithm: Take 2

Observation: The number of different subproblems generated by LIS smaller(A[1..n], x) is O(n2). Memoization the recursive algorithm leads to an O(n2) running time! Question: What are the recursive subproblem generated by LIS smaller(A[1..n], x)? For 0 ≤ i < n LIS smaller(A[1..i], y) where y is either x or

  • ne of A[i + 1], . . . , A[n].

Observation: previous recursion also generates only O(n2)

  • subproblems. Slightly harder to see.

Sariel (UIUC) CS473 11 Spring 2011 11 / 38

slide-23
SLIDE 23

Recursive Algorithm: Take 2

Observation: The number of different subproblems generated by LIS smaller(A[1..n], x) is O(n2). Memoization the recursive algorithm leads to an O(n2) running time! Question: What are the recursive subproblem generated by LIS smaller(A[1..n], x)? For 0 ≤ i < n LIS smaller(A[1..i], y) where y is either x or

  • ne of A[i + 1], . . . , A[n].

Observation: previous recursion also generates only O(n2)

  • subproblems. Slightly harder to see.

Sariel (UIUC) CS473 11 Spring 2011 11 / 38

slide-24
SLIDE 24

Recursive Algorithm: Take 2

Observation: The number of different subproblems generated by LIS smaller(A[1..n], x) is O(n2). Memoization the recursive algorithm leads to an O(n2) running time! Question: What are the recursive subproblem generated by LIS smaller(A[1..n], x)? For 0 ≤ i < n LIS smaller(A[1..i], y) where y is either x or

  • ne of A[i + 1], . . . , A[n].

Observation: previous recursion also generates only O(n2)

  • subproblems. Slightly harder to see.

Sariel (UIUC) CS473 11 Spring 2011 11 / 38

slide-25
SLIDE 25

Recursive Algorithm: Take 2

Observation: The number of different subproblems generated by LIS smaller(A[1..n], x) is O(n2). Memoization the recursive algorithm leads to an O(n2) running time! Question: What are the recursive subproblem generated by LIS smaller(A[1..n], x)? For 0 ≤ i < n LIS smaller(A[1..i], y) where y is either x or

  • ne of A[i + 1], . . . , A[n].

Observation: previous recursion also generates only O(n2)

  • subproblems. Slightly harder to see.

Sariel (UIUC) CS473 11 Spring 2011 11 / 38

slide-26
SLIDE 26

Recursive Algorithm: Take 3

LISEnding(A[1..n]): length of longest increasing sub-sequence that ends in A[n]. Question: can we obtain a recursive expression? LISEnding(A[1..n]) = max

i:A[i]<A[n](1 + LISEnding(A[1..i]))

Sariel (UIUC) CS473 12 Spring 2011 12 / 38

slide-27
SLIDE 27

Recursive Algorithm: Take 3

LISEnding(A[1..n]): length of longest increasing sub-sequence that ends in A[n]. Question: can we obtain a recursive expression? LISEnding(A[1..n]) = max

i:A[i]<A[n](1 + LISEnding(A[1..i]))

Sariel (UIUC) CS473 12 Spring 2011 12 / 38

slide-28
SLIDE 28

Recursive Algorithm: Take 3

LIS ending(A[1..n]):

if (n = 0) return 0

m = 1

for i = 1 to n − 1 do if (A[i] < A[n]) then

m = max(m, 1 + LIS ending(A[1..i]))

return m

LIS(A[1..n]): return maxn

i=1LIS ending(A[1 . . . i])

Question: How many distinct subproblems generated by LISEnding(A[1..n])? n.

Sariel (UIUC) CS473 13 Spring 2011 13 / 38

slide-29
SLIDE 29

Recursive Algorithm: Take 3

LIS ending(A[1..n]):

if (n = 0) return 0

m = 1

for i = 1 to n − 1 do if (A[i] < A[n]) then

m = max(m, 1 + LIS ending(A[1..i]))

return m

LIS(A[1..n]): return maxn

i=1LIS ending(A[1 . . . i])

Question: How many distinct subproblems generated by LISEnding(A[1..n])? n.

Sariel (UIUC) CS473 13 Spring 2011 13 / 38

slide-30
SLIDE 30

Recursive Algorithm: Take 3

LIS ending(A[1..n]):

if (n = 0) return 0

m = 1

for i = 1 to n − 1 do if (A[i] < A[n]) then

m = max(m, 1 + LIS ending(A[1..i]))

return m

LIS(A[1..n]): return maxn

i=1LIS ending(A[1 . . . i])

Question: How many distinct subproblems generated by LISEnding(A[1..n])? n.

Sariel (UIUC) CS473 13 Spring 2011 13 / 38

slide-31
SLIDE 31

Iterative Algorithm via Memoization

Compute the values LISEnding(A[1..i]) iteratively in a bottom up fashion.

LIS ending(A[1..n]): Array L[1..n] (* L[i] = value of LIS ending(A[1..i]) *)

for i = 1 to n do

L[i] = 1

for j = 1 to i − 1 do if (A[j] < A[i]) do

L[i] = max(L[i], 1 + L[j]) return L LIS(A[1..n]): L = LIS ending(A[1..n])

return the maximum value in L

Sariel (UIUC) CS473 14 Spring 2011 14 / 38

slide-32
SLIDE 32

Iterative Algorithm via Memoization

Simplifying:

LIS(A[1..n]): Array L[1..n] (* L[i] stores the value LIS-Ending(A[1..i]) *) m = 0

for i = 1 to n do

L[i] = 1

for j = 1 to i − 1 do if (A[j] < A[i]) do

L[i] = max(L[i], 1 + L[j]) m = max(m, L[i])

return m

Correctness: Via induction following the recursion Running time: O(n2), Space: Θ(n)

Sariel (UIUC) CS473 15 Spring 2011 15 / 38

slide-33
SLIDE 33

Iterative Algorithm via Memoization

Simplifying:

LIS(A[1..n]): Array L[1..n] (* L[i] stores the value LIS-Ending(A[1..i]) *) m = 0

for i = 1 to n do

L[i] = 1

for j = 1 to i − 1 do if (A[j] < A[i]) do

L[i] = max(L[i], 1 + L[j]) m = max(m, L[i])

return m

Correctness: Via induction following the recursion Running time: O(n2), Space: Θ(n)

Sariel (UIUC) CS473 15 Spring 2011 15 / 38

slide-34
SLIDE 34

Iterative Algorithm via Memoization

Simplifying:

LIS(A[1..n]): Array L[1..n] (* L[i] stores the value LIS-Ending(A[1..i]) *) m = 0

for i = 1 to n do

L[i] = 1

for j = 1 to i − 1 do if (A[j] < A[i]) do

L[i] = max(L[i], 1 + L[j]) m = max(m, L[i])

return m

Correctness: Via induction following the recursion Running time: O(n2), Space: Θ(n)

Sariel (UIUC) CS473 15 Spring 2011 15 / 38

slide-35
SLIDE 35

Iterative Algorithm via Memoization

Simplifying:

LIS(A[1..n]): Array L[1..n] (* L[i] stores the value LIS-Ending(A[1..i]) *) m = 0

for i = 1 to n do

L[i] = 1

for j = 1 to i − 1 do if (A[j] < A[i]) do

L[i] = max(L[i], 1 + L[j]) m = max(m, L[i])

return m

Correctness: Via induction following the recursion Running time: O(n2), Space: Θ(n)

Sariel (UIUC) CS473 15 Spring 2011 15 / 38

slide-36
SLIDE 36

Iterative Algorithm via Memoization

Simplifying:

LIS(A[1..n]): Array L[1..n] (* L[i] stores the value LIS-Ending(A[1..i]) *) m = 0

for i = 1 to n do

L[i] = 1

for j = 1 to i − 1 do if (A[j] < A[i]) do

L[i] = max(L[i], 1 + L[j]) m = max(m, L[i])

return m

Correctness: Via induction following the recursion Running time: O(n2), Space: Θ(n)

Sariel (UIUC) CS473 15 Spring 2011 15 / 38

slide-37
SLIDE 37

Iterative Algorithm via Memoization

Simplifying:

LIS(A[1..n]): Array L[1..n] (* L[i] stores the value LIS-Ending(A[1..i]) *) m = 0

for i = 1 to n do

L[i] = 1

for j = 1 to i − 1 do if (A[j] < A[i]) do

L[i] = max(L[i], 1 + L[j]) m = max(m, L[i])

return m

Correctness: Via induction following the recursion Running time: O(n2), Space: Θ(n)

Sariel (UIUC) CS473 15 Spring 2011 15 / 38

slide-38
SLIDE 38

Example

Example

Sequence: 6, 3, 5, 2, 7, 8, 1 Longest increasing subsequence: 3, 5, 7, 8 L[i] is value of longest increasing subsequence ending in A[i] Recursive algorithm computes L[i] from L[1] to L[i − 1] Iterative algorithm builds up the values from L[1] to L[n]

Sariel (UIUC) CS473 16 Spring 2011 16 / 38

slide-39
SLIDE 39

Example

Example

Sequence: 6, 3, 5, 2, 7, 8, 1 Longest increasing subsequence: 3, 5, 7, 8 L[i] is value of longest increasing subsequence ending in A[i] Recursive algorithm computes L[i] from L[1] to L[i − 1] Iterative algorithm builds up the values from L[1] to L[n]

Sariel (UIUC) CS473 16 Spring 2011 16 / 38

slide-40
SLIDE 40

Memoizing LIS smaller

LIS(A[1..n]): A[n + 1] = ∞ (* add a sentinel at the end *) Array L[(n + 1), (n + 1)] (* two-dimensional array*) (* L[i, j] for j ≥ i stores the value LIS smaller(A[1..i],A[j]) *)

for j = 1 to n + 1 do

L[0, j] = 0

for i = 1 to n + 1 do for j = i to n + 1 do

L[i, j] = L[i − 1, j]

if (A[i] < A[j]) then

L[i, j] = max(L[i, j], 1 + L[i − 1, i])

return L[n, (n + 1)]

Correctness: Via induction following the recursion (take 2) Running time: O(n2), Space: Θ(n2)

Sariel (UIUC) CS473 17 Spring 2011 17 / 38

slide-41
SLIDE 41

Memoizing LIS smaller

LIS(A[1..n]): A[n + 1] = ∞ (* add a sentinel at the end *) Array L[(n + 1), (n + 1)] (* two-dimensional array*) (* L[i, j] for j ≥ i stores the value LIS smaller(A[1..i],A[j]) *)

for j = 1 to n + 1 do

L[0, j] = 0

for i = 1 to n + 1 do for j = i to n + 1 do

L[i, j] = L[i − 1, j]

if (A[i] < A[j]) then

L[i, j] = max(L[i, j], 1 + L[i − 1, i])

return L[n, (n + 1)]

Correctness: Via induction following the recursion (take 2) Running time: O(n2), Space: Θ(n2)

Sariel (UIUC) CS473 17 Spring 2011 17 / 38

slide-42
SLIDE 42

Memoizing LIS smaller

LIS(A[1..n]): A[n + 1] = ∞ (* add a sentinel at the end *) Array L[(n + 1), (n + 1)] (* two-dimensional array*) (* L[i, j] for j ≥ i stores the value LIS smaller(A[1..i],A[j]) *)

for j = 1 to n + 1 do

L[0, j] = 0

for i = 1 to n + 1 do for j = i to n + 1 do

L[i, j] = L[i − 1, j]

if (A[i] < A[j]) then

L[i, j] = max(L[i, j], 1 + L[i − 1, i])

return L[n, (n + 1)]

Correctness: Via induction following the recursion (take 2) Running time: O(n2), Space: Θ(n2)

Sariel (UIUC) CS473 17 Spring 2011 17 / 38

slide-43
SLIDE 43

Memoizing LIS smaller

LIS(A[1..n]): A[n + 1] = ∞ (* add a sentinel at the end *) Array L[(n + 1), (n + 1)] (* two-dimensional array*) (* L[i, j] for j ≥ i stores the value LIS smaller(A[1..i],A[j]) *)

for j = 1 to n + 1 do

L[0, j] = 0

for i = 1 to n + 1 do for j = i to n + 1 do

L[i, j] = L[i − 1, j]

if (A[i] < A[j]) then

L[i, j] = max(L[i, j], 1 + L[i − 1, i])

return L[n, (n + 1)]

Correctness: Via induction following the recursion (take 2) Running time: O(n2), Space: Θ(n2)

Sariel (UIUC) CS473 17 Spring 2011 17 / 38

slide-44
SLIDE 44

Memoizing LIS smaller

LIS(A[1..n]): A[n + 1] = ∞ (* add a sentinel at the end *) Array L[(n + 1), (n + 1)] (* two-dimensional array*) (* L[i, j] for j ≥ i stores the value LIS smaller(A[1..i],A[j]) *)

for j = 1 to n + 1 do

L[0, j] = 0

for i = 1 to n + 1 do for j = i to n + 1 do

L[i, j] = L[i − 1, j]

if (A[i] < A[j]) then

L[i, j] = max(L[i, j], 1 + L[i − 1, i])

return L[n, (n + 1)]

Correctness: Via induction following the recursion (take 2) Running time: O(n2), Space: Θ(n2)

Sariel (UIUC) CS473 17 Spring 2011 17 / 38

slide-45
SLIDE 45

Memoizing LIS smaller

LIS(A[1..n]): A[n + 1] = ∞ (* add a sentinel at the end *) Array L[(n + 1), (n + 1)] (* two-dimensional array*) (* L[i, j] for j ≥ i stores the value LIS smaller(A[1..i],A[j]) *)

for j = 1 to n + 1 do

L[0, j] = 0

for i = 1 to n + 1 do for j = i to n + 1 do

L[i, j] = L[i − 1, j]

if (A[i] < A[j]) then

L[i, j] = max(L[i, j], 1 + L[i − 1, i])

return L[n, (n + 1)]

Correctness: Via induction following the recursion (take 2) Running time: O(n2), Space: Θ(n2)

Sariel (UIUC) CS473 17 Spring 2011 17 / 38

slide-46
SLIDE 46

Longest increasing subsequence

Another way to get quadratic time algorithm

G =({s, 1, . . . , n} , {}): directed graph.

∀i, j: If i < j and A[i] < A[j] then add the edge i → j to G. ∀i: Add s → i.

The graph G is a DAG. LIS corresponds to longest path in G starting at s. We know how to compute this in O(|V(G)| + |E(G)|) = O(n2). Comment: One can compute LIS in O(n log n) time with a bit more work.

Sariel (UIUC) CS473 18 Spring 2011 18 / 38

slide-47
SLIDE 47

Dynamic Programming

Find a “smart” recursion for the problem in which the number of distinct subproblems is small; polynomial in the original problem size. Eliminate recursion and find an iterative algorithm to compute the problems bottom up by storing the intermediate values in an appropriate data structure; need to find the right way or order the subproblem evaluation. Estimate the number of subproblems, the time to evaluate each subproblem and the space needed to store the value. Evaluate the total running time. Optimize the resulting algorithm further

Sariel (UIUC) CS473 19 Spring 2011 19 / 38

slide-48
SLIDE 48

Part II Weighted Interval Scheduling

Sariel (UIUC) CS473 20 Spring 2011 20 / 38

slide-49
SLIDE 49

Weighted Interval Scheduling

Input A set of jobs with start times, finish times and weights (or profits). Goal Schedule jobs so that total weight of jobs is maximized. Two jobs with overlapping intervals cannot both be scheduled!

2 1 2 3 1 4 10 10 1 1 Sariel (UIUC) CS473 21 Spring 2011 21 / 38

slide-50
SLIDE 50

Weighted Interval Scheduling

Input A set of jobs with start times, finish times and weights (or profits). Goal Schedule jobs so that total weight of jobs is maximized. Two jobs with overlapping intervals cannot both be scheduled!

2 1 2 3 1 4 10 10 1 1 Sariel (UIUC) CS473 21 Spring 2011 21 / 38

slide-51
SLIDE 51

Interval Scheduling

Input A set of jobs with start and finish times to be scheduled

  • n a resource; special case where all jobs have weight 1.

Goal Schedule as many jobs as possible. Greedy strategy of considering jobs according to finish times produces optimal schedule (to be seen later).

Sariel (UIUC) CS473 22 Spring 2011 22 / 38

slide-52
SLIDE 52

Interval Scheduling

Input A set of jobs with start and finish times to be scheduled

  • n a resource; special case where all jobs have weight 1.

Goal Schedule as many jobs as possible. Greedy strategy of considering jobs according to finish times produces optimal schedule (to be seen later).

Sariel (UIUC) CS473 22 Spring 2011 22 / 38

slide-53
SLIDE 53

Interval Scheduling

Input A set of jobs with start and finish times to be scheduled

  • n a resource; special case where all jobs have weight 1.

Goal Schedule as many jobs as possible. Greedy strategy of considering jobs according to finish times produces optimal schedule (to be seen later).

Sariel (UIUC) CS473 22 Spring 2011 22 / 38

slide-54
SLIDE 54

Interval Scheduling

Input A set of jobs with start and finish times to be scheduled

  • n a resource; special case where all jobs have weight 1.

Goal Schedule as many jobs as possible. Greedy strategy of considering jobs according to finish times produces optimal schedule (to be seen later).

Sariel (UIUC) CS473 22 Spring 2011 22 / 38

slide-55
SLIDE 55

Interval Scheduling

Input A set of jobs with start and finish times to be scheduled

  • n a resource; special case where all jobs have weight 1.

Goal Schedule as many jobs as possible. Greedy strategy of considering jobs according to finish times produces optimal schedule (to be seen later).

Sariel (UIUC) CS473 22 Spring 2011 22 / 38

slide-56
SLIDE 56

Interval Scheduling

Input A set of jobs with start and finish times to be scheduled

  • n a resource; special case where all jobs have weight 1.

Goal Schedule as many jobs as possible. Greedy strategy of considering jobs according to finish times produces optimal schedule (to be seen later).

Sariel (UIUC) CS473 22 Spring 2011 22 / 38

slide-57
SLIDE 57

Interval Scheduling

Input A set of jobs with start and finish times to be scheduled

  • n a resource; special case where all jobs have weight 1.

Goal Schedule as many jobs as possible. Greedy strategy of considering jobs according to finish times produces optimal schedule (to be seen later).

Sariel (UIUC) CS473 22 Spring 2011 22 / 38

slide-58
SLIDE 58

Greedy Strategies

Earliest finish time first Largest weight/profit first Largest weight to length ratio first Shortest length first . . . None of the above strategies lead to an optimum solution. Moral: Greedy strategies often don’t work!

Sariel (UIUC) CS473 23 Spring 2011 23 / 38

slide-59
SLIDE 59

Greedy Strategies

Earliest finish time first Largest weight/profit first Largest weight to length ratio first Shortest length first . . . None of the above strategies lead to an optimum solution. Moral: Greedy strategies often don’t work!

Sariel (UIUC) CS473 23 Spring 2011 23 / 38

slide-60
SLIDE 60

Reduction to Max Weight Independent Set Problem

Given weighted interval scheduling instance I create an instance

  • f max weight independent set on a graph G(I) as follows.

For each interval i create a vertex vi with weight wi. Add an edge between vi and vj if i and j overlap.

Claim: max weight independent set in G(I) has weight equal to max weight set of intervals in I that do not overlap We do not know an efficient (polynomial time) algorithm for independent set! Can we take advantage of the interval structure to find an efficient algorithm?

Sariel (UIUC) CS473 24 Spring 2011 24 / 38

slide-61
SLIDE 61

Reduction to Max Weight Independent Set Problem

Given weighted interval scheduling instance I create an instance

  • f max weight independent set on a graph G(I) as follows.

For each interval i create a vertex vi with weight wi. Add an edge between vi and vj if i and j overlap.

Claim: max weight independent set in G(I) has weight equal to max weight set of intervals in I that do not overlap We do not know an efficient (polynomial time) algorithm for independent set! Can we take advantage of the interval structure to find an efficient algorithm?

Sariel (UIUC) CS473 24 Spring 2011 24 / 38

slide-62
SLIDE 62

Reduction to Max Weight Independent Set Problem

Given weighted interval scheduling instance I create an instance

  • f max weight independent set on a graph G(I) as follows.

For each interval i create a vertex vi with weight wi. Add an edge between vi and vj if i and j overlap.

Claim: max weight independent set in G(I) has weight equal to max weight set of intervals in I that do not overlap We do not know an efficient (polynomial time) algorithm for independent set! Can we take advantage of the interval structure to find an efficient algorithm?

Sariel (UIUC) CS473 24 Spring 2011 24 / 38

slide-63
SLIDE 63

Conventions

Definition

Let the requests be sorted according to finish time, i.e., i < j implies fi ≤ fj Define p(j) to be the largest i (less than j) such that job i and job j are not in conflict

Example

1 2 3 4 5 6 v1 = 2 v2 = 4 v3 = 4 v4 = 7 v5 = 2 v6 = 1 p(1) = 0 p(2) = 0 p(3) = 1 p(4) = 0 p(5) = 3 p(6) = 3 Sariel (UIUC) CS473 25 Spring 2011 25 / 38

slide-64
SLIDE 64

Towards a Recursive Solution

Observation

Consider an optimal schedule O Case n ∈ O : None of the jobs between n and p(n) can be

  • scheduled. Moreover O must contain an optimal

schedule for the first p(n) jobs. Case n ∈ O : O is an optimal schedule for the first n − 1 jobs.

Sariel (UIUC) CS473 26 Spring 2011 26 / 38

slide-65
SLIDE 65

Towards a Recursive Solution

Observation

Consider an optimal schedule O Case n ∈ O : None of the jobs between n and p(n) can be

  • scheduled. Moreover O must contain an optimal

schedule for the first p(n) jobs. Case n ∈ O : O is an optimal schedule for the first n − 1 jobs.

Sariel (UIUC) CS473 26 Spring 2011 26 / 38

slide-66
SLIDE 66

A Recursive Algorithm

Let Oi be value of an optimal schedule for the first i jobs.

Schedule(n):

if n = 0 then return 0 if n = 1 then return w(v1)

Op(n) ←Schedule(p(n)) On−1 ←Schedule(n − 1)

if (Op(n) + w(vn) < On−1) then

On = On−1

else

On = Op(n) + w(vn)

return On

Time Analysis

Running time is T(n) = T(p(n)) + T(n − 1) + O(1) which is . . .

Sariel (UIUC) CS473 27 Spring 2011 27 / 38

slide-67
SLIDE 67

A Recursive Algorithm

Let Oi be value of an optimal schedule for the first i jobs.

Schedule(n):

if n = 0 then return 0 if n = 1 then return w(v1)

Op(n) ←Schedule(p(n)) On−1 ←Schedule(n − 1)

if (Op(n) + w(vn) < On−1) then

On = On−1

else

On = Op(n) + w(vn)

return On

Time Analysis

Running time is T(n) = T(p(n)) + T(n − 1) + O(1) which is . . .

Sariel (UIUC) CS473 27 Spring 2011 27 / 38

slide-68
SLIDE 68

Bad Example

Figure: Bad instance for recursive algorithm

Running time on this instance is T(n) = T(n − 1) + T(n − 2) + O(1) = Θ(φn) where φ ≈ 1.618 is the golden ratio.

Sariel (UIUC) CS473 28 Spring 2011 28 / 38

slide-69
SLIDE 69

Bad Example

Figure: Bad instance for recursive algorithm

Running time on this instance is T(n) = T(n − 1) + T(n − 2) + O(1) = Θ(φn) where φ ≈ 1.618 is the golden ratio.

Sariel (UIUC) CS473 28 Spring 2011 28 / 38

slide-70
SLIDE 70

Analysis of the Problem

n − 2 n − 3 n − 3 n − 4 n − 1 n − 2 n . . . . . . . . . . . .

Figure: Label of node indicates size of sub-problem. Tree of sub-problems grows very quickly

Sariel (UIUC) CS473 29 Spring 2011 29 / 38

slide-71
SLIDE 71

Memo(r)ization

Observation

Number of different sub-problems in recursive algorithm is O(n); they are O1, O2, . . . , On−1 Exponential time is due to recomputation of solutions to sub-problems

Solution

Store optimal solution to different sub-problems, and perform recursive call only if not already computed.

Sariel (UIUC) CS473 30 Spring 2011 30 / 38

slide-72
SLIDE 72

Recursive Solution with Memoization

schdIMem(j)

if j = 0 then return 0 if M[j] is defined then (* sub-problem already solved *) return M[j] if M[j] is not defined then

M[j] = max

  • w(vj) + schdIMem(p(j)),

schdIMem(j − 1)

  • return M[j]

Time Analysis

Each invocation, O(1) time plus: either return a computed value, or generate 2 recursive calls and fill one M[·] Initially no entry of M[] is filled; at the end all entries of M[] are filled So total time is O(n)

Sariel (UIUC) CS473 31 Spring 2011 31 / 38

slide-73
SLIDE 73

Recursive Solution with Memoization

schdIMem(j)

if j = 0 then return 0 if M[j] is defined then (* sub-problem already solved *) return M[j] if M[j] is not defined then

M[j] = max

  • w(vj) + schdIMem(p(j)),

schdIMem(j − 1)

  • return M[j]

Time Analysis

Each invocation, O(1) time plus: either return a computed value, or generate 2 recursive calls and fill one M[·] Initially no entry of M[] is filled; at the end all entries of M[] are filled So total time is O(n)

Sariel (UIUC) CS473 31 Spring 2011 31 / 38

slide-74
SLIDE 74

Recursive Solution with Memoization

schdIMem(j)

if j = 0 then return 0 if M[j] is defined then (* sub-problem already solved *) return M[j] if M[j] is not defined then

M[j] = max

  • w(vj) + schdIMem(p(j)),

schdIMem(j − 1)

  • return M[j]

Time Analysis

Each invocation, O(1) time plus: either return a computed value, or generate 2 recursive calls and fill one M[·] Initially no entry of M[] is filled; at the end all entries of M[] are filled So total time is O(n)

Sariel (UIUC) CS473 31 Spring 2011 31 / 38

slide-75
SLIDE 75

Recursive Solution with Memoization

schdIMem(j)

if j = 0 then return 0 if M[j] is defined then (* sub-problem already solved *) return M[j] if M[j] is not defined then

M[j] = max

  • w(vj) + schdIMem(p(j)),

schdIMem(j − 1)

  • return M[j]

Time Analysis

Each invocation, O(1) time plus: either return a computed value, or generate 2 recursive calls and fill one M[·] Initially no entry of M[] is filled; at the end all entries of M[] are filled So total time is O(n)

Sariel (UIUC) CS473 31 Spring 2011 31 / 38

slide-76
SLIDE 76

Recursive Solution with Memoization

schdIMem(j)

if j = 0 then return 0 if M[j] is defined then (* sub-problem already solved *) return M[j] if M[j] is not defined then

M[j] = max

  • w(vj) + schdIMem(p(j)),

schdIMem(j − 1)

  • return M[j]

Time Analysis

Each invocation, O(1) time plus: either return a computed value, or generate 2 recursive calls and fill one M[·] Initially no entry of M[] is filled; at the end all entries of M[] are filled So total time is O(n)

Sariel (UIUC) CS473 31 Spring 2011 31 / 38

slide-77
SLIDE 77

Automatic Memoization

Fact

Many functional languages (like LISP) automatically do memoization for recursive function calls!

Sariel (UIUC) CS473 32 Spring 2011 32 / 38

slide-78
SLIDE 78

Back to Weighted Interval Scheduling

Iterative Solution

M[0] = 0

for i = 1 to n do

M[i] = max

  • w(vi) + M[p(i)], M[i − 1]
  • M: table of subproblems

Implicitly dynamic programming fills the values of M Recursion determines order in which table is filled up Think of decomposing problem first (recursion) and then worry about setting up table — this comes naturally from recursion

Sariel (UIUC) CS473 33 Spring 2011 33 / 38

slide-79
SLIDE 79

Back to Weighted Interval Scheduling

Iterative Solution

M[0] = 0

for i = 1 to n do

M[i] = max

  • w(vi) + M[p(i)], M[i − 1]
  • M: table of subproblems

Implicitly dynamic programming fills the values of M Recursion determines order in which table is filled up Think of decomposing problem first (recursion) and then worry about setting up table — this comes naturally from recursion

Sariel (UIUC) CS473 33 Spring 2011 33 / 38

slide-80
SLIDE 80

Example

30 70 80 20 10 1 2 3 4 5 p(5) = 2, p(4) = 1, p(3) = 1, p(2) = 0, p(1) = 0

Sariel (UIUC) CS473 34 Spring 2011 34 / 38

slide-81
SLIDE 81

Computing Solutions + First Attempt

Memoization + Recursion/Iteration allows one to compute the

  • ptimal value. What about the actual schedule?

M[0] = 0 S[0] is empty schedule

for i = 1 to n do

M[i] = max

  • w(vi) + M[p(i)], M[i − 1]
  • if w(vi) + M[p(i)] < M[i − 1] then

S[i] = S[i − 1]

else

S[i] = S[p(i)] ∪ {i}

Na¨ ıvely updating S[] takes O(n) time Total running time is O(n2) Using pointers running time can be improved to O(n).

Sariel (UIUC) CS473 35 Spring 2011 35 / 38

slide-82
SLIDE 82

Computing Solutions + First Attempt

Memoization + Recursion/Iteration allows one to compute the

  • ptimal value. What about the actual schedule?

M[0] = 0 S[0] is empty schedule

for i = 1 to n do

M[i] = max

  • w(vi) + M[p(i)], M[i − 1]
  • if w(vi) + M[p(i)] < M[i − 1] then

S[i] = S[i − 1]

else

S[i] = S[p(i)] ∪ {i}

Na¨ ıvely updating S[] takes O(n) time Total running time is O(n2) Using pointers running time can be improved to O(n).

Sariel (UIUC) CS473 35 Spring 2011 35 / 38

slide-83
SLIDE 83

Computing Solutions + First Attempt

Memoization + Recursion/Iteration allows one to compute the

  • ptimal value. What about the actual schedule?

M[0] = 0 S[0] is empty schedule

for i = 1 to n do

M[i] = max

  • w(vi) + M[p(i)], M[i − 1]
  • if w(vi) + M[p(i)] < M[i − 1] then

S[i] = S[i − 1]

else

S[i] = S[p(i)] ∪ {i}

Na¨ ıvely updating S[] takes O(n) time Total running time is O(n2) Using pointers running time can be improved to O(n).

Sariel (UIUC) CS473 35 Spring 2011 35 / 38

slide-84
SLIDE 84

Computing Solutions + First Attempt

Memoization + Recursion/Iteration allows one to compute the

  • ptimal value. What about the actual schedule?

M[0] = 0 S[0] is empty schedule

for i = 1 to n do

M[i] = max

  • w(vi) + M[p(i)], M[i − 1]
  • if w(vi) + M[p(i)] < M[i − 1] then

S[i] = S[i − 1]

else

S[i] = S[p(i)] ∪ {i}

Na¨ ıvely updating S[] takes O(n) time Total running time is O(n2) Using pointers running time can be improved to O(n).

Sariel (UIUC) CS473 35 Spring 2011 35 / 38

slide-85
SLIDE 85

Computing Solutions + First Attempt

Memoization + Recursion/Iteration allows one to compute the

  • ptimal value. What about the actual schedule?

M[0] = 0 S[0] is empty schedule

for i = 1 to n do

M[i] = max

  • w(vi) + M[p(i)], M[i − 1]
  • if w(vi) + M[p(i)] < M[i − 1] then

S[i] = S[i − 1]

else

S[i] = S[p(i)] ∪ {i}

Na¨ ıvely updating S[] takes O(n) time Total running time is O(n2) Using pointers running time can be improved to O(n).

Sariel (UIUC) CS473 35 Spring 2011 35 / 38

slide-86
SLIDE 86

Computing Solutions + First Attempt

Memoization + Recursion/Iteration allows one to compute the

  • ptimal value. What about the actual schedule?

M[0] = 0 S[0] is empty schedule

for i = 1 to n do

M[i] = max

  • w(vi) + M[p(i)], M[i − 1]
  • if w(vi) + M[p(i)] < M[i − 1] then

S[i] = S[i − 1]

else

S[i] = S[p(i)] ∪ {i}

Na¨ ıvely updating S[] takes O(n) time Total running time is O(n2) Using pointers running time can be improved to O(n).

Sariel (UIUC) CS473 35 Spring 2011 35 / 38

slide-87
SLIDE 87

Computing Implicit Solutions

Observation

Solution can be obtained from M[] in O(n) time, without any additional information

findSolution( j )

if (j = 0) then return empty schedule if (vj + M[p(j)] > M[j − 1]) then return findSolution(p(j)) ∪ {j} else return findSolution(j − 1)

Makes O(n) recursive calls, so findSolution runs in O(n) time.

Sariel (UIUC) CS473 36 Spring 2011 36 / 38

slide-88
SLIDE 88

Computing Implicit Solutions

A generic strategy for computing solutions in dynamic programming: keep track of the decision in computing the optimum value of a sub-problem. decision space depends on recursion

  • nce the optimum values are computed, go back and use the

decision values to compute an optimum solution. Question: What is the decision in computing M[i]? A: Whether to include i or not.

Sariel (UIUC) CS473 37 Spring 2011 37 / 38

slide-89
SLIDE 89

Computing Implicit Solutions

A generic strategy for computing solutions in dynamic programming: keep track of the decision in computing the optimum value of a sub-problem. decision space depends on recursion

  • nce the optimum values are computed, go back and use the

decision values to compute an optimum solution. Question: What is the decision in computing M[i]? A: Whether to include i or not.

Sariel (UIUC) CS473 37 Spring 2011 37 / 38

slide-90
SLIDE 90

Computing Implicit Solutions

M[0] = 0

for i = 1 to n do

M[i] = max(vi + M[p(i)], M[i − 1])

if (vi + M[p(i)] > M[i − 1])then

Decision[i] = 1 (* 1: i included in solution M[i] *)

else

Decision[i] = 0 (* 0: i not included in solution M[i] *) S = ∅, i = n

while (i > 0) do if (Decision[i] = 1) then

S = S ∪ {i} i = p(i)

else

i = i − 1

return S

Sariel (UIUC) CS473 38 Spring 2011 38 / 38

slide-91
SLIDE 91

Notes

Sariel (UIUC) CS473 39 Spring 2011 39 / 38

slide-92
SLIDE 92

Notes

Sariel (UIUC) CS473 40 Spring 2011 40 / 38

slide-93
SLIDE 93

Notes

Sariel (UIUC) CS473 41 Spring 2011 41 / 38

slide-94
SLIDE 94

Notes

Sariel (UIUC) CS473 42 Spring 2011 42 / 38