Slide 1

Slide 1 text

EFFECTING PURE CHANGE HOW ANYTHING EVER GETS DONE IN FP ANUPAM JAIN

Slide 2

Slide 2 text

2 Hello!

Slide 3

Slide 3 text

3 ❑ MEETUP: https://www.meetup.com/DelhiNCR-Haskell-And- Functional-Programming-Languages-Group ❑ TELEGRAM: https://t.me/fpncr ❑ GITHUB: https://github.com/fpncr ❑ SLACK: https://functionalprogramming.slack.com #FPNCR FPNCR

Slide 4

Slide 4 text

4 Effects

Slide 5

Slide 5 text

5 ❑ Programming with Mathematical Functions. ❑ A Function has Input and Output. ❑ Referentially Transparent ❑ A complete Program is just a function! Functional Programming

Slide 6

Slide 6 text

6 ❑ No Global State ❑ No Assignments ❑ No Statements ❑ No Input-Output Referential Transparency

Slide 7

Slide 7 text

7 ❑ Easier to Reason About ❑ Easier to Optimize ❑ Easier to Parallelize ❑ Easier to Test ❑ ??? Why?

Slide 8

Slide 8 text

8 Side Effects !!

Slide 9

Slide 9 text

9 What Happens If we Ignore the Danger

Slide 10

Slide 10 text

10 write :: String -> () read :: () -> String What could go wrong

Slide 11

Slide 11 text

11 write "Do you want a pizza?" if (read() == "Yes") orderPizza() write "Should I launch missiles?" if (read() == "Yes") launchMissiles() What could go wrong PSEUDO-CODE

Slide 12

Slide 12 text

12 write "Do you want a pizza?" if (read() == "Yes") orderPizza() write "Should I launch missiles?" if (read() == "Yes") launchMissiles() What could go wrong May be optimized away

Slide 13

Slide 13 text

13 write "Do you want a pizza?" if (read() == "Yes") orderPizza() write "Should I launch missiles?" if (read() == "Yes") launchMissiles() What could go wrong May reuse value from previous read()

Slide 14

Slide 14 text

14 OCAML let val ha = (print "ha") in ha; ha end HASKELL let ha = putStr “ha” in ha >> ha Referential Transparency is Vital ha haha

Slide 15

Slide 15 text

15 Taming Side Effects

Slide 16

Slide 16 text

16 ❑ Separate “Effects” from “Values” (Hint: Strong Types are great for this) ❑ Effects are forever. No way to recover a “pure” Value. ❑ As much as possible, tag the flavor of side effects. “Print” cannot launch nukes. Contain. Not Prohibit.

Slide 17

Slide 17 text

17 Effectful Logic Pure Logic Outside World

Slide 18

Slide 18 text

18 ❑ React – Functional Views ❑ The Elm Architecture ❑ Functional Reactive Programming ❑ Monads and Algebraic Effects Examples

Slide 19

Slide 19 text

19 render() { let n = this.state.count return {n} } React – Functional Views

Slide 20

Slide 20 text

20 view address model = button [onClick address Increment] [text (toString model)] update action model = case action of Increment -> model + 1 The Elm Architecture

Slide 21

Slide 21 text

21 ❑ Interface between things that are static and those that vary ❑ Forms a graph of relationships between varying things ❑ The program creates the graph, and then lets things run. Akin to cellular automata. Functional Reactive Programming

Slide 22

Slide 22 text

22 ❑ Event a – e.g. Button Clicks ❑ Behaviour a – e.g. System Time ❑ toBehaviour :: Event a -> Behaviour a ❑ mergeEvents :: Event a -> Event a -> Event a ❑ zipBehaviour :: (a -> b -> c) -> Behaviour a -> Behaviour b -> Behaviour c FRP Primitives

Slide 23

Slide 23 text

23 ❑ Event a – e.g. Button Clicks ❑ Behaviour a – e.g. System Time ❑ hold :: Event a -> Behaviour a ❑ sample :: Behaviour a -> Event b -> Event (a,b) ❑ merge :: Event a -> Event a -> Event a ❑ zip :: Behaviour a -> Behaviour b -> Behaviour (a,b) FRP Primitives

Slide 24

Slide 24 text

24 ❑ text :: Behaviour String -> IO () ❑ button :: Event () ❑ gui = do clicks <- button text hold “Not Clicked” (map (_ -> “Clicked!”) clicks) FRP GUIs

Slide 25

Slide 25 text

25 Monads

Slide 26

Slide 26 text

26 ❑ Tagged values ❑ Provide explicit control over sequencing Monads IO String

Slide 27

Slide 27 text

27 ❑ Values can be evaluated in any order (or left unevaluated). ❑ Effects need explicit control over sequencing and sharing. ❑ The only way sequencing is possible in FP is through nested functions. i.e. Callbacks. Controlling Sequencing write "Hello" (() -> write "World")

Slide 28

Slide 28 text

28 write "Do you want a pizza?" ( () -> read ( ans -> if (ans == "Yes") orderPizza ( () -> write "Should I launch missiles?" ( () -> read ( ans -> if (ans == "Yes") launchNukes () ) ) ) ) ) Continuation Passing Style

Slide 29

Slide 29 text

29 x <- foo … Sprinkle Some Syntactic Sugar foo (x -> …)

Slide 30

Slide 30 text

30 () <- write "Do you want a pizza?“ ans <- read if(ans == "Yes") () <- orderPizza () <- write "Should I launch missiles?“ ans <- read if (ans == "Yes") () <- launchNukes Sprinkle Some Syntactic Sugar write "Do you want a pizza?" (() -> read (ans -> if (ans == "Yes") orderPizza (() -> write "Should I launch missiles?" (() -> read (ans -> if (ans == "Yes") launchNukes () ) ) ) ) )

Slide 31

Slide 31 text

31 do write "Do you want a pizza?“ ans <- read if(ans == "Yes") orderPizza write "Should I launch missiles?“ ans <- read if (ans == "Yes") launchNukes Do notation

Slide 32

Slide 32 text

32 ❑ Callbacks are generally associated with Asynchronous code ❑ Do notation avoids “Callback Hell” Asynchronous Computations are Effects ajax GET “google.com" (response -> …)

Slide 33

Slide 33 text

33 do post <- ajax GET “/post/1“ map post.comments (\cid -> do comment <- ajax GET “/comments/{cid}“ … ) Avoiding Callback Hell ajax GET “/post/1" (post -> map post.comments (\cid -> ajax GET “/comments/{cid}" (comment -> … ) ) )

Slide 34

Slide 34 text

34 if(ans == "Yes") orderPizza else ??? Where is the Else block?

Slide 35

Slide 35 text

35 if(ans == "Yes") orderPizza else return () Where is the Else block?

Slide 36

Slide 36 text

36 Type: IO a Bind: IO a -> (a -> IO b) -> IO b Return: a -> IO a The Monad

Slide 37

Slide 37 text

37 ❑ Reader ❑ Logger ❑ State ❑ Exceptions ❑ Random ❑ … Bring Your Own Monad

Slide 38

Slide 38 text

38 Type: Reader e a :: e -> a Bind: Reader e a -> (a -> Reader e b) -> Reader e b Return: a -> Reader e a ask: Reader e e runReader: e -> Reader e a -> a Reader Monad

Slide 39

Slide 39 text

39 main = runReader myConfig do res <- foo bar res foo = do config <- ask; … bar res = do config <- ask; … Reader Monad

Slide 40

Slide 40 text

40 “Haskell” is the world’s finest imperative programming language. ~Simon Peyton Jones

Slide 41

Slide 41 text

41 “Haskell” is the world’s finest imperative programming language. ~Simon Peyton Jones (Creator of Haskell)

Slide 42

Slide 42 text

42 ❑ Like Monads, you can define your own Effects ❑ But you can define the usage and handling of the effects separately ❑ And effects compose freely (pun intended) Algebraic Effects

Slide 43

Slide 43 text

43 data Console callback = Read (String -> callback) | Write String callback handle (Read cb) = … handle (Write s cb) = … Console Effect PSEUDO-CODE

Slide 44

Slide 44 text

44 data Console callback = Read (String -> callback) | Write String callback handle (Read cb) = s = do readLine(); cb(s) handle (Write s cb) = do console.log(s); cb() Console Effect PSEUDO-CODE

Slide 45

Slide 45 text

45 handle (Write s cb) = do console.log(s); console.log(s); cb(); handle (Write s cb) = cb(); handle (Write s cb) = do cb(); console.log(s); handle (Write s cb) = do if(test(s)) console.log(s); cb(); You can do more things PSEUDO-CODE

Slide 46

Slide 46 text

46 handle (Return x) = return (x,””); handle (Write s cb) = (x,rest) = do cb(); return (x, s:rest); Returning Values from Handlers PSEUDO-CODE

Slide 47

Slide 47 text

47 Side Effects !!

Slide 48

Slide 48 text

48 Best Friends !!

Slide 49

Slide 49 text

49 Thank You