Saturday, August 8, 2015

Chicken McNugget Problem

McNuggets

The problem is to find the largest number of McNuggets that cannot be bought in exact quantity, given that McNuggets come in packs of 6, 9 and 20. This is called the Frobenius Number, as described in this Numberphile video: How to order 43 Chicken McNuggets

First, we solved Problems 1 and 2 from this problem set: MIT OpenCourseWare Problem Set on Diophantine Equations


Theorem: If it is possible to buy x, x+1, ..., x+5 sets of McNuggets, for some x, then it is possible to buy any number of McNuggets >= x, given that McNuggets come in 6, 9 and 20 packs.


Implementing an exhaustive search for the Frobenius Number using the pseudocode outlined in the problem set was a bit cumbersome, because it requires multiple passes through the nested loops. Let's try a different approach.

The following python code will generate a list of all purchasable amounts with up to 50 coming from any one pack size. We have already shown that 50, 51, 52, 53, 54, 55, and all higher amounts are purchasable. This code is efficient, because it requires only one pass through the three nested loops.

In [1]:
bigList=[]
for Box6 in xrange(0,50,6):
    for Box9 in xrange(0,50,9):
        for Box20 in xrange(0,50,20):
            bigList.append(Box6+Box9+Box20)

print bigList
[0, 20, 40, 9, 29, 49, 18, 38, 58, 27, 47, 67, 36, 56, 76, 45, 65, 85, 6, 26, 46, 15, 35, 55, 24, 44, 64, 33, 53, 73, 42, 62, 82, 51, 71, 91, 12, 32, 52, 21, 41, 61, 30, 50, 70, 39, 59, 79, 48, 68, 88, 57, 77, 97, 18, 38, 58, 27, 47, 67, 36, 56, 76, 45, 65, 85, 54, 74, 94, 63, 83, 103, 24, 44, 64, 33, 53, 73, 42, 62, 82, 51, 71, 91, 60, 80, 100, 69, 89, 109, 30, 50, 70, 39, 59, 79, 48, 68, 88, 57, 77, 97, 66, 86, 106, 75, 95, 115, 36, 56, 76, 45, 65, 85, 54, 74, 94, 63, 83, 103, 72, 92, 112, 81, 101, 121, 42, 62, 82, 51, 71, 91, 60, 80, 100, 69, 89, 109, 78, 98, 118, 87, 107, 127, 48, 68, 88, 57, 77, 97, 66, 86, 106, 75, 95, 115, 84, 104, 124, 93, 113, 133]

Unfortunately, this list is a mess. It is unsorted and contains duplicates. Fortunately, the list can be easily sorted:

In [2]:
bigList.sort()
print bigList
[0, 6, 9, 12, 15, 18, 18, 20, 21, 24, 24, 26, 27, 27, 29, 30, 30, 32, 33, 33, 35, 36, 36, 36, 38, 38, 39, 39, 40, 41, 42, 42, 42, 44, 44, 45, 45, 45, 46, 47, 47, 48, 48, 48, 49, 50, 50, 51, 51, 51, 52, 53, 53, 54, 54, 55, 56, 56, 56, 57, 57, 57, 58, 58, 59, 59, 60, 60, 61, 62, 62, 62, 63, 63, 64, 64, 65, 65, 65, 66, 66, 67, 67, 68, 68, 68, 69, 69, 70, 70, 71, 71, 71, 72, 73, 73, 74, 74, 75, 75, 76, 76, 76, 77, 77, 77, 78, 79, 79, 80, 80, 81, 82, 82, 82, 83, 83, 84, 85, 85, 85, 86, 86, 87, 88, 88, 88, 89, 89, 91, 91, 91, 92, 93, 94, 94, 95, 95, 97, 97, 97, 98, 100, 100, 101, 103, 103, 104, 106, 106, 107, 109, 109, 112, 113, 115, 115, 118, 121, 124, 127, 133]

Duplicates can be removed by converting the list to a set, since sets contain no duplicates.

In [3]:
bigSet=set(bigList)
print bigSet
set([0, 6, 9, 12, 15, 18, 20, 21, 24, 26, 27, 29, 30, 32, 33, 35, 36, 38, 39, 40, 41, 42, 44, 45, 46, 47, 48, 49, 50, 51, 52, 53, 54, 55, 56, 57, 58, 59, 60, 61, 62, 63, 64, 65, 66, 67, 68, 69, 70, 71, 72, 73, 74, 75, 76, 77, 78, 79, 80, 81, 82, 83, 84, 85, 86, 87, 88, 89, 91, 92, 93, 94, 95, 97, 98, 100, 101, 103, 104, 106, 107, 109, 112, 113, 115, 118, 121, 124, 127, 133])

This is the set of purchasable amounts, but we really want the set of unpurchasable amounts. We can take advantage of Python Set Operations by subtracting this set from a complete set of numbers of 0 through 49.

In [4]:
completeList = range(50)
completeSet = set(completeList)
print completeSet
set([0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46, 47, 48, 49])

Putting everything together...
In [5]:
bigList=[]                                   # create an empty list
for Box6 in xrange(0,50,6):                  # loop over all possible purchasable amounts from a pack of 6
    for Box9 in xrange(0,50,9):              # loop over all possible purchasable amounts from a pack of 9
        for Box20 in xrange(0,50,20):        # loop over all possible purchasable amounts from a pack of 20
            bigList.append(Box6+Box9+Box20)  # put all purchasable amounts found into the list
                    
bigList.sort()                               # sort the big ugly list
bigSet=set(bigList)                          # remove duplicates by converting to a set

completeList=range(50)                       # generate a simple list 0,1,2,...,49
completeSet=set(completeList)                # convert simple list to a set

print completeSet-bigSet                     # print members of completeSet not in bigSet
set([1, 2, 3, 4, 5, 7, 8, 10, 11, 13, 14, 16, 17, 19, 22, 23, 25, 28, 31, 34, 37, 43])

So this is our final list of unpurchasable amounts. The largest unpurchasable amount is 43.


Additional Resources