(* CSE 341, Spring 2013 *)
(* Lecture 8: Lexical Scope and Function Closures *)
(* lexical scope examples *)
(* first example *)
val x = 1
fun f y = x + y
val x = 2
val y = 3
val z = f (x + y)
(* second example *)
val x = 1
fun f y =
let
val x = y+1
in
fn z => x + y + z
end
val x = 3
val g = f 4 (* always adds 9 to its argument. always *)
val y = 5
val z = g 6
(* third example *)
fun f g =
let
val x = 3
in
g 2
end
val x = 4
fun h y = x + y (* always adds 4 to its argument. always *)
val z = f h
(* why lexical scope *)
(* f1 and f2 are always the same, no matter where the result is used *)
fun f1 y =
let
val x = y + 1
in
fn z => x + y + z
end
fun f2 y =
let
val q = y + 1
in
fn z => q + y + z
end
val x = 17 (* irrelevant *)
val a1 = (f1 7) 4
val a2 = (f2 7) 4
(* f3 and f4 are always the same, no matter what argument is passed in *)
fun f3 g =
let
val x = 3 (* irrelevant *)
in
g 2
end
fun f4 g =
g 2
val x = 17
val a3 = f3 (fn y => x + y)
val a4 = f3 (fn y => 17 + y)
(* under dynamic scope, the call "g 6" below would try to add a string
(from looking up x) and would have an unbound variable (looking up y),
even though f1 type-checked with type int -> (int -> int) *)
val x = "hi"
val g = f1 7
val z = g 4
(* Being able to pass closures that have free variables (private data)
makes higher-order functions /much/ more useful *)
fun filter (f,xs) =
case xs of
[] => []
| x::xs' => if f x then x::(filter(f,xs')) else filter(f,xs')
fun greaterThanX x = fn y => y > x
fun noNegatives xs = filter(greaterThanX ~1, xs)
fun allGreater (xs,n) = filter (fn x => x > n, xs)
fun allShorterThan1 (xs,s) =
filter (fn x => String.size x < (print "!"; String.size s), xs)
(* function bodies are evaluated when function is called, in an environment
where function was defined -- with expressions already bound to values,
i.e., results of computations *)
fun allShorterThan2 (xs,s) =
let
val i = (print "!"; String.size s)
in
filter(fn x => String.size x < i, xs)
end
val _ = print "\nwithAllShorterThan1: "
val x1 = allShorterThan1(["1","333","22","4444"],"xxx")
val _ = print "\nwithAllShorterThan2: "
val x2 = allShorterThan2(["1","333","22","4444"],"xxx")
val _ = print "\n"
(* Another hall-of-fame higher-order function *)
(* note this is "fold left" if order matters
can also do "fold right" *)
fun fold (f,acc,xs) =
case xs of
[] => acc
| x::xs' => fold (f,f(acc,x),xs')
(* examples not using private data *)
fun f5 xs = fold ((fn (x,y) => x+y), 0, xs)
fun f6 xs = fold ((fn (x,y) => x andalso y >= 0), true, xs)
(* examples using private data *)
fun f7 (xs,lo,hi) =
fold ((fn (x,y) =>
x + (if y >= lo andalso y <= hi then 1 else 0)),
0, xs)
fun f8 (xs,s) =
let
val i = String.size s
in
fold((fn (x,y) => x andalso String.size y < i), true, xs)
end
fun f9 (g,xs) = fold((fn(x,y) => x andalso g y), true, xs)
fun f8again (xs,s) =
let
val i = String.size s
in
f9(fn y => String.size y < i, xs)
end