1 Functional Programming and ML. 2 What’s wrong with Imperative Languages? State State Introduces context sensitivity Introduces context sensitivity Harder.

Slides:



Advertisements
Similar presentations
More ML Compiling Techniques David Walker. Today More data structures lists More functions More modules.
Advertisements

Modern Programming Languages, 2nd ed.
ML Lists.1 Standard ML Lists. ML Lists.2 Lists  A list is a finite sequence of elements. [3,5,9] ["a", "list" ] []  Elements may appear more than once.
A First Look at ML.
CSE341: Programming Languages Lecture 2 Functions, Pairs, Lists Dan Grossman Winter 2013.
CMSC 330: Organization of Programming Languages Tuples, Types, Conditionals and Recursion or “How many different OCaml topics can we cover in a single.
ML Lists.1 Standard ML Lists. ML Lists.2 Lists  A list is a finite sequence of elements. [3,5,9] ["a", "list" ] []  ML lists are immutable.  Elements.
ML Datatypes.1 Standard ML Data types. ML Datatypes.2 Concrete Datatypes  The datatype declaration creates new types  These are concrete data types,
CSE 3341/655; Part 4 55 A functional program: Collection of functions A function just computes and returns a value No side-effects In fact: No program.
1 Programming Languages (CS 550) Lecture Summary Functional Programming and Operational Semantics for Scheme Jeremy R. Johnson.
CSE341: Programming Languages Lecture 16 Datatype-Style Programming With Lists or Structs Dan Grossman Winter 2013.
Getting started with ML ML is a functional programming language. ML is statically typed: The types of literals, values, expressions and functions in a.
Functional Programming. Pure Functional Programming Computation is largely performed by applying functions to values. The value of an expression depends.
Lambda Calculus and Lisp PZ03J. Lambda Calculus The lambda calculus is a model for functional programming like Turing machines are models for imperative.
ML: a quasi-functional language with strong typing Conventional syntax: - val x = 5; (*user input *) val x = 5: int (*system response*) - fun len lis =
Chapter ElevenModern Programming Languages1 A Fourth Look At ML.
A First Look at ML Chapter FiveModern Programming Languages, 2nd ed.1.
Patterns in ML functions. Formal vs. actual parameters Here's a function definition (in C): –int add (int x, int y) { return x + y; } –x and y are the.
INF 212 ANALYSIS OF PROG. LANGS Type Systems Instructors: Crista Lopes Copyright © Instructors.
5/11/2015IT 3271 Types in ML (Ch 11) datatype bool = true | false; datatype 'element list = nil | :: of 'element * 'element list n Predefined, but not.
Functional Design and Programming Lecture 1: Functional modeling, design and programming.
ML: a quasi-functional language with strong typing Conventional syntax: - val x = 5; (*user input *) val x = 5: int (*system response*) - fun len lis =
Chapter 15 Other Functional Languages. Copyright © 2007 Addison-Wesley. All rights reserved. Functional Languages Scheme and LISP have a simple syntax.
Introduction to ML - Part 2 Kenny Zhu. What is next? ML has a rich set of structured values Tuples: (17, true, “stuff”) Records: {name = “george”, age.
Introduction to ML Last time: Basics: integers, Booleans, tuples,... simple functions introduction to data types This time, we continue writing an evaluator.
Encapsulation by Subprograms and Type Definitions
Introduction to ML A Quasi-Functional Language With Strong Typing And Type Inference.
Type Checking  Legality checks  Operator determination  Overload resolution.
Type Inference: CIS Seminar, 11/3/2009 Type inference: Inside the Type Checker. A presentation by: Daniel Tuck.
Introduction to ML A Quasi-Functional Language With Strong Typing And Type Inference.
CS 2104 : Prog. Lang. Concepts. Functional Programming I Lecturer : Dr. Abhik Roychoudhury School of Computing From Dr. Khoo Siau Cheng’s lecture notes.
CS235 Languages and Automata Department of Computer Science Wellesley College Introduction to Standard ML Wednesday, September 23, 2009 Reading: Beginning.
PrasadCS7761 Haskell Data Types/ADT/Modules Type/Class Hierarchy Lazy Functional Language.
0 REVIEW OF HASKELL A lightening tour in 45 minutes.
ISBN Chapter 15 Functional Programming Languages.
Introduction to Functional Programming and ML CS 331 Principles of Programming Languages.
CSC 580 – Theory of Programming Languages, Spring, 2009 Week 9: Functional Languages ML and Haskell, Dr. Dale E. Parson.
Chapter 9: Functional Programming in a Typed Language.
Functional Programming With examples in F#. Pure Functional Programming Functional programming involves evaluating expressions rather than executing commands.
A Second Look At ML 1. Outline Patterns Local variable definitions A sorting example 2.
CS 2104 – Prog. Lang. Concepts Functional Programming II Lecturer : Dr. Abhik Roychoudhury School of Computing From Dr. Khoo Siau Cheng’s lecture notes.
Error Example - 65/4; ! Toplevel input: ! 65/4; ! ^^ ! Type clash: expression of type ! int ! cannot have type ! real.
Chapter SevenModern Programming Languages1 A Second Look At ML.
A FIRST LOOK AT ML Chapter Five Modern Programming Languages 1.
1 FP Foundations, Scheme In Text: Chapter Chapter 14: FP Foundations, Scheme Mathematical Functions Def: A mathematical function is a mapping of.
ISBN Chapter 15 Functional Programming Languages.
Cs776(Prasad)L6sml971 SML-97 Specifics SML/NJ 110.
1.SML Docs Standard Basis 2.First-Class Functions Anonymous Style Points Higher-Order 3.Examples Agenda.
1 Objective Caml (Ocaml) Aaron Bloomfield CS 415 Fall 2005.
Cs776 (Prasad)L2HOF1 Higher-Order Functions. cs776 (Prasad)L2HOF2 Higher-Order Functions A function that takes a function as argument and/or returns a.
Functional Programming
Functional Programming
Principles of programming languages 12: Functional programming
ML: a quasi-functional language with strong typing
CSE341: Programming Languages Lecture 6 Nested Patterns Exceptions Tail Recursion Dan Grossman Spring 2017.
ML Again ( Chapter 7) Patterns Local variable definitions
A lightening tour in 45 minutes
Closures and Streams cs784(Prasad) L11Clos
Objective caml Daniel Jackson MIT Lab for Computer Science 6898: Advanced Topics in Software Design March 18, 2002.
FP Foundations, Scheme In Text: Chapter 14.
CSE341: Programming Languages Lecture 6 Nested Patterns Exceptions Tail Recursion Dan Grossman Spring 2013.
CSE341: Programming Languages Lecture 6 Nested Patterns Exceptions Tail Recursion Dan Grossman Autumn 2018.
CSCE 314: Programming Languages Dr. Dylan Shell
CSE-321 Programming Languages Introduction to Functional Programming
CSE341: Programming Languages Lecture 6 Nested Patterns Exceptions Tail Recursion Zach Tatlock Winter 2018.
Abstraction and Repetition
CSE341: Programming Languages Lecture 6 Nested Patterns Exceptions Tail Recursion Dan Grossman Spring 2016.
CSE341: Programming Languages Lecture 6 Nested Patterns Exceptions Tail Recursion Dan Grossman Spring 2019.
Brett Wortzman Summer 2019 Slides originally created by Dan Grossman
CSE341: Programming Languages Lecture 6 Nested Patterns Exceptions Tail Recursion Dan Grossman Autumn 2017.
Presentation transcript:

1 Functional Programming and ML

2 What’s wrong with Imperative Languages? State State Introduces context sensitivity Introduces context sensitivity Harder to reuse functions in different context Harder to reuse functions in different context Easy to develop inconsistent state Easy to develop inconsistent state int balance = account.getBalance; balance += deposit; // Now there are two different values stored in two different places Sequence of function calls may change behavior of a function Sequence of function calls may change behavior of a function Oh, didn’t you know you have to call C.init() before you… Oh, didn’t you know you have to call C.init() before you… Lack of Referential Transparency Lack of Referential Transparency These issues can make imperative programs hard to understand These issues can make imperative programs hard to understand

3 What is functional programming? A style of programming that avoids the use of assignments A style of programming that avoids the use of assignments Similar to the way structured programming avoided the use of goto statements Similar to the way structured programming avoided the use of goto statements No Assignment, No variables No Assignment, No variables val a = 3; -- a named constant, initialized to 3 val a = 3; -- a named constant, initialized to 3 State changes only in predictable ways State changes only in predictable ways Bindings may be created and destroyed Bindings may be created and destroyed Values associated with bindings don’t change Values associated with bindings don’t change Referential Transparency Referential Transparency Easier to understand programs Easier to understand programs Easier to reuse parts of programs Easier to reuse parts of programs

4 Some Sums x is a vector of integers Imperative Describes how to calculate result Describes how to calculate result Iterator it = x.iterator(); int result = 0; while(it.hasNext()) { result += it.next(); result += it.next();}Functional Defines what the result is Defines what the result is function sum [] = 0 | sum (x::xs) = x + (sum xs) | sum (x::xs) = x + (sum xs)+/x

5 History Developed at Edinburgh University (Scotland) by Robin Milner & others in the late 1970’s Developed at Edinburgh University (Scotland) by Robin Milner & others in the late 1970’s A M eta L anguage for theorem proving A M eta L anguage for theorem proving SML is Standard ML – widely used (except in France, where they use CAML) SML is Standard ML – widely used (except in France, where they use CAML)

6 SML Implementations Standard ML of New Jersey can be found on the SMLNJ web site, Standard ML of New Jersey can be found on the SMLNJ web site, Poly/ML is at Poly/ML is at

7 ML: a quasi-functional language with strong typing Conventional syntax: Conventional syntax: > val x = 5; (*user input *) > val x = 5; (*user input *) val x = 5: int (*system response*) val x = 5: int (*system response*) > fun len lis = if (null lis) then 0 else 1 + len (tl lis); val len = fn : ‘a list -> int > fun len lis = if (null lis) then 0 else 1 + len (tl lis); val len = fn : ‘a list -> int Type inference for local entities Type inference for local entities > x * x * x; > x * x * x; val it = 125: int (* it denotes the last computation*) val it = 125: int (* it denotes the last computation*)

8 ML’s Imperative Features Reference types: val p = ref 5 Reference types: val p = ref 5 Dereferencing syntax: !p + 1 Dereferencing syntax: !p + 1 Changes state Changes state Statements Statements if E then E 1 else E 2 – an expression if E then E 1 else E 2 – an expression while E do E 1 – iteration implies state change while E do E 1 – iteration implies state change Each E 1 E 2 may change state Each E 1 E 2 may change state Avoid these Avoid these

9 Tuples Ordered lists of elements Ordered lists of elements Denoted by comma separated list enclosed in parenthesis Denoted by comma separated list enclosed in parenthesis (a, b) is a two-tuple, or pair of type int * int (a, b) is a two-tuple, or pair of type int * int (1, 2, 3) is a 3-tuple of type int * int * int (1, 2, 3) is a 3-tuple of type int * int * int Elements may be of any type, including other tuples Elements may be of any type, including other tuples > (“hi”, 1.0, 2, (0, 0)); val it : string * real * int * (int * int)

10 Records Records are tuples in which the components – called fields – are named Records are tuples in which the components – called fields – are named Records are denoted by a comma separated list of name value bindings enclosed in curly braces Records are denoted by a comma separated list of name value bindings enclosed in curly braces {name = “Jones”, age = 25, salary = 65000} We can define an abstract record type: We can define an abstract record type: > type emp = {name : string, age : int, sal : int}; type emp > fun getSal (e : emp) = #sal e; val getSal = fn : emp -> int

11 Lists A list is a sequence of elements, all of which have the same type A list is a sequence of elements, all of which have the same type Lists are denoted by a sequence of comma separated elements enclosed in square brackets: Lists are denoted by a sequence of comma separated elements enclosed in square brackets: > [1, 2, 3, 4] val it = [1, 2, 3, 4] : int list Similar to LISP, with conventional syntax: Similar to LISP, with conventional syntax: hd, tl, :: instead of car, cdr, cons for head, tail and concatenate element > fun append (x, y) = if null (x) then y > fun append (x, y) = if null (x) then y else hd (x) :: append (tl (x), y); else hd (x) :: append (tl (x), y); val append = fn: ‘a list * ‘a list -> ‘a list val append = fn: ‘a list * ‘a list -> ‘a list (* a function that takes a pair of lists and yields a list *) (* a function that takes a pair of lists and yields a list *) ‘a is a type variable ‘a is a type variable

12 Patterns fun append (x, y) = if null (x) then y else hd (x) :: append (tl (x), y); else hd (x) :: append (tl (x), y); Now, with patterns Now, with patterns > fun append ([], y) = y | append (x::xs, y) = x::append(xs,y); | append (x::xs, y) = x::append(xs,y); val append = fn : 'a list * 'a list -> 'a list Clearly expresses basis and recursive parts of a recursive definition of append Clearly expresses basis and recursive parts of a recursive definition of append append ([1,2,3],[4,5,6]); val it = [1, 2, 3, 4, 5, 6] : int list

13 Patterns help replace Imperative statements in Functional Programs Selection Selection if x = 0 then y = 1 else y = 2 fun fif(0) = 1 | fif(-) = 2; | fif(-) = 2; Loops generally handled by (tail) recursion Loops generally handled by (tail) recursion fun sum [] = 0 sum (x::xs) = x + (sum xs) sum (x::xs) = x + (sum xs)

14 Currying: partial bindings Curried functions take one argument Curried functions take one argument a b c means ((a b) c) (* parentheses are lisp notation*) a is a function (a b) yields another function that is applied to c > fun add x y = x + y; > fun add x y = x + y; val add = fn : int -> int -> int > add 2; > add 2; val it = fn : int -> int val it = fn : int -> int > it 3; val it = 5 : int val it = 5 : int Keep in mind: Keep in mind: add 2 2 (* Curried function *) add 2 2 (* Curried function *) add (2, 2) (* function takes one tuple *) add (2, 2) (* function takes one tuple *)

15 Even Better than Sums fun reduce f i [] = i | reduce f i (x::xs) = f x (reduce f i xs); | reduce f i (x::xs) = f x (reduce f i xs); fun add a b = a + b > reduce add 0 [2, 3, 4]; val it = 9 : int fun times a b = a * b > reduce times 1 [2, 3, 4]; val it = 24 : int fun timesReduce x = reduce times 0 > timesReduce [2, 3, 4]; val it = 24 : int

16 Let declarations Let declarations bind a value with a name over an explicit scope Let declarations bind a value with a name over an explicit scope fun fib 0 = 0 | fib n = let fun fibb (x, prev, curr) = if x=1 then curr | fib n = let fun fibb (x, prev, curr) = if x=1 then curr else fibb (x-1, curr, prev + curr) else fibb (x-1, curr, prev + curr)in fibb(n, 0, 1) fibb(n, 0, 1) end; val fib = fn : int -> int val fib = fn : int -> int > fib 20; > fib 20; val it = 6765 : int val it = 6765 : int

17 The type system Primitive types: int, char, real, string Primitive types: int, char, real, string Constructors: list, array, product (record), function Constructors: list, array, product (record), function an expression has a corresponding type expression an expression has a corresponding type expression the interpreter builds the type expression for each input the interpreter builds the type expression for each input type checking requires that type expression of functions and their arguments match, and that type expression of context match result of function type checking requires that type expression of functions and their arguments match, and that type expression of context match result of function

18 Type inference > fun incr x = x + 1; > fun incr x = x + 1; val incr = fn : int -> int val incr = fn : int -> int because of its appearance in (x+1), x must be integer because of its appearance in (x+1), x must be integer > fun add2 x = incr (incr x); > fun add2 x = incr (incr x); val add2 = fn : int -> int val add2 = fn : int -> int incr returns an integer, so add2 does as well incr returns an integer, so add2 does as well x is argument of incr, so must be integer x is argument of incr, so must be integer val wrong = incr 7; val wrong = incr 7; Error: operator and operand don’t agree Error: operator and operand don’t agree

19 Polymorphism > fun len x = if null x then 0 > fun len x = if null x then 0 = else 1 + len (tl x); = else 1 + len (tl x); works for any kind of list. What is its type expression? works for any kind of list. What is its type expression? val len : fn = ‘a list -> int val len : fn = ‘a list -> int ‘a denotes a type variable. Implicit universal quantification: ‘a denotes a type variable. Implicit universal quantification: for any a, len applies to a list of a’s. for any a, len applies to a list of a’s. > fun copy lis = if null lis then nil > fun copy lis = if null lis then nil = else hd (lis) :: copy (tl lis); = else hd (lis) :: copy (tl lis);

20 Type inference and unification Type expressions are built by solving a set of equations Type expressions are built by solving a set of equations Substituting type expressions for type variables Substituting type expressions for type variables > fun foo [] = 0 > fun foo [] = 0 | foo (x::xs) = x + (foo xs); | foo (x::xs) = x + (foo xs); foo : ‘a -> ‘b ? ‘b must be int, since result of foo for an empty list is an int and the result of foo is also an operand of + ‘b must be int, since result of foo for an empty list is an int and the result of foo is also an operand of + ‘a must be int list, since x is also operand of + ‘a must be int list, since x is also operand of + val foo = fn : int list -> int val foo = fn : int list -> int

21 Unification algorithm A type variable can be unified with another variable A type variable can be unified with another variable ‘a unifies with ‘b => ‘a and ‘b are the same ‘a unifies with ‘b => ‘a and ‘b are the same A type variable can be unified with a constant A type variable can be unified with a constant ‘a unifies with int => all occurences of ‘a mean int ‘a unifies with int => all occurences of ‘a mean int A type variable can be unified with a type expression A type variable can be unified with a type expression ‘a unifies with ‘b list ‘a unifies with ‘b list ‘a does not unify with ‘a list ‘a does not unify with ‘a list A constant can be unified with itself int is int A constant can be unified with itself int is int An expression can be unified with another expression if the constructors are identical and if the arguments can be unified: An expression can be unified with another expression if the constructors are identical and if the arguments can be unified: (int -> int) list unifies with ‘a list, ‘a is a function on integers (int -> int) list unifies with ‘a list, ‘a is a function on integers

22 Type system does not handle overloading well > fun plus x y = x + y; > fun plus x y = x + y; operator is overloaded, cannot be resolved from context (error in some versions, defaults to int in others) operator is overloaded, cannot be resolved from context (error in some versions, defaults to int in others) Can use explicit typing to select interpretation: Can use explicit typing to select interpretation: > fun mix (x, y,z) = x * y + z : real; > fun mix (x, y,z) = x * y + z : real; val mix = fn : (real * real * real) -> real val mix = fn : (real * real * real) -> real

23 Parametric polymorphism vs. generics A function whose type expression has type variables applies to an infinite set of types. A function whose type expression has type variables applies to an infinite set of types. Equality of type expressions means structural equivalence. Equality of type expressions means structural equivalence. All applications of a polymorphic function use the same body: no need to instantiate. All applications of a polymorphic function use the same body: no need to instantiate. > let val ints = [1, 2, 3]; > let val ints = [1, 2, 3]; = val strs = [“this”, “that”]; = val strs = [“this”, “that”]; = in = in = len ints + len strs (* int list -> int, string list -> int *) = len ints + len strs (* int list -> int, string list -> int *) = end; = end; val it = 5: int val it = 5: int

24 User-defined types and inference A user-defined type introduces constructors: A user-defined type introduces constructors: datatype tree = leaf of int | node of tree * tree datatype tree = leaf of int | node of tree * tree leaf and node are type constructors leaf and node are type constructors > fun sum (leaf (t)) = t > fun sum (leaf (t)) = t | sum (node (t1, t2)) = sum t1 + sum t2; | sum (node (t1, t2)) = sum t1 + sum t2; val sum = fn : tree -> int val sum = fn : tree -> int

25 Parameterized Data Types > fun flatten (leaf (t)) = [t] > fun flatten (leaf (t)) = [t] | flatten (node (t1, t2)) = flatten flatten (t2); | flatten (node (t1, t2)) = flatten flatten (t2); val flatten = fn : tree -> int list val flatten = fn : tree -> int list > datatype ‘a gentree = leaf of ‘a > datatype ‘a gentree = leaf of ‘a | node of ‘a gentree * ‘a gentree; | node of ‘a gentree * ‘a gentree; > val names = node (leaf (“this”), leaf (“that”)); > val names = node (leaf (“this”), leaf (“that”)); val names = … string gentree val names = … string gentree