class: center, middle # Far more than you've ever wanted to know about ADTs [Nicolas Rinaudo] • [@NicolasRinaudo@functional.cafe] --- class: center, middle # Far more than you've ever wanted to know about .highlight[ADTs] [Nicolas Rinaudo] • [@NicolasRinaudo@functional.cafe] --- class: center, middle .center[data:image/s3,"s3://crabby-images/10e61/10e61f78aa299e0b8c821f14c3bb3db2f7e25ee4" alt="Our proud hero"] --- ## Valid programs data:image/s3,"s3://crabby-images/2453a/2453ace182c0c788177e500e3e78c6a1f3f5ac96" alt="Facing south" --- ## Valid programs data:image/s3,"s3://crabby-images/ec1dc/ec1dc6ea340d7bf2970741f82bc5df7d314cce4f" alt="Sending face north" ```haskell face north ``` --- ## Valid programs data:image/s3,"s3://crabby-images/a71a3/a71a34fda1be754e72798ae840d44194a5f60343" alt="Facing north" ```haskell face north ``` --- ## Valid programs data:image/s3,"s3://crabby-images/6fb53/6fb533dda11753c90f667d2616f52c5c7fd1fde6" alt="Sending face west" ```haskell face west ``` --- ## Valid programs data:image/s3,"s3://crabby-images/7dec5/7dec5466cfda0a670c9439b80f65eb39c00dd284" alt="Facing west" ```haskell face west ``` --- ## Valid programs data:image/s3,"s3://crabby-images/ba8f5/ba8f53551f14a3ee5401e8f20dc717f00bdbbd5d" alt="Sending face south" ```haskell face south ``` --- ## Valid programs data:image/s3,"s3://crabby-images/2453a/2453ace182c0c788177e500e3e78c6a1f3f5ac96" alt="Facing south" ```haskell face south ``` --- ## Valid programs data:image/s3,"s3://crabby-images/ec1dc/ec1dc6ea340d7bf2970741f82bc5df7d314cce4f" alt="Sending face east" ```haskell face east ``` --- ## Valid programs data:image/s3,"s3://crabby-images/c692a/c692aa83e18c252c8633349d99a69c6353ace6d2" alt="Facing east" ```haskell face east ``` --- ## Valid programs data:image/s3,"s3://crabby-images/5b437/5b4377117d4bd91b2bb41474b8d05aa0a9e25bf6" alt="Sending start" ```haskell start ``` --- ## Valid programs data:image/s3,"s3://crabby-images/b7654/b7654897ef4687e5ae669cbdc6bf6d01a937a444" alt="Started" ```haskell start ``` --- ## Valid programs data:image/s3,"s3://crabby-images/ac938/ac9388097385026d9039c487981849f7a4d6227b" alt="Sending stop" ```haskell stop ``` --- ## Valid programs data:image/s3,"s3://crabby-images/5ad82/5ad8234eca5939eb4cdc175eca5a82b9015e1de1" alt="Stopped" ```haskell stop ``` --- ## Invalid programs data:image/s3,"s3://crabby-images/d2477/d2477582be5f3666994cb953991abfadfd7beb76" alt="Sending triple_backflip" ```haskell triple_backflip ``` --- ## Invalid programs data:image/s3,"s3://crabby-images/a5f15/a5f15512df364e89a669a602954ecb958885a62f" alt="Confused" ```haskell triple_backflip ``` --- ## Invalid programs data:image/s3,"s3://crabby-images/7fb87/7fb87a1c5036a37609cae9dccf0a3a0b01577a07" alt="Kaboom" ```haskell triple_backflip ``` --- ## Invalid programs data:image/s3,"s3://crabby-images/d2477/d2477582be5f3666994cb953991abfadfd7beb76" alt="Sending face -35" ```haskell face -35 ``` --- ## Invalid programs data:image/s3,"s3://crabby-images/899bf/899bf00884d3fac105c4e0369fa6b1ad0c23c9ff" alt="Rejected" ```haskell face -35 ``` --- ## Invalid programs data:image/s3,"s3://crabby-images/91101/9110131f25b2e5f0dff92080f91aaa4e56d5035d" alt="Rejected" ```haskell face -35 ``` --- class: center, middle name: direction # Representing `Direction` --- ## Magic values .diff-add[ ```scala *`object Direction:` ``` ] --- ## Magic values .diff-add[ ```scala *object Direction`:` * `val North: Int = ???` * `val East: Int = ???` * `val South: Int = ???` * `val West: Int = ???` ``` ] --- ## Magic values .diff-rm[ ```scala object Direction`:` val North: Int = `???` val East: Int = `???` val South: Int = `???` val West: Int = `???` ``` ] --- ## Magic values .diff-add[ ```scala object Direction: * val North: Int = `1` * val East: Int = `2` * val South: Int = `3` * val West: Int = `4` ``` ] --- ## Magic values ```scala def label(d: Int) = ??? ``` --- ## Magic values ```scala def label(`d: Int`) = ??? ``` --- ## Magic values .diff-rm[ ```scala *def label(d: Int) = `???` ``` ] --- ## Magic values .diff-add[ ```scala *def label(d: Int) = `d match` * `case Direction.North ⇒ ???` * `case Direction.East ⇒ ???` * `case Direction.South ⇒ ???` * `case Direction.West ⇒ ???` ``` ] --- ## Magic values .diff-rm[ ```scala def label(d: Int) = d match * case Direction.North ⇒ `???` * case Direction.East ⇒ `???` * case Direction.South ⇒ `???` * case Direction.West ⇒ `???` ``` ] --- ## Magic values .diff-add[ ```scala def label(d: Int) = d match * case Direction.North ⇒ `"north"` * case Direction.East ⇒ `"east"` * case Direction.South ⇒ `"south"` * case Direction.West ⇒ `"east"` ``` ] --- ## Magic values data:image/s3,"s3://crabby-images/5b437/5b4377117d4bd91b2bb41474b8d05aa0a9e25bf6" alt="Sending face -35" ```scala label(-35) ``` --- ## Magic values data:image/s3,"s3://crabby-images/7fb87/7fb87a1c5036a37609cae9dccf0a3a0b01577a07" alt="Kaboom" ```scala label(`-35`) // 💥 scala.MatchError: -35 ``` --- ## Type aliases .diff-add[ ```scala *`type Direction = Int` * object Direction: val North: Int = 1 val East: Int = 2 val South: Int = 3 val West: Int = 4 ``` ] --- ## Type aliases .diff-rm[ ```scala type Direction = Int object Direction: * val North: `Int` = 1 * val East: `Int` = 2 * val South: `Int` = 3 * val West: `Int` = 4 ``` ] --- ## Type aliases .diff-add[ ```scala type Direction = Int object Direction: * val North: `Direction` = 1 * val East: `Direction` = 2 * val South: `Direction` = 3 * val West: `Direction` = 4 ``` ] --- ## Type aliases .diff-rm[ ```scala *def label(d: `Int`) = d match case Direction.North ⇒ "north" case Direction.East ⇒ "east" case Direction.South ⇒ "south" case Direction.West ⇒ "west" ``` ] --- ## Type aliases .diff-add[ ```scala *def label(d: `Direction`) = d match case Direction.North ⇒ "north" case Direction.East ⇒ "east" case Direction.South ⇒ "south" case Direction.West ⇒ "west" ``` ] --- ## Type aliases data:image/s3,"s3://crabby-images/5b437/5b4377117d4bd91b2bb41474b8d05aa0a9e25bf6" alt="Sending face -35" ```scala label(-35) ``` --- ## Type aliases data:image/s3,"s3://crabby-images/7fb87/7fb87a1c5036a37609cae9dccf0a3a0b01577a07" alt="Kaboom" ```scala label(`-35`) // 💥 scala.MatchError: -35 ``` --- ## Scala's `Enumeration` .diff-rm[ ```scala *`type Direction = Int` object Direction: * `val North: Direction = 1` * `val East: Direction = 2` * `val South: Direction = 3` * `val West: Direction = 4` ``` ] --- ## Scala's `Enumeration` ```scala object Direction: ``` --- ## Scala's `Enumeration` .diff-add[ ```scala *object Direction `extends Enumeration`: ``` ] --- ## Scala's `Enumeration` .diff-add[ ```scala object Direction extends Enumeration: * `val North, East, South, West = Value` ``` ] --- ## Scala's `Enumeration` .diff-rm[ ```scala *def label(d: `Direction`) = d match case Direction.North ⇒ "north" case Direction.East ⇒ "east" case Direction.South ⇒ "south" case Direction.West ⇒ "west" ``` ] --- ## Scala's `Enumeration` .diff-add[ ```scala *def label(d: `Direction.Value`) = d match case Direction.North ⇒ "north" case Direction.East ⇒ "east" case Direction.South ⇒ "south" case Direction.West ⇒ "west" ``` ] --- ## Scala's `Enumeration` data:image/s3,"s3://crabby-images/5b437/5b4377117d4bd91b2bb41474b8d05aa0a9e25bf6" alt="Sending face -35" ```scala label(-35) ``` --- ## Scala's `Enumeration` data:image/s3,"s3://crabby-images/899bf/899bf00884d3fac105c4e0369fa6b1ad0c23c9ff" alt="Rejected" ```scala label(`-35`) // ⛔ Found: Int // Required: Direction.Value ``` --- ## Scala's `Enumeration` .diff-rm[ ```scala def label(d: Direction.Value) = d match case Direction.North ⇒ "north" case Direction.East ⇒ "east" case Direction.South ⇒ "south" * `case Direction.West ⇒ "west"` ``` ] --- ## Scala's `Enumeration` .diff-add[ ```scala def label(d: Direction.Value) = d match case Direction.North ⇒ "north" case Direction.East ⇒ "east" case Direction.South ⇒ "south" *`//case Direction.West ⇒ "west"` ``` ] --- ## Scala's `Enumeration` data:image/s3,"s3://crabby-images/5b437/5b4377117d4bd91b2bb41474b8d05aa0a9e25bf6" alt="Sending face west" ```scala label(Direction.West) ``` --- ## Scala's `Enumeration` data:image/s3,"s3://crabby-images/7fb87/7fb87a1c5036a37609cae9dccf0a3a0b01577a07" alt="Kaboom" ```scala label(`Direction.West`) // 💥 scala.MatchError: West ``` --- ## Hand-written enumeration .diff-add[ ```scala *`trait Direction` ``` ] --- ## Hand-written enumeration .diff-add[ ```scala trait Direction * `object Direction:` ``` ] --- ## Hand-written enumeration .diff-add[ ```scala trait Direction object Direction: * `case object North extends Direction` ``` ] --- ## Hand-written enumeration .diff-add[ ```scala trait Direction object Direction: case object North extends Direction * `case object East extends Direction` ``` ] --- ## Hand-written enumeration .diff-add[ ```scala trait Direction object Direction: case object North extends Direction case object East extends Direction * `case object South extends Direction` ``` ] --- ## Hand-written enumeration .diff-add[ ```scala trait Direction object Direction: case object North extends Direction case object East extends Direction case object South extends Direction * `case object West extends Direction` ``` ] --- ## Hand-written enumeration .diff-rm[ ```scala *def label(d: `Direction.Value`) = d match case Direction.North ⇒ "north" case Direction.East ⇒ "east" case Direction.South ⇒ "south" //case Direction.West ⇒ "west" ``` ] --- ## Hand-written enumeration .diff-add[ ```scala *def label(d: `Direction`) = d match case Direction.North ⇒ "north" case Direction.East ⇒ "east" case Direction.South ⇒ "south" //case Direction.West ⇒ "west" ``` ] --- ## Hand-written enumeration data:image/s3,"s3://crabby-images/5b437/5b4377117d4bd91b2bb41474b8d05aa0a9e25bf6" alt="Sending face west" ```scala label(Direction.West) ``` --- ## Hand-written enumeration data:image/s3,"s3://crabby-images/7fb87/7fb87a1c5036a37609cae9dccf0a3a0b01577a07" alt="Kaboom" ```scala label(`Direction.West`) // 💥 scala.MatchError: West ``` --- ## Hand-written enumeration .diff-add[ ```scala *`sealed` trait Direction object Direction: case object North extends Direction case object East extends Direction case object South extends Direction case object West extends Direction ``` ] --- ## Hand-written enumeration ```scala def label(d: Direction) = d match case Direction.North ⇒ "north" case Direction.East ⇒ "east" case Direction.South ⇒ "south" //case Direction.West ⇒ "west" ``` --- ## Hand-written enumeration ```scala def label(d: Direction) = d match case Direction.North ⇒ "north" case Direction.East ⇒ "east" case Direction.South ⇒ "south" //case Direction.West ⇒ "west" // ⛔ match may not be exhaustive. // It would fail on pattern case: West ``` --- ## [Key takeaways](#closing) -- Enumerations: -- * make nonsensical values impossible to represent. -- * guarantee that our code handles all necessary cases. -- * provided you're not using `scala.Enumeration` though... --- class: center, middle name: command # Representing `Command` --- ## Case class ```scala case class Command( ) ``` ```scala val cmd = Command( ) ``` --- ## Case class .diff-add[ ```scala case class Command( * `order: String` ) ``` ```scala val cmd = Command( * `"face"` ) ``` ] --- ## Case class .diff-add[ ```scala case class Command( * order: String`,` * `dir : Option[Direction]` ) ``` ```scala val cmd = Command( * "face"`,` * `Some(Direction.North)` ) ``` ] --- ## Case class data:image/s3,"s3://crabby-images/5b437/5b4377117d4bd91b2bb41474b8d05aa0a9e25bf6" alt="Triple backflip" ```scala Command( "triple backflip", Some(Direction.South) ) ``` --- ## Case class data:image/s3,"s3://crabby-images/5b437/5b4377117d4bd91b2bb41474b8d05aa0a9e25bf6" alt="Triple backflip" ```scala Command( `"triple backflip"`, Some(Direction.South) ) ``` --- ## Case class data:image/s3,"s3://crabby-images/7fb87/7fb87a1c5036a37609cae9dccf0a3a0b01577a07" alt="Kaboom" ```scala Command( "triple backflip", Some(Direction.South) ) ``` --- ## `Order` as enumeration .diff-add[ ```scala *`sealed trait Order` ``` ] --- ## `Order` as enumeration .diff-add[ ```scala sealed trait Order * *`object Order:` ``` ] --- ## `Order` as enumeration .diff-add[ ```scala sealed trait Order *object Order`:` * `case object Face extends Order` ``` ] --- ## `Order` as enumeration .diff-add[ ```scala sealed trait Order object Order: * `case object Face extends Order` * `case object Start extends Order` ``` ] --- ## `Order` as enumeration .diff-add[ ```scala sealed trait Order object Order: * `case object Face extends Order` * `case object Start extends Order` * `case object Stop extends Order` ``` ] --- ## `Order` as enumeration .diff-rm[ ```scala case class Command( * order: `String`, dir : Option[Direction] ) ``` ```scala val cmd = Command( * `"face"`, Some(Direction.North) ) ``` ] --- ## `Order` as enumeration .diff-add[ ```scala case class Command( * order: `Order`, dir : Option[Direction] ) ``` ```scala val cmd = Command( * `Order.Face`, Some(Direction.North) ) ``` ] --- ## `Order` as enumeration data:image/s3,"s3://crabby-images/5b437/5b4377117d4bd91b2bb41474b8d05aa0a9e25bf6" alt="Sending triple backflip" ```scala Command( "triple backflip", Some(Direction.South) ) ``` --- ## `Order` as enumeration data:image/s3,"s3://crabby-images/899bf/899bf00884d3fac105c4e0369fa6b1ad0c23c9ff" alt="Rejected" ```scala Command( `"triple backflip"`, Some(Direction.South) ) // ⛔ Found: String // Required: Order ``` --- ## `Order` as enumeration data:image/s3,"s3://crabby-images/899bf/899bf00884d3fac105c4e0369fa6b1ad0c23c9ff" alt="Rejected" .diff-rm[ ```scala Command( * `"triple backflip"`, Some(Direction.South) ) ``` ] --- ## `Order` as enumeration data:image/s3,"s3://crabby-images/5b437/5b4377117d4bd91b2bb41474b8d05aa0a9e25bf6" alt="Sending start south" .diff-add[ ```scala Command( * `Order.Start`, Some(Direction.South) ) ``` ] --- ## `Order` as enumeration data:image/s3,"s3://crabby-images/5b437/5b4377117d4bd91b2bb41474b8d05aa0a9e25bf6" alt="Sending start south" ```scala Command( Order.`Start`, Some(Direction.`South`) ) ``` --- ## `Order` as enumeration data:image/s3,"s3://crabby-images/7fb87/7fb87a1c5036a37609cae9dccf0a3a0b01577a07" alt="Kaboom" ```scala Command( Order.Start, Some(Direction.South) ) ``` --- ## `Order` as "enumeration" .diff-rm[ ```scala sealed trait Order object Order: * case `object Face` extends Order case object Start extends Order case object Stop extends Order ``` ] --- ## `Order` as "enumeration" .diff-add[ ```scala sealed trait Order object Order: * case `class Face(dir: Direction)` extends Order * case object Start extends Order * case object Stop extends Order ``` ] --- ## `Order` as "enumeration" .diff-rm[ ```scala *sealed trait `Order` *object `Order`: * case class Face(dir: Direction) extends `Order` * case object Start extends `Order` * case object Stop extends `Order` ``` ] --- ## `Command` as "enumeration" .diff-add[ ```scala *sealed trait `Command` *object `Command`: * case class Face(dir: Direction) extends `Command` * case object Start extends `Command` * case object Stop extends `Command` ``` ] --- ## `Command` as "enumeration" data:image/s3,"s3://crabby-images/5b437/5b4377117d4bd91b2bb41474b8d05aa0a9e25bf6" alt="Sending start south" ```scala Command( Order.Start, Some(Direction.South) ) ``` --- ## `Command` as "enumeration" data:image/s3,"s3://crabby-images/899bf/899bf00884d3fac105c4e0369fa6b1ad0c23c9ff" alt="Rejected" ```scala `Command`( Order.Start, Some(Direction.South) ) // ⛔ object Command does not take parameters ``` --- ## `Command` as "enumeration" data:image/s3,"s3://crabby-images/5b437/5b4377117d4bd91b2bb41474b8d05aa0a9e25bf6" alt="Sending" ```scala Command.Start ``` --- ## `Command` as "enumeration" data:image/s3,"s3://crabby-images/b7654/b7654897ef4687e5ae669cbdc6bf6d01a937a444" alt="Sending" ```scala Command.Start ``` --- ## [Key takeaways](#closing) -- * Enumerations are everywhere. -- * Enumerations are not enough. -- * Something "like an enumeration" might be, however. --- class: center, middle name: composition # Composing commands --- ## Script data:image/s3,"s3://crabby-images/2453a/2453ace182c0c788177e500e3e78c6a1f3f5ac96" alt="Waiting command" ```haskell start ``` --- ## Script data:image/s3,"s3://crabby-images/2453a/2453ace182c0c788177e500e3e78c6a1f3f5ac96" alt="Waiting command" ```haskell face east start stop ``` --- ## Script data:image/s3,"s3://crabby-images/ec1dc/ec1dc6ea340d7bf2970741f82bc5df7d314cce4f" alt="Sending script" ```haskell face east start stop ``` --- ## Script data:image/s3,"s3://crabby-images/c692a/c692aa83e18c252c8633349d99a69c6353ace6d2" alt="Starting" ```haskell `face east` start stop ``` --- ## Script data:image/s3,"s3://crabby-images/b7654/b7654897ef4687e5ae669cbdc6bf6d01a937a444" alt="Continue advancing" ```haskell face east `start` stop ``` --- ## Script data:image/s3,"s3://crabby-images/5ad82/5ad8234eca5939eb4cdc175eca5a82b9015e1de1" alt="Stopping" ```haskell face east start `stop` ``` --- ## Chaining commands .diff-add[ ```scala sealed trait Command object Command: case class Face(dir: Direction) extends Command case object Start extends Command case object Stop extends Command * * `case class Chain(` * `) extends Command` ``` ] --- ## Chaining commands .diff-add[ ```scala sealed trait Command object Command: case class Face(dir: Direction) extends Command case object Start extends Command case object Stop extends Command case class Chain( * `cmd1: Command` ) extends Command ``` ] --- ## Chaining commands .diff-add[ ```scala sealed trait Command object Command: case class Face(dir: Direction) extends Command case object Start extends Command case object Stop extends Command case class Chain( * cmd1: Command`,` * `cmd2: Command` ) extends Command ``` ] --- ## Chaining commands ```scala Command.Chain( Command.Chain( Command.Chain( Command.Face(Direction.East), Command.Chain( Command.Start, Command.Stop ) ), Command.Face(Direction.West) ), Command.Chain( Command.Start, Command.Stop ) ) ``` --- ## Chaining commands ```scala Command.Chain( Command.Chain( Command.Chain( Command.`Face`(Direction.`East`), Command.Chain( Command.Start, Command.Stop ) ), Command.Face(Direction.West) ), Command.Chain( Command.Start, Command.Stop ) ) ``` --- ## Chaining commands ```scala Command.Chain( Command.Chain( Command.Chain( Command.Face(Direction.East), Command.Chain( Command.`Start`, Command.Stop ) ), Command.Face(Direction.West) ), Command.Chain( Command.Start, Command.Stop ) ) ``` --- ## Chaining commands ```scala Command.Chain( Command.Chain( Command.Chain( Command.Face(Direction.East), Command.Chain( Command.Start, Command.`Stop` ) ), Command.Face(Direction.West) ), Command.Chain( Command.Start, Command.Stop ) ) ``` --- ## Chaining commands ```scala Command.Chain( Command.Chain( Command.Chain( Command.Face(Direction.East), Command.Chain( Command.Start, Command.Stop ) ), Command.`Face`(Direction.`West`) ), Command.Chain( Command.Start, Command.Stop ) ) ``` --- ## Chaining commands ```scala Command.Chain( Command.Chain( Command.Chain( Command.Face(Direction.East), Command.Chain( Command.Start, Command.Stop ) ), Command.Face(Direction.West) ), Command.Chain( Command.`Start`, Command.Stop ) ) ``` --- ## Chaining commands ```scala Command.Chain( Command.Chain( Command.Chain( Command.Face(Direction.East), Command.Chain( Command.Start, Command.Stop ) ), Command.Face(Direction.West) ), Command.Chain( Command.Start, Command.`Stop` ) ) ``` --- ## DSL ```scala extension (cmd1: Command) def ~>(cmd2: Command): Command = Command.Chain(cmd1, cmd2) ``` ```scala val startStop: Command = Command.Start ~> Command.Stop ``` --- ## DSL ```scala extension (`cmd1: Command`) def ~>(cmd2: Command): Command = Command.Chain(cmd1, cmd2) ``` ```scala val startStop: Command = `Command.Start` ~> Command.Stop ``` --- ## DSL ```scala extension (cmd1: Command) def `~>`(cmd2: Command): Command = Command.Chain(cmd1, cmd2) ``` ```scala val startStop: Command = Command.Start `~>` Command.Stop ``` --- ## DSL ```scala extension (cmd1: Command) def ~>(`cmd2: Command`): Command = Command.Chain(cmd1, cmd2) ``` ```scala val startStop: Command = Command.Start ~> Command.`Stop` ``` --- ## DSL ```scala extension (cmd1: Command) def ~>(cmd2: Command): Command = Command.`Chain(cmd1, cmd2)` ``` ```scala val `startStop`: Command = Command.Start ~> Command.Stop ``` --- ## DSL .diff-add[ ```scala *`val start = Command.Start` ``` ] --- ## DSL .diff-add[ ```scala val start = Command.Start *`val stop = Command.Stop` ``` ] --- ## DSL .diff-add[ ```scala val start = Command.Start val stop = Command.Stop *`def face(dir: Direction) = Command.Face(dir)` ``` ] --- ## DSL .diff-add[ ```scala *`val north = Direction.North` ``` ] --- ## DSL .diff-add[ ```scala val north = Direction.North *`val east = Direction.East` ``` ] --- ## DSL .diff-add[ ```scala val north = Direction.North val east = Direction.East *`val south = Direction.South` ``` ] --- ## DSL .diff-add[ ```scala val north = Direction.North val east = Direction.East val south = Direction.South *`val west = Direction.West` ``` ] --- ## Compound commands ```scala def move(d: Direction) = face(d) ~> start ~> stop ``` --- ## Compound commands ```scala def move(`d: Direction`) = face(d) ~> start ~> stop ``` --- ## Compound commands ```scala def move(d: Direction) = `face(d)` ~> start ~> stop ``` --- ## Compound commands ```scala def move(d: Direction) = face(d) ~> `start` ~> stop ``` --- ## Compound commands ```scala def move(d: Direction) = face(d) ~> start ~> `stop` ``` --- ## Compound commands data:image/s3,"s3://crabby-images/2453a/2453ace182c0c788177e500e3e78c6a1f3f5ac96" alt="Sending script" ```scala move(east) ~> move(west) ``` --- ## Compound commands data:image/s3,"s3://crabby-images/ec1dc/ec1dc6ea340d7bf2970741f82bc5df7d314cce4f" alt="Sending script" ```scala move(east) ~> move(west) ``` --- ## Compound commands data:image/s3,"s3://crabby-images/c692a/c692aa83e18c252c8633349d99a69c6353ace6d2" alt="Sending script" ```scala `move(east)` ~> move(west) ``` --- ## Compound commands data:image/s3,"s3://crabby-images/b7654/b7654897ef4687e5ae669cbdc6bf6d01a937a444" alt="Sending script" ```scala `move(east)` ~> move(west) ``` --- ## Compound commands data:image/s3,"s3://crabby-images/5ad82/5ad8234eca5939eb4cdc175eca5a82b9015e1de1" alt="Sending script" ```scala `move(east)` ~> move(west) ``` --- ## Compound commands data:image/s3,"s3://crabby-images/7dec5/7dec5466cfda0a670c9439b80f65eb39c00dd284" alt="Sending script" ```scala move(east) ~> `move(west)` ``` --- ## Compound commands data:image/s3,"s3://crabby-images/2bc5e/2bc5ef9dae866e99c35b0f930b12910ca8ea675b" alt="Sending script" ```scala move(east) ~> `move(west)` ``` --- ## Compound commands data:image/s3,"s3://crabby-images/7d87a/7d87a75c38a89a70353d3daaee27ea7c8a9d708c" alt="Sending script" ```scala move(east) ~> `move(west)` ``` --- ## [Key takeaways](#closing) -- We've completed our data structure by using: -- * records (`Chain`, `Face`). -- * enumerated types (`Command`, `Direction`). -- * recursive types (`Command` is expressed in terms of itself). --- class: center, middle name: adt # Algebraic Data Types --- ## Sum types ```scala Command = Face | Start | Stop | Chain ``` --- ## Sum types ```scala Command = `Face` | Start | Stop | Chain ``` --- ## Sum types ```scala Command = Face | `Start` | Stop | Chain ``` --- ## Sum types ```scala Command = Face | Start | `Stop` | Chain ``` --- ## Sum types ```scala Command = Face | Start | Stop | `Chain` ``` --- ## Sum types ```scala Command = Face | Start | Stop | Chain ``` > A sum type is a discriminated union of values, and can be thought of as an `OR` on types. --- ## Sum types ```scala Command = Face `|` Start `|` Stop `|` Chain ``` > A sum type is a discriminated union of values, and can be thought of as an .highlight[`OR`] on types. --- ## Sum types ```scala Command = Face | Start | Stop | Chain ``` > A sum type is a .highlight[discriminated] union of values, and can be thought of as an `OR` on types. --- ## Union types ```c union int_or_string { int as_int; char* as_string; }; ``` --- ## Union types ```c union `int_or_string` { int as_int; char* as_string; }; ``` --- ## Union types ```c union int_or_string { `int as_int;` char* as_string; }; ``` --- ## Union types ```c union int_or_string { int as_int; `char* as_string;` }; ``` --- ## Union types ```c #include
void print_union(union int_or_string e) { ??? } ``` --- ## Union types ```c #include
void print_union(union int_or_string e) { `???` } ``` --- ## Manually discriminated union types ```c enum typetag { int_tag, string_tag }; ``` --- ## Manually discriminated union types ```c enum `typetag` { int_tag, string_tag }; ``` --- ## Manually discriminated union types ```c enum typetag { `int_tag`, string_tag }; ``` --- ## Manually discriminated union types ```c enum typetag { int_tag, `string_tag` }; ``` --- ## Manually discriminated union types ```c union int_or_string { int as_int; char* as_string; }; ``` --- ## Manually discriminated union types .diff-add[ ```c union int_or_string { * `enum typetag discriminator;` * int as_int; char* as_string; }; ``` ] --- ## Manually discriminated union types .diff-rm[ ```c *`union` int_or_string { enum typetag discriminator; int as_int; char* as_string; }; ``` ] --- ## Manually discriminated union types .diff-add[ ```c *`struct` int_or_string { enum typetag discriminator; int as_int; char* as_string; }; ``` ] --- ## Manually discriminated union types .diff-rm[ ```c struct int_or_string { enum typetag discriminator; * `int as_int;` * `char* as_string;` }; ``` ] --- ## Manually discriminated union types .diff-add[ ```c struct int_or_string { enum typetag discriminator; * `union {` * ` int as_int;` * ` char* as_string;` * `} value;` }; ``` ] --- ## Manually discriminated union types ```c struct int_or_string { enum typetag discriminator; * union { * int as_int; * char* as_string; * } value; }; ``` --- ## Manually discriminated union types ```c struct int_or_string { * enum typetag discriminator; union { int as_int; char* as_string; } value; }; ``` --- ## Manually discriminated union types .diff-rm[ ```c #include
*void print_union(`union` int_or_string e) { ??? } ``` ] --- ## Manually discriminated union types .diff-add[ ```c #include
*void print_union(`struct` int_or_string e) { ??? } ``` ] --- ## Manually discriminated union types .diff-rm[ ```c #include
void print_union(struct int_or_string e) { * `???` } ``` ] --- ## Manually discriminated union types .diff-add[ ```c #include
void print_union(struct int_or_string e) { * `if(e.discriminator == int_tag) {` * `printf("int: %i", e.value.as_int);` * `}` * `else {` * `printf("string: %s", e.value.as_string);` * `}` } ``` ] --- ## Manually discriminated union types ```c #include
void print_union(struct int_or_string e) { if(`e.discriminator == int_tag`) { printf("int: %i", e.value.as_int); } else { printf("string: %s", e.value.as_string); } } ``` --- ## Manually discriminated union types ```c #include
void print_union(struct int_or_string e) { if(e.discriminator == int_tag) { printf("int: %i", `e.value.as_int`); } else { printf("string: %s", e.value.as_string); } } ``` --- ## Manually discriminated union types ```c #include
void print_union(struct int_or_string e) { if(e.discriminator == int_tag) { printf("int: %i", e.value.as_int); } else { printf("string: %s", `e.value.as_string`); } } ``` --- ## Sum types ```scala Command = Face | Start | Stop | Chain ``` > A sum type is a .highlight[discriminated] union of values, and can be thought of as an `OR` on types. --- ## Sum types > An ADT is a sum type [...] --- ## Product types ```scala Command = Face | Start | Stop | Chain ``` --- ## Product types .diff-rm[ ```scala Command = Face | Start | Stop * | `Chain` ``` ] --- ## Product types .diff-add[ ```scala Command = Face | Start | Stop * | `Command & Command` ``` ] --- ## Product types ```scala Command = Face | Start | Stop | `Command` & Command ``` --- ## Product types ```scala Command = Face | Start | Stop | Command & `Command` ``` --- ## Product types ```scala Command = Face | Start | Stop | Command & Command ``` > A product type is an aggregation of values, and can be thought of as an `AND` on types. --- ## Product types ```scala Command = Face | Start | Stop | Command `&` Command ``` > A product type is an aggregation of values, and can be thought of as an .highlight[`AND`] on types. --- ## Product types > An ADT is a sum type of product types [...] --- ## Algebraic Data Types ```scala Command = Face | Start | Stop | Command & Command ``` --- ## Algebraic Data Types ```scala `Command` = Face | Start | Stop | Command & Command ``` --- ## Algebraic Data Types ```scala Command = Face | Start | Stop | `Command` & `Command` ``` --- ## Algebraic Data Types > An ADT is a potentially recursive sum type of product types. --- ## Dedicated syntax ```scala sealed trait Command object Command: case class Face(dir: Direction) extends Command case object Start extends Command case object Stop extends Command case class Chain( cmd1: Command, cmd2: Command ) extends Command ``` --- ## Dedicated syntax .diff-rm[ ```scala *`sealed trait Command` * *`object` Command: case class Face(dir: Direction) extends Command case object Start extends Command case object Stop extends Command case class Chain( cmd1: Command, cmd2: Command ) extends Command ``` ] --- ## Dedicated syntax .diff-add[ ```scala *`enum` Command: case class Face(dir: Direction) extends Command case object Start extends Command case object Stop extends Command case class Chain( cmd1: Command, cmd2: Command ) extends Command ``` ] --- ## Dedicated syntax .diff-rm[ ```scala enum Command: * case `class` Face(dir: Direction) extends Command * case `object` Start extends Command * case `object` Stop extends Command * case `class` Chain( cmd1: Command, cmd2: Command ) extends Command ``` ] --- ## Dedicated syntax .diff-rm[ ```scala enum Command: * case Face(dir: Direction) `extends Command` * case Start `extends Command` * case Stop `extends Command` case Chain( cmd1: Command, cmd2: Command * ) `extends Command` ``` ] --- ## Dedicated syntax ```scala enum Command: case Face(dir: Direction) case Start case Stop case Chain( cmd1: Command, cmd2: Command ) ``` --- ## [Key takeaways](#closing) -- Algebraic Data Types are: -- * sum types. -- * product types. -- * potentially recursive. --- class: center, middle name: safe-composition # Safe command composition --- ## Illegal state transition data:image/s3,"s3://crabby-images/5b437/5b4377117d4bd91b2bb41474b8d05aa0a9e25bf6" alt="Start " ```scala start ~> start ``` --- ## Illegal state transition data:image/s3,"s3://crabby-images/b7654/b7654897ef4687e5ae669cbdc6bf6d01a937a444" alt="Start " ```scala `start` ~> start ``` --- ## Illegal state transition data:image/s3,"s3://crabby-images/8e70c/8e70c166bdf5e1c8ee27546c2692394833325891" alt="Start " ```scala start ~> `start` ``` --- ## Illegal state transition data:image/s3,"s3://crabby-images/7fb87/7fb87a1c5036a37609cae9dccf0a3a0b01577a07" alt="Start " ```scala start ~> start ``` --- ## Illegal state transition data:image/s3,"s3://crabby-images/ac938/ac9388097385026d9039c487981849f7a4d6227b" alt="Start" ```scala stop ~> stop ``` --- ## Illegal state transition data:image/s3,"s3://crabby-images/5ad82/5ad8234eca5939eb4cdc175eca5a82b9015e1de1" alt="Start" ```scala `stop` ~> stop ``` --- ## Illegal state transition data:image/s3,"s3://crabby-images/a5f15/a5f15512df364e89a669a602954ecb958885a62f" alt="Start " ```scala stop ~> `stop` ``` --- ## Illegal state transition data:image/s3,"s3://crabby-images/7fb87/7fb87a1c5036a37609cae9dccf0a3a0b01577a07" alt="Start" ```scala stop ~> stop ``` --- ## Illegal state transition data:image/s3,"s3://crabby-images/5b437/5b4377117d4bd91b2bb41474b8d05aa0a9e25bf6" alt="Start" ```scala start ~> face(north) ``` --- ## Illegal state transition data:image/s3,"s3://crabby-images/b7654/b7654897ef4687e5ae669cbdc6bf6d01a937a444" alt="Start" ```scala `start` ~> face(north) ``` --- ## Illegal state transition data:image/s3,"s3://crabby-images/67e47/67e47ef6e2fdc288c9086b4f8d2b615e3a5df7be" alt="Start" ```scala start ~> `face(north)` ``` --- ## Illegal state transition data:image/s3,"s3://crabby-images/7fb87/7fb87a1c5036a37609cae9dccf0a3a0b01577a07" alt="Start" .foreground[data:image/s3,"s3://crabby-images/dfd8c/dfd8c14058178372f0f083b241a5b98ae9215845" alt="Big badaboom"] ```scala start ~> face(north) ``` --- ## Tracking state ```scala final abstract class Idle final abstract class Moving ``` --- ## Tracking state ```scala final abstract class `Idle` final abstract class Moving ``` --- ## Tracking state ```scala final abstract class Idle final abstract class `Moving` ``` --- ## Tracking state .diff-add[ ```scala *enum Command`[Before, After]`: case Face(dir: Direction) case Start case Stop case Chain( cmd1: Command, cmd2: Command ) ``` ] --- ## Tracking state .diff-add[ ```scala enum Command[Before, After]: * case Face(dir: Direction) `extends Command[Idle, Idle]` case Start case Stop case Chain( cmd1: Command, cmd2: Command ) ``` ] --- ## Tracking state .diff-add[ ```scala enum Command[Before, After]: case Face(dir: Direction) extends Command[Idle, Idle] * case Start `extends Command[Idle, Moving]` case Stop case Chain( cmd1: Command, cmd2: Command ) ``` ] --- ## Tracking state .diff-add[ ```scala enum Command[Before, After]: case Face(dir: Direction) extends Command[Idle, Idle] case Start extends Command[Idle, Moving] * case Stop `extends Command[Moving, Idle]` case Chain( cmd1: Command, cmd2: Command ) ``` ] --- ## Tracking state ```scala enum Command[Before, After]: case Face(dir: Direction) extends Command[Idle, Idle] case Start extends Command[Idle, Moving] case Stop extends Command[Moving, Idle] `case Chain(` `cmd1: Command,` `cmd2: Command` `)` ``` --- ## Tracking state .diff-add[ ```scala enum Command[Before, After]: case Face(dir: Direction) extends Command[Idle, Idle] case Start extends Command[Idle, Moving] case Stop extends Command[Moving, Idle] case Chain( * cmd1: Command`[A, B]`, cmd2: Command ) ``` ] --- ## Tracking state .diff-add[ ```scala enum Command[Before, After]: case Face(dir: Direction) extends Command[Idle, Idle] case Start extends Command[Idle, Moving] case Stop extends Command[Moving, Idle] case Chain( cmd1: Command[A, B], * cmd2: Command`[B, C]` ) ``` ] --- ## Tracking state ```scala enum Command[Before, After]: case Face(dir: Direction) extends Command[Idle, Idle] case Start extends Command[Idle, Moving] case Stop extends Command[Moving, Idle] case Chain( cmd1: Command[A, `B`], cmd2: Command[`B`, C] ) ``` --- ## Tracking state .diff-add[ ```scala enum Command[Before, After]: case Face(dir: Direction) extends Command[Idle, Idle] case Start extends Command[Idle, Moving] case Stop extends Command[Moving, Idle] case Chain( cmd1: Command[A, B], cmd2: Command[B, C] * ) `extends Command[A, C]` ``` ] --- ## Tracking state .diff-add[ ```scala enum Command[Before, After]: case Face(dir: Direction) extends Command[Idle, Idle] case Start extends Command[Idle, Moving] case Stop extends Command[Moving, Idle] * case Chain`[A, B, C]`( cmd1: Command[A, B], cmd2: Command[B, C] ) extends Command[A, C] ``` ] --- ## DSL .diff-add[ ```scala *extension (cmd1: Command`[A, B]`) def ~>(cmd2: Command): Command = Command.Chain(cmd1, cmd2) ``` ] --- ## DSL .diff-add[ ```scala *extension `[A, B]`(cmd1: Command[A, B]) def ~>(cmd2: Command): Command = Command.Chain(cmd1, cmd2) ``` ] --- ## DSL .diff-add[ ```scala extension [A, B](cmd1: Command[A, B]) * def ~>(cmd2: Command`[B, C]`): Command = Command.Chain(cmd1, cmd2) ``` ] --- ## DSL .diff-add[ ```scala extension [A, B](cmd1: Command[A, B]) * def ~>[`C`](cmd2: Command[B, C]): Command = Command.Chain(cmd1, cmd2) ``` ] --- ## DSL .diff-add[ ```scala extension [A, B](cmd1: Command[A, B]) * def ~>[C](cmd2: Command[B, C]): Command`[A, C]` = Command.Chain(cmd1, cmd2) ``` ] --- ## Illegal state transition data:image/s3,"s3://crabby-images/5b437/5b4377117d4bd91b2bb41474b8d05aa0a9e25bf6" alt="Start" ```scala start ~> start ``` --- ## Illegal state transition data:image/s3,"s3://crabby-images/899bf/899bf00884d3fac105c4e0369fa6b1ad0c23c9ff" alt="Start " ```scala start ~> `start` // ⛔ Found: Command[Idle, Moving] // Required: Command[Moving, C] ``` --- ## Illegal state transition data:image/s3,"s3://crabby-images/5b437/5b4377117d4bd91b2bb41474b8d05aa0a9e25bf6" alt="Start" ```scala stop ~> stop ``` --- ## Illegal state transition data:image/s3,"s3://crabby-images/899bf/899bf00884d3fac105c4e0369fa6b1ad0c23c9ff" alt="Start" ```scala stop ~> `stop` // ⛔ Found: Command[Moving, Idle] // Required: Command[Idle, C] ``` --- ## Illegal state transition data:image/s3,"s3://crabby-images/5b437/5b4377117d4bd91b2bb41474b8d05aa0a9e25bf6" alt="Start" ```scala start ~> face(north) ``` --- ## Illegal state transition data:image/s3,"s3://crabby-images/899bf/899bf00884d3fac105c4e0369fa6b1ad0c23c9ff" alt="Start" ```scala start ~> `face(north)` // ⛔ Found: Command[Idle, Idle] // Required: Command[Moving, C] ``` --- ## Safe state transition data:image/s3,"s3://crabby-images/2453a/2453ace182c0c788177e500e3e78c6a1f3f5ac96" alt="Start" ```scala face(east) ~> start ~> stop ``` --- ## Safe state transition data:image/s3,"s3://crabby-images/ec1dc/ec1dc6ea340d7bf2970741f82bc5df7d314cce4f" alt="Start" ```scala face(east) ~> start ~> stop ``` --- ## Safe state transition data:image/s3,"s3://crabby-images/c692a/c692aa83e18c252c8633349d99a69c6353ace6d2" alt="Start" ```scala `face(east)` ~> start ~> stop ``` --- ## Safe state transition data:image/s3,"s3://crabby-images/b7654/b7654897ef4687e5ae669cbdc6bf6d01a937a444" alt="Start" ```scala face(east) ~> `start` ~> stop ``` --- ## Safe state transition data:image/s3,"s3://crabby-images/5ad82/5ad8234eca5939eb4cdc175eca5a82b9015e1de1" alt="Start" ```scala face(east) ~> start ~> `stop` ``` --- ## Smarter exhaustivity checks ```scala def movingLabel(cmd: Command[Moving, _]) = cmd match case Command.Stop => "stop" case _: Command.Chain[_, _, _] => "chain" ``` --- ## Smarter exhaustivity checks ```scala def movingLabel(cmd: `Command[Moving, _]`) = cmd match case Command.Stop => "stop" case _: Command.Chain[_, _, _] => "chain" ``` --- ## Smarter exhaustivity checks ```scala def movingLabel(cmd: Command[Moving, _]) = cmd match case `Command.Stop` => "stop" case _: Command.Chain[_, _, _] => "chain" ``` --- ## Smarter exhaustivity checks ```scala def movingLabel(cmd: Command[Moving, _]) = cmd match case Command.Stop => "stop" case _: `Command.Chain[_, _, _]` => "chain" ``` --- ## Smarter exhaustivity checks data:image/s3,"s3://crabby-images/5b437/5b4377117d4bd91b2bb41474b8d05aa0a9e25bf6" alt="Start" ```scala movingLabel(Command.Start) ``` --- ## Smarter exhaustivity checks data:image/s3,"s3://crabby-images/899bf/899bf00884d3fac105c4e0369fa6b1ad0c23c9ff" alt="Start" ```scala movingLabel(`Command.Start`) // ⛔ Found: Command[Idle, Moving] // Required: Command[Moving, ?] ``` --- ## Smarter exhaustivity checks .diff-rm[ ```scala def movingLabel(cmd: Command[Moving, _]) = cmd match case Command.Stop => "stop" * `case _: Command.Chain[_, _, _] => "chain"` ``` ] --- ## Smarter exhaustivity checks .diff-add[ ```scala def movingLabel(cmd: Command[Moving, _]) = cmd match case Command.Stop => "stop" *`//case _: Command.Chain[_, _, _] => "chain"` ``` ] --- ## Smarter exhaustivity checks ```scala def movingLabel(cmd: Command[Moving, _]) = cmd match case Command.Stop => "stop" //case _: `Command.Chain[_, _, _]` => "chain" // ⛔ match may not be exhaustive. // It would fail on pattern case: Command.Chain(_, _) ``` --- ## Smarter exhaustivity checks .diff-rm[ ```scala def movingLabel(cmd: Command[Moving, _]) = cmd match case Command.Stop => "stop" *`//case _: Command.Chain[_, _, _] => "chain"` ``` ] --- ## Smarter exhaustivity checks .diff-add[ ```scala def movingLabel(cmd: Command[Moving, _]) = cmd match case Command.Stop => "stop" case _: Command.Chain[_, _, _] => "chain" * `case Command.Start => "start"` ``` ] --- ## Smarter exhaustivity checks ```scala def movingLabel(cmd: Command[Moving, _]) = cmd match case Command.Stop => "stop" case _: Command.Chain[_, _, _] => "chain" case `Command.Start` => "start" // ⛔ Found: Command[Idle, Moving] // Required: Command[Moving, ?] ``` --- ## [Key takeaways](#closing) -- We used type constraints on our sum type's members to: -- * make illegal state transitions impossible to represent. -- * guarantee that our code handles all necessary cases. -- * guarantee that our code handles *only* necessary cases. --- class: center, middle name: gadt # Generalised Algebraic Data Types --- ## Sum type ```scala enum Command[Before, After]: case Face(dir: Direction) extends Command[Idle, Idle] case Start extends Command[Idle, Moving] case Stop extends Command[Moving, Idle] case Chain[A, B, C]( cmd1: Command[A, B], cmd2: Command[B, C] ) extends Command[A, C] ``` --- ## Sum type ```scala enum `Command`[Before, After]: case Face(dir: Direction) extends Command[Idle, Idle] case Start extends Command[Idle, Moving] case Stop extends Command[Moving, Idle] case Chain[A, B, C]( cmd1: Command[A, B], cmd2: Command[B, C] ) extends Command[A, C] ``` --- ## Sum type ```scala enum Command[Before, After]: case `Face`(dir: Direction) extends Command[Idle, Idle] case `Start` extends Command[Idle, Moving] case `Stop` extends Command[Moving, Idle] case `Chain`[A, B, C]( cmd1: Command[A, B], cmd2: Command[B, C] ) extends Command[A, C] ``` --- ## Sum type > A GADT is a sum type [...] --- ## Witness type ```scala enum Command[`Before`, After]: case Face(dir: Direction) extends Command[`Idle`, Idle] case Start extends Command[`Idle`, Moving] case Stop extends Command[`Moving`, Idle] case Chain[A, B, C]( cmd1: Command[A, B], cmd2: Command[B, C] ) extends Command[`A`, C] ``` --- ## Witness type ```scala enum Command[Before, `After`]: case Face(dir: Direction) extends Command[Idle, `Idle`] case Start extends Command[Idle, `Moving`] case Stop extends Command[Moving, `Idle`] case Chain[A, B, C]( cmd1: Command[A, B], cmd2: Command[B, C] ) extends Command[A, `C`] ``` --- ## Witness type > A witness type describes properties of a sum type's variants at the type level. --- ## GADT > A GADT is a sum type with one or more witness types [...] --- ## Type equality ```scala def movingLabel(cmd: Command[`Moving`, _]) = cmd match case Command.Stop => "stop" //case _: Command.Chain[_, _, _] => "chain" case Command.Start => "start" ``` --- ## Type equality ```scala def movingLabel(cmd: Command[Moving, _]) = cmd match case Command.Stop => "stop" `//case _: Command.Chain[_, _, _] => "chain"` case Command.Start => "start" ``` --- ## Type equality ```scala def movingLabel(cmd: Command[Moving, _]) = cmd match case Command.Stop => "stop" //case _: Command.Chain[_, _, _] => "chain" `case Command.Start => "start"` ``` --- ## Type equality > Type equality is information available to the compiler about each witness type, allowing it to refine pattern matches. --- ## GADT > A GADT is a sum type with one or more witness types, each equipped with a type equality. --- ## [Key takeaways](#closing) -- * A GADT is a sum type with one or more witness types, each equipped with a type equality. --- ## [Key takeaways](#closing) * ~~A GADT is a sum type with one or more witness types, each equipped with a type equality.~~ -- * A GADTs is a sum type with interesting properties for pattern matching. --- class: center, middle name: algebra # The algebra of types --- ## Cardinality > The cardinality of type `A` is written \\(\\vert A \\vert\\) and is the number of values of type `A`. --- ## Cardinality .center[data:image/s3,"s3://crabby-images/35d1b/35d1b1b3e1d12f41063984c47e29c7fbd52e9cd6" alt="Nothing"] --- ## Cardinality .center[data:image/s3,"s3://crabby-images/4e07d/4e07dd564dd28b6e2f1e473a23e8940200bbe057" alt="Nothing"] -- \\(\vert Nothing \\vert = 0\\) --- ## Cardinality .center[data:image/s3,"s3://crabby-images/99094/99094ba7354b47375e8418167cb6be38e37b6ea7" alt="Unit"] --- ## Cardinality .center[data:image/s3,"s3://crabby-images/d968d/d968dfccce513b517d0d031998a5d40fa3063204" alt="Unit"] -- \\(\vert Unit \\vert = 1\\) --- ## Cardinality .center[data:image/s3,"s3://crabby-images/45b70/45b701f21b47c10a08879e8d5d301f45bd021922" alt="Boolean"] --- ## Cardinality .center[data:image/s3,"s3://crabby-images/79e92/79e92a332c55310e2fa50827e757c544fefad673" alt="Boolean"] --- ## Cardinality .center[data:image/s3,"s3://crabby-images/f3199/f3199ff708e2c7dfabba7ae6540e9d9b6b8d574c" alt="Boolean"] -- \\(\vert Boolean \\vert = 2\\) --- ## Sum types ```scala enum +[+A, +B]: case Left[A](value: A) extends +[A, Nothing] case Right[B](value: B) extends +[Nothing, B] ``` --- ## Sum types ```scala enum `+[+A, +B]`: case Left[A](value: A) extends +[A, Nothing] case Right[B](value: B) extends +[Nothing, B] ``` --- ## Sum types ```scala enum +[+A, +B]: case `Left[A]`(value: A) extends +[A, Nothing] case Right[B](value: B) extends +[Nothing, B] ``` --- ## Sum types ```scala enum +[+A, +B]: case Left[A](`value: A`) extends +[A, Nothing] case Right[B](value: B) extends +[Nothing, B] ``` --- ## Sum types ```scala enum +[+A, +B]: case Left[A](value: A) extends +[A, Nothing] case `Right[B]`(value: B) extends +[Nothing, B] ``` --- ## Sum types ```scala enum +[+A, +B]: case Left[A](value: A) extends +[A, Nothing] case Right[B](`value: B`) extends +[Nothing, B] ``` --- ## Sum types .diff-rm[ ```scala enum +[+A, +B]: * case Left[A](value: A) extends `+[A, Nothing]` * case Right[B](value: B) extends `+[Nothing, B]` ``` ] --- ## Sum types .diff-add[ ```scala enum +[+A, +B]: * case Left[A](value: A) extends `(A + Nothing)` * case Right[B](value: B) extends `(Nothing + B)` ``` ] --- ## Sum types .center[data:image/s3,"s3://crabby-images/837fb/837fbb42f949a8dc770c649772d1f38b0a10d1f6" alt="+"] --- ## Sum types .center[data:image/s3,"s3://crabby-images/0538b/0538b2d67729f98128582b4b43d7d6c8980c5679" alt="+"] --- ## Sum types .center[data:image/s3,"s3://crabby-images/bb7c2/bb7c276dafb10c97d4c3a93f666058b15c1b14ba" alt="+"] --- ## Sum types .center[data:image/s3,"s3://crabby-images/fd1a5/fd1a574a09ec25ee1a560db41af736a90718cabe" alt="+"] \\(\vert A + B \\vert = \vert A \vert \ldots\\) --- ## Sum types .center[data:image/s3,"s3://crabby-images/01f67/01f674e47de1e218ea622677122584ec17d5fa61" alt="+"] \\(\vert A + B \\vert = \vert A \vert + \vert B \vert \\) --- ## Type isomorphisms > We'll say that types `A` and `B` are isomorphic if they have the same cardinality, and write `A ~= B`. --- ## Type isomorphisms > We'll say that types `A` and `B` are isomorphic if they have the same cardinality, and write .highlight[`A ~= B`]. --- ## Sum types .section[Algebra] ```scala 1 + 1 = 2 ``` -- .section[Types] ```scala Unit + Unit ~= Boolean ``` --- ## Sum types .section[Algebra] ```scala `1` + `1` = 2 ``` .section[Types] ```scala `Unit` + `Unit` ~= Boolean ``` --- ## Sum types .section[Algebra] ```scala 1 + 1 = `2` ``` .section[Types] ```scala Unit + Unit ~= `Boolean` ``` --- ## Sum types .section[Algebra] ```scala 1 `+` 1 = 2 ``` .section[Types] ```scala Unit `+` Unit ~= Boolean ``` --- ## Sum types .section[Algebra] ```scala 1 + 1 `=` 2 ``` .section[Types] ```scala Unit + Unit `~=` Boolean ``` --- ## Sum types .center[data:image/s3,"s3://crabby-images/af99a/af99acf259dca3eaa543452f2a4e47608b708281" alt="Unit + Unit = Boolean"] --- ## Sum types .center[data:image/s3,"s3://crabby-images/4b636/4b6363ae65298b1091b616f2cfd3db840e30edb5" alt="Unit + Unit = Boolean"] --- ## Sum types .center[data:image/s3,"s3://crabby-images/e5a5f/e5a5fff8e04dd42d482df73960e2e6497d171392" alt="Unit + Unit = Boolean"] --- ## Sum types .center[data:image/s3,"s3://crabby-images/34bd0/34bd0e43149cbe47180c19885e861968459ab3e4" alt="Unit + Unit = Boolean"] --- ## Sum types .center[data:image/s3,"s3://crabby-images/22917/229176e4c5eb7a4f12edee6137fa232901cde382" alt="Unit + Unit = Boolean"] --- ## Sum types .center[data:image/s3,"s3://crabby-images/dc696/dc696b8a15f11828bf8f4b4465062ea5296c5ecf" alt="Unit + Unit = Boolean"] --- ## Sum types associativity .section[Algebra] ```scala a + (b + c) = (a + b) + c ``` -- .section[Types] ```scala A + (B + C) ~= (A + B) + C ``` --- ## Sum types associativity .center[data:image/s3,"s3://crabby-images/d5a8b/d5a8bf5a568d915eed4622f65a2a4dca97c45e1c" alt="A + B + C"] --- ## Sum types associativity .center[data:image/s3,"s3://crabby-images/fb356/fb356b79e6566d418b265e5aec4b73ac8a1343ce" alt="A + B + C"] --- ## Sum types associativity .center[data:image/s3,"s3://crabby-images/94f1d/94f1d9c495cb5d2c1b6d030ca13857c751a1461c" alt="A + B + C"] --- ## Sum types associativity .center[data:image/s3,"s3://crabby-images/d7719/d7719c84f4f4cdcb855d8799137212bc6c2c39c0" alt="A + B + C"] --- ## Sum types associativity .center[data:image/s3,"s3://crabby-images/e6a8f/e6a8ff13e63e1e36542a9a269645747b2c10cd12" alt="A + B + C"] --- ## Sum types associativity .center[data:image/s3,"s3://crabby-images/9da18/9da18a03dfb8b125a4242b77905aad29a88c6c70" alt="A + B + C"] --- ## Sum types associativity .center[data:image/s3,"s3://crabby-images/357a5/357a5c74576ebc3594b5db1fb3da6e1e6875ad55" alt="A + B + C"] --- ## Sum types associativity .center[data:image/s3,"s3://crabby-images/4171c/4171c29de92e5c23594ee7019009f4c6a320bf2d" alt="A + B + C"] --- ## Sum types commutativity .section[Algebra] ```scala a + b = b + a ``` -- .section[Types] ```scala A + B ~= B + A ``` --- ## Sum types commutativity .center[data:image/s3,"s3://crabby-images/8d35a/8d35aa7fdf8593161ef4c667921bf5404a2076bb" alt="A + B"] --- ## Sum types commutativity .center[data:image/s3,"s3://crabby-images/5a390/5a390081d1e23c3769680e4e8a2ccc37e91cc36b" alt="A + B"] --- ## Sum types commutativity .center[data:image/s3,"s3://crabby-images/b6778/b677893805fcbba2cd8e956bcd433315d4696a7b" alt="A + B"] --- ## Sum types commutativity .center[data:image/s3,"s3://crabby-images/6fe91/6fe912d0d9fdce709fa3fbbe6db2dc1cc7946d4b" alt="A + B"] --- ## Sum types commutativity .center[data:image/s3,"s3://crabby-images/6784f/6784f285f8fbca1cd355c6a844ca74569c3ee32a" alt="A + B"] --- ## Sum types commutativity .center[data:image/s3,"s3://crabby-images/5e267/5e26783e372dede3f9e67426cbf5f276ab031181" alt="A + B"] --- ## Sum types neutral element .section[Algebra] ```scala a + 0 = a ``` -- .section[Types] ```scala A + Nothing ~= A ``` --- ## Sum types neutral element .center[data:image/s3,"s3://crabby-images/ed55d/ed55d4900baba7e2712f3dffa33d43eed5b23060" alt="A + B"] --- ## Sum types neutral element .center[data:image/s3,"s3://crabby-images/41067/41067adf4aff16f3c6ac0cd43634a91d130e5f99" alt="A + B"] --- ## Sum types neutral element .center[data:image/s3,"s3://crabby-images/64a3f/64a3f1c9caad67586b749b3580ee0605bca26a41" alt="A + B"] --- ## Sum types neutral element .center[data:image/s3,"s3://crabby-images/2d929/2d929bba64cfb61f78fffec4f2020629e092c5c6" alt="A + B"] --- ## Sum types neutral element .center[data:image/s3,"s3://crabby-images/a59b9/a59b9afde516956dc767e581934657b6f54e4946" alt="A + B"] --- ## Product types ```scala case class ✕[A, B](first: A, second: B) ``` --- ## Product types ```scala case class `✕[A, B]`(first: A, second: B) ``` --- ## Product types ```scala case class ✕[A, B](`first: A`, second: B) ``` --- ## Product types ```scala case class ✕[A, B](first: A, `second: B`) ``` --- ## Product types ```scala extension [A](a: A) def ✕[B](b: B): ✕[A, B] = new ✕(a, b) ``` --- ## Product types ```scala extension [A](`a: A`) def ✕[B](b: B): ✕[A, B] = new ✕(a, b) ``` --- ## Product types ```scala extension [A](a: A) def `✕`[B](b: B): ✕[A, B] = new ✕(a, b) ``` --- ## Product types ```scala extension [A](a: A) def ✕[B](`b: B`): ✕[A, B] = new ✕(a, b) ``` --- ## Product types ```scala extension [A](a: A) def ✕[B](b: B): `✕[A, B]` = new ✕(a, b) ``` --- ## Product types .diff-rm[ ```scala extension [A](a: A) * def ✕[B](b: B): `✕[A, B]` = new ✕(a, b) ``` ] --- ## Product types .diff-add[ ```scala extension [A](a: A) * def ✕[B](b: B): `A ✕ B` = new ✕(a, b) ``` ] --- ## Product types .center[data:image/s3,"s3://crabby-images/00ad2/00ad2497e65df1499d3790cea796a4151dc9229c" alt="✕"] --- ## Product types .center[data:image/s3,"s3://crabby-images/9b27d/9b27da4d59700c387ff8f3746ca7bc6287c8ab98" alt="✕"] --- ## Product types .center[data:image/s3,"s3://crabby-images/1d300/1d300dc37b941ea1643aef7a2d20555195368da5" alt="✕"] --- ## Product types .center[data:image/s3,"s3://crabby-images/57efb/57efbf69d3b873a97c979b56372392207ca0f890" alt="✕"] --- ## Product types .center[data:image/s3,"s3://crabby-images/79935/79935ecd5a6d749dedf7899dd0036fe1054c2e4a" alt="✕"] \\(\vert A \times B \\vert = \vert A \vert \ldots\\) --- ## Product types .center[data:image/s3,"s3://crabby-images/da77a/da77ace03f5c39b5a98f46515fa0546f2c699c93" alt="✕"] \\(\vert A \times B \\vert = \vert A \vert \times \vert B \vert \\) --- ## Product types .section[Algebra] ```scala a + a = 2 * a ``` -- .section[Types] ```scala A + A ~= Boolean ✕ A ``` --- ## Product types .section[Algebra] ```scala a + a = 2 `*` a ``` .section[Types] ```scala A + A ~= Boolean `✕` A ``` --- ## Product types .center[data:image/s3,"s3://crabby-images/ecdfa/ecdfa16238600b293392f7fb87e1a5d8a3bae2a3" alt="A + A"] --- ## Product types .center[data:image/s3,"s3://crabby-images/7a83b/7a83be6b604d5359050a487c73e96be7bcd4f4cf" alt="A + A"] --- ## Product types .center[data:image/s3,"s3://crabby-images/9f12a/9f12a894e0910c4ce5e87496726d3b1cdffe125f" alt="A + A"] --- ## Product types .center[data:image/s3,"s3://crabby-images/3e0e6/3e0e611260ed438ec982e3e6afb3e6d018b8d745" alt="A + A"] --- ## Product types .center[data:image/s3,"s3://crabby-images/af6e1/af6e14bd0fb58310405cbf7284c19dbccc6fee9e" alt="A + A"] --- ## Product types .center[data:image/s3,"s3://crabby-images/8cb38/8cb38c043cc901db6d60a374a88feffb87d06a5a" alt="A + A"] --- ## Product types associativity .section[Algebra] ```scala (a * b) * c = a * (b * c) ``` -- .section[Types] ```scala (A ✕ B) ✕ C ~= A ✕ (B ✕ C) ``` --- ## Product types associativity .center[data:image/s3,"s3://crabby-images/00e9a/00e9a3313c184c893c6f71191af43df4ef52a81f" alt="A ✕ B ✕ C"] --- ## Product types associativity .center[data:image/s3,"s3://crabby-images/f7a88/f7a8848f7ffb12a04b28018ca2ef01b664365f07" alt="A ✕ B ✕ C"] --- ## Product types associativity .center[data:image/s3,"s3://crabby-images/be38e/be38ea7be0c5f6c6573e0d3592a0863b00169974" alt="A ✕ B ✕ C"] --- ## Product types associativity .center[data:image/s3,"s3://crabby-images/77a6e/77a6e460ed4531bc0af7e0c4a38f557c76b2defe" alt="A ✕ B ✕ C"] --- ## Product types commutativity .section[Algebra] ```scala a * b = b * a ``` -- .section[Types] ```scala A ✕ B ~= B ✕ A ``` --- ## Product types commutativity .center[data:image/s3,"s3://crabby-images/cb9bf/cb9bf557a98000f6c036b40fcdf78a09437436e9" alt="A ✕ B"] --- ## Product types commutativity .center[data:image/s3,"s3://crabby-images/650a0/650a045ee5037cf63a5c54d98ddd23339601d6f8" alt="A ✕ B"] --- ## Product types commutativity .center[data:image/s3,"s3://crabby-images/ac65b/ac65ba9768091d6a002b58af7b6bf0acc5a2e628" alt="A ✕ B"] --- ## Product types commutativity .center[data:image/s3,"s3://crabby-images/b3b7c/b3b7c660e8228a36ca4ceef9054580aa30d236b7" alt="A ✕ B"] --- ## Product types neutral element .section[Algebra] ```scala a * 1 = a ``` -- .section[Types] ```scala A ✕ Unit ~= A ``` --- ## Product types neutral element .center[data:image/s3,"s3://crabby-images/ea2a0/ea2a0995d5ffbdce49d34b6f6e5d86c78e8daf9c" alt="A ✕ Unit"] --- ## Product types neutral element .center[data:image/s3,"s3://crabby-images/d6a86/d6a860f4288360a7d1ea6e054d9e1fb6ce379c4f" alt="A ✕ Unit"] --- ## Product types neutral element .center[data:image/s3,"s3://crabby-images/93347/93347ee36829d1db00b48ec263fca0f7741aa176" alt="A ✕ Unit"] --- ## Product types neutral element .center[data:image/s3,"s3://crabby-images/3a172/3a172d56f02d5b7fc920eca064608db7d497fff5" alt="A ✕ Unit"] --- ## Recursive types ```scala enum List[+A]: case Nil case Cons(head: A, tail: List[A]) ``` --- ## Recursive types ```scala enum `List[+A]`: case Nil case Cons(head: A, tail: List[A]) ``` --- ## Recursive types ```scala enum List[+A]: case `Nil` case Cons(head: A, tail: List[A]) ``` --- ## Recursive types ```scala enum List[+A]: case Nil case `Cons`(head: A, tail: List[A]) ``` --- ## Recursive types ```scala enum List[+A]: case Nil case Cons(`head: A`, tail: List[A]) ``` --- ## Recursive types ```scala enum List[+A]: case Nil case Cons(head: A, `tail: List[A]`) ``` --- ## Recursive types ```scala enum `List[+A]`: case Nil case Cons(head: A, tail: List[A]) ``` --- ## List of fixed size \\( \vert List_0[A] \vert\\) --- ## List of fixed size \\( \vert List_0[A] \vert = \vert Nil \vert\\) --- ## List of fixed size \\( \vert List_0[A] \vert = 1\\) --- ## List of fixed size \\( \vert List_1[A] \vert\\) --- ## List of fixed size \\( \vert List_1[A] \vert = \vert A \times List_0[A] \vert\\) --- ## List of fixed size \\( \vert List_1[A] \vert = \vert A \vert \times \vert List_0[A] \vert\\) --- ## List of fixed size \\( \vert List_1[A] \vert = \vert A \vert \times 1\\) --- ## List of fixed size \\( \vert List_1[A] \vert = \vert A \vert\\) --- ## List of fixed size \\( \vert List_2[A] \vert\\) --- ## List of fixed size \\( \vert List_2[A] \vert = \vert A \times List_1[A] \vert\\) --- ## List of fixed size \\( \vert List_2[A] \vert = \vert A \vert \times \vert List_1[A] \vert\\) --- ## List of fixed size \\( \vert List_2[A] \vert = \vert A \vert \times \vert A \vert\\) --- ## List of fixed size \\( \vert List_2[A] \vert = \vert A \vert^2\\) --- ## List of fixed size \\( \vert List_n[A] \vert = \vert A \vert^n\\) --- ## List of maximum size \\( \vert List^n[A] \vert\\) --- ## List of maximum size \\(\\vert List^n[A] \\vert = \vert \displaystyle\sum_{i = 0}^n List_i[A] \vert^i\\) --- ## List of maximum size \\(\\vert List^n[A] \\vert = \displaystyle\sum_{i = 0}^n \vert List_i[A] \vert\\) --- ## List of maximum size \\(\\vert List^n[A] \\vert = \displaystyle\sum_{i = 0}^n \vert A \vert^i\\) --- ## List of maximum size \\(\\vert List^n[A] \\vert = \frac{\vert A \vert^{n + 1} - 1}{\vert A \vert - 1}\\) --- ## List of maximum size \\(\\vert List^2[Boolean] \\vert = \frac{\vert Boolean \vert^{2 + 1} - 1}{\vert Boolean \vert - 1}\\) --- ## List of maximum size \\(\\vert List^2[Boolean] \\vert = \frac{2^{2 + 1} - 1}{2 - 1}\\) --- ## List of maximum size \\(\\vert List^2[Boolean] \\vert = 7\\) --- ## List of maximum size .center[data:image/s3,"s3://crabby-images/5fdfe/5fdfe0020d72459e600393b33e3b4447f1a4157b" alt="List"] --- ## List of maximum size .center[data:image/s3,"s3://crabby-images/9f900/9f900637ada595c665c209f8fe9ee72f61d8da01" alt="List"] --- ## List of maximum size .center[data:image/s3,"s3://crabby-images/beace/beace7ad0fb7522e2504369fe0d0c85b963cf084" alt="List"] --- ## List of maximum size .center[data:image/s3,"s3://crabby-images/ba55f/ba55f50429ce58ca1453668e8fa2592c2711dd6a" alt="List"] --- ## List of maximum size .center[data:image/s3,"s3://crabby-images/a798a/a798ad4da6192b87138015238119ea8870fe56d1" alt="List"] --- ## [Key takeaways](#closing) -- * ADTs have a deep connection to the algebra you know. -- * you can use this connection to prove fun facts about types. -- * you can also use it to pad talks and look clever. --- class: center, middle name: closing # In closing --- ## If you only remember 1 slide... -- * ADTs make data structures simpler and safer. -- * So do GADTs, only more so. -- * Thinking about data structures that way will have a lasting impact on the way you write code. --- class: center, middle name: questions [
][Slides] [Nicolas Rinaudo] • [@NicolasRinaudo@functional.cafe] [@NicolasRinaudo@functional.cafe]:https://functional.cafe/@NicolasRinaudo [Nicolas Rinaudo]:https://nrinaudo.github.io/ [Slides]:https://nrinaudo.github.io/far-more-adt/ --- ## Table of contents * [Representing Direction](#direction) * [Representing Command](#command) * [Composing commands](#composition) * [Algebraic Data Types](#adt) * [Safe command composition](#safe-composition) * [Generalised Algebraic Data Types](#gadt) * [The algebra of types](#algebra)