We talked about a technique to recursively processing list elements once at a time in which you have to focus on two things:
- How to process an empty list (the base case of the recursion)
- How to process a non empty list in terms of the result of processing the tail of the list (the list except the first element)
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
fun mapUsingListFunctions(f, ls) = | |
if null ls | |
then [] | |
else f(hd ls)::mapUsingListFunctions(f, tl ls) | |
val test = mapUsingListFunctions( | |
(fn x => x * x), | |
[1, 2, 3]) | |
(* val test = [1,4,9] : int list *) |
It uses the following list functions:
- hd to extract the head of the list (its first element)
- tl to extract the tail of the list (the list except the first element)
- null that returns true if the list is empty and false otherwise
We can use a similar approach to write map in Racket:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
#lang racket | |
(define (map-using-list-functions f ls) | |
(if (empty? ls) | |
'() | |
(cons (f (first ls)) | |
(map-using-list-functions f (rest ls))))) | |
(map-using-list-functions | |
(lambda (x) (* x x)) | |
'(1 2 3)) | |
; '(1 4 9) |
Here the name of the functions are a bit more expressive: first, rest and empty?.
The cons function adds an element at the beginning of a list.
A more idiomatic way to write map in SML would be using pattern matching:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
fun mapUsingPatternMatching(f, ls) = | |
case ls of | |
[] => [] | |
| head::tail => f(head)::mapUsingPatternMatching(f, tail) | |
val test = mapUsingPatternMatching( | |
(fn x => x * x), | |
[1, 2, 3]) | |
(* val test = [1,4,9] : int list *) |
This version is using the same approach but pattern matching makes the code more concise and expressive.
As a bonus here is a beautiful way to append two lists in SML using the same technique and pattern matching:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
fun append(xs, ys) = | |
case xs of | |
[] => ys | |
| head::tail => head::append(tail, ys) | |
val testAppend1 = append([1, 2, 3, 4], [5, 6, 7]) | |
(* val testAppend1 = [1,2,3,4,5,6,7] : int list *) | |
val testAppend2 = append([], [5, 6, 7]) | |
(* val testAppend2 = [5,6,7] : int list *) | |
val testAppend3 = append([1, 2, 3, 4], []) | |
(* val testAppend3 = [1,2,3,4] : int list *) |
This is very well explained in the first unit of of the Programming Languages course from Washington University at Coursera that I completed last year.
There is going to be a new edition starting up Oct 2. I cannot recommend it highly enough.
I did a kata with Erlang using this technique:
ReplyDeletehttps://github.com/tooky/vowels-erlang/commit/da979ee7f6b0311be51ef987454500fbdb0013f2
Thanks for sharing!