import List
grid = The 20x20 grid as a list of lists.
myMaximum :: (Ord t) => [t] -> t
myMaximum [] = 0
myMaximum xs = maximum xs
takeBy :: Int -> [a] -> [[a]]
takeBy n = filter ((n==) . length) . map (take n) . tails
maxBy :: (Num t, Ord t) => Int -> [t] -> t
maxBy n = myMaximum . map product . takeBy n
maxInRows :: (Num t, Ord t) => Int -> [[t]] -> t
maxInRows n = maximum . map (maxBy n)
maxInCols :: (Num t, Ord t) => Int -> [[t]] -> t
maxInCols n = maxInRows n . transpose
maxInLRupper :: (Num t, Ord t) => Int -> [[t]] -> t
maxInLRupper n = maxInRows n . transpose . zipWith drop [0..]
maxInLRlower :: (Num t, Ord t) => Int -> [[t]] -> t
maxInLRlower n = maxInRows n . transpose . zipWith drop [0..] . transpose
maxInRLupper :: (Num t, Ord t) => Int -> [[t]] -> t
maxInRLupper n = maxInLRupper n . map reverse
maxInRLlower :: (Num t, Ord t) => Int -> [[t]] -> t
maxInRLlower n = maxInLRlower n . map reverse
findMaxProduct :: (Num t, Ord t) => [[t]] -> Int -> t
findMaxProduct xs n = maximum $ map (flip ($ n) xs) [maxInRows, maxInCols,
maxInLRupper, maxInLRlower, maxInRLupper, maxInRLlower]
main = findMaxProduct grid 4
We will take a look at the code line by line (note that the blank lines in between functions are important in Haskell, as they indicate that the definition of the function is over). The only change I made to the code was to add the function types before the function definitions, since I think it is good practice to do so (especially if trying to understand someone else's code). Also, note that it is a little awkward to read this code top-down, since it starts with the helper functions that give the solution in the end. When writing in Haskell, I usually find myself writing the code in the opposite order.
The code import List brings in Data.List which handles some of the more advanced list management tools (see the description here).
The function myMaximum just returns the maximum of a nonempty list, or zero if the list is empty. Note that when Haskell calls myMaximum on a list, it calls the first version of the function that matches the pattern in the order they were listed in the source file.
The next line is a doozy, and there is a lot going on here, so bear with me:
takeBy :: Int -> [a] -> [[a]]
takeBy n = filter ((n==) . length) . map (take n) . tails
Looking at the function declaration, one would be tempted to say that takeBy takes two arguments, an Int and a list of as. However, note that we have only specified one input variable, effectively making this function return a function with input [a] and output [[a]].
So what does this function do? The '.' denotes function composition in Haskell, so this function applies tails, then map (take n), and then filter ((n==) . length) to a list, and returns a list of lists.
The function tails returns all final segments of the argument, longest first. For example, tails [1,2,3] would return [[1,2,3],[2,3],[3],[]].
The function map is one of the mainstays of Haskell. It is of type (a -> b) -> [a] -> [b], which means it takes a function turning type a objects to type b objects, a list of a objects, and produces a list of b objects; it just applies the function given to each of the elements of the given list to produce a new list. The map allows one to perform some iterative tasks in a simple way. So, in our example, we are applying (take n) to each element of the list returned from tails, which in effect, takes the first n elements of each of the final segments of the list input. For example, one has
map (take 4) . tails "abcdefg" = ["abcd","bcde","cdef","defg","efg","fg","g",""]
In this problem, we are trying to say something about the maximal product of all length 4 'words' in the 20x20 grid. We clearly have some that are too short in this example, so we need to take off the ones that are not length 4. That is where the next function comes in. The function filter is of type (a -> Bool) -> [a] -> [a] and simply selects those elements in the list that evaluate to True when passed to the first argument. The function we are concerned with is ((n==).length), which is of type [a] -> Bool and first takes the length (of type [a] -> Int) and then checks if that number is equal to n. So, takeBy selects the initial segments of length n of the final segments of the list that is passed to it.
Now that we have slogged through that line, the rest will be easier. The next function is
maxBy :: (Ord t) => Int -> [t] -> t
maxBy n = myMaximum . map product . takeBy n
This function applies (takeby n) to a list, takes the product over all those lists with map product, and then finds the maximum of all those values with myMaximum. The next two functions are pretty simple:
maxInRows :: (Num t, Ord t) => Int -> [[t]] -> t
maxInRows n = maximum . map (maxBy n)
maxInCols :: (Num t, Ord t) => Int -> [[t]] -> t
maxInCols n = maxInRows n . transpose
The function maxInRows maps the (maxBy n) function over each row, and then takes the maximum of all those values in the resulting list. The function maxInCols uses the useful function transpose makes the rows in the grid columns and the columns rows. A nice thing about this function is that it works for non-square grids, and we will use this to our advantage.
The next two functions include an ingenious use of the function zipWith
maxInLRupper :: (Num t, Ord t) => Int -> [[t]] -> t
maxInLRupper n = maxInRows n . transpose . zipWith drop [0..]
maxInLRlower :: (Num t, Ord t) => Int -> [[t]] -> t
maxInLRlower n = maxInRows n . transpose . zipWith drop [0..] . transpose
maxInRLupper :: (Num t, Ord t) => Int -> [[t]] -> t
maxInRLupper n = maxInLRupper n . map reverse
maxInRLlower :: (Num t, Ord t) => Int -> [[t]] -> t
maxInRLlower n = maxInLRlower n . map reverse
The functions (maxInRows n) and transpose are easy enough to understand. The function zipWith is of type (a -> b -> c) -> [a] -> [b] -> [c]; it applies the function passed to it to an element each from the lists [a] and [b], producing a list [c]. If one list is longer than the other, zipWith automatically stops once one of the lists is exhausted. For example, one has the following:
zipWith (+) [1..10] [1..] = [2,4,6,8,10,12,14,16,18,20]
So, the function maxInLRupper n applies zipWith drop [0..] to an input grid, which just takes the 'upper left hand corner' of the grid by dropping the first few elements. In other words, one has
zipWith drop [0..] [[1,2,3],[4,5,6],[7,8,9]] = [[1,2,3],[5,6],[9]]
In maxInLRupper, we transpose the grid to get the left to right downward diagonals, and then compute maxInRows on that resulting list. To find those in the lower left hand corner, we apply an extra transpose before performing the computation. To get the downward to the left diagonals, we map reverse on the grid to reverse each row, and then use the previous functions to find the maximums on the diagonals.
findMaxProduct :: (Num t, Ord t) => [[t]] -> Int -> t
findMaxProduct xs n = maximum $ map (flip ($ n) xs) [maxInRows, maxInCols,
maxInLRupper, maxInLRlower, maxInRLupper, maxInRLlower]
I am still working on wrapping my head around the last function. What the command says is easy enough: map the function (flip ($ n) xs) to the elements of the list (which is a list of functions), and then take the maximum of the resulting list. However, we have to understand what the command flip ($ 4) xs maxInRows does.
The function flip takes a function of type a -> b -> c and returns a function that behaves the same, but is of type b -> a -> c. So, below is how this works:
flip ($ 4) xs maxInRows = ($ 4) maxInRows xs
= maxInRows 4 xs
The first equality is by the definition of flip, and the last equality is by definition of the function ($ 4). The operator ($) is shorthand for function evaluation and is useful to reduce the proliferation of parenthesis. In this case, ($ a) applied to a function f of type a -> b evaluates f at 4.
Phew! I am still trying to wrap my head around this last bit, but I know that I learned a lot working through this code.
By the way, after I posted this I realized I completely omitted the discussion on the stuff before the => in the function definitions. That will be the next post.
No comments:
Post a Comment