class: center, middle # Things that are things, but not other things [Nicolas Rinaudo] • [@NicolasRinaudo@functional.cafe] ??? I want to talk about categorical abstractions - monads, and stuff. Not because this is a particularly original topic, but because, well. I've never been to Warsaw, I really wanted to, and submitting a talk on monads is usually a very good way of getting through a CFP, at least in the Scala community. So far, everything seems to be working according to plan. The problem, of course, is that I now have to find something interesting to say about monads. And one thing that I often found frustrating when learning about that family of abstractions is how just about everything is a monad - why even have all these intermediate abstractions if everything is all the things, all the time? This ties in to one of the ways I learn things: when given a definition of a thing, it helps me just as much to have examples of things that are *not* the thing as of things that are the thing. I really wished I'd had such examples of things that are things, but not other things, while learning abstractions. This talk is my attempt at providing such examples. --- ## Overview .center[data:image/s3,"s3://crabby-images/fc50b/fc50bceee15634ded2878e19f878694d373d5013" alt="Overview"] --- ## Overview .center[data:image/s3,"s3://crabby-images/15cf4/15cf4b900504f62406faf0cb79d086e5eb11a390" alt="Overview"] --- ## Overview .center[data:image/s3,"s3://crabby-images/113ec/113ecdbdd71650ed35d96a008691dffa050ad290" alt="Overview"] --- ## Overview .center[data:image/s3,"s3://crabby-images/ff251/ff25193b162978a37d5dd59ca45e2cbe61235b6f" alt="Overview"] --- ## Overview .center[data:image/s3,"s3://crabby-images/ae47a/ae47a64de535d09d60a9f93c01253e8038e6265b" alt="Overview"] --- ## Overview .center[data:image/s3,"s3://crabby-images/4ed2d/4ed2d4c66fea665c825b896884fdaa6b50c0cf1c" alt="Overview"] --- class: center, middle # Warm up: not a Functor --- ## Functor .center[data:image/s3,"s3://crabby-images/fc50b/fc50bceee15634ded2878e19f878694d373d5013" alt="Overview"] --- ## Functor .center[data:image/s3,"s3://crabby-images/8f330/8f33003863957773ddec5bd2dade37f9f940e907" alt="Overview"] --- ## Functor .center[data:image/s3,"s3://crabby-images/23111/231113983929f8a4e255ba345e8a3322977abfab" alt="Functor"] ```scala def map[A, B](f: A => B)(`fa: F[A]`): F[B] ``` --- ## Functor .center[data:image/s3,"s3://crabby-images/560c6/560c6621026f44808037e59398f95b54bb8b5764" alt="Functor"] ```scala def map[A, B](f: A => B)(fa: F[A]): `F[B]` ``` --- ## Functor .center[data:image/s3,"s3://crabby-images/9dff1/9dff19abfebaa42cead8a2a179ef9d3fb6f3fe18" alt="Functor"] ```scala def map[A, B](`f: A => B`)(fa: F[A]): F[B] ``` --- ## Functor .center[data:image/s3,"s3://crabby-images/8fe64/8fe648143adc14c97fbcf1a60e91591e6c22a349" alt="Functor"] ```scala def `map`[A, B](f: A => B)(fa: F[A]): F[B] ``` --- ## Functor .center[data:image/s3,"s3://crabby-images/47da8/47da8842af6e4a976bb0073b2516659b6dfd9da1" alt="Functor"] ```scala def map[A, B](f: A => B)(fa: F[A]): F[B] ``` --- ## Function .center[data:image/s3,"s3://crabby-images/3c8dc/3c8dca6fc0d0f41068dc739757d15c2b1f84ac93" alt="Overview"] ```scala type F[A] = `X => A` def map[A, B](f: A => B)(fa: F[A]): F[B] = ??? ``` --- ## Function .center[data:image/s3,"s3://crabby-images/fe83c/fe83c6e47177902e784cd41c46e510d4c64dbf7b" alt="Overview"] ```scala type F[A] = X => A def map[A, B](f: A => B)(`fa: F[A]`): F[B] = ??? ``` --- ## Function .center[data:image/s3,"s3://crabby-images/982aa/982aae33b0237ccd69cb53e49d1b32df279b0cdb" alt="Overview"] ```scala type F[A] = X => A def map[A, B](`f: A => B`)(fa: F[A]): F[B] = ??? ``` --- ## Function .center[data:image/s3,"s3://crabby-images/17086/17086d5c776ae708d1f73d8d67d9466849c96d63" alt="Overview"] ```scala type F[A] = X => A def map[A, B](f: A => B)(fa: F[A]): `F[B]` = ??? ``` --- ## Function .center[data:image/s3,"s3://crabby-images/3c8dc/3c8dca6fc0d0f41068dc739757d15c2b1f84ac93" alt="Overview"] .diff-rm[ ```scala type F[A] = X => A def map[A, B](f: A => B)(fa: F[A]): F[B] = * `???` ``` ] --- ## Function .center[data:image/s3,"s3://crabby-images/96a6e/96a6e28ea4ce4f4dad596048d0d5b6fdce7d8ada" alt="Overview"] .diff-add[ ```scala type F[A] = X => A def map[A, B](f: A => B)(fa: F[A]): F[B] = * `fa andThen f` ``` ] --- ## Breaking Function .center[data:image/s3,"s3://crabby-images/a7502/a75024bb9f41307c81637bbc7ff71634fa50e69c" alt="Overview"] .diff-rm[ ```scala *type F[A] = `X => A` def map[A, B](f: A => B)(fa: F[A]): F[B] = fa andThen f ``` ] --- ## Breaking Function .center[data:image/s3,"s3://crabby-images/7f8c2/7f8c2f382e108c89d28f42caba5f65bb5aa0e1a3" alt="Overview"] .diff-add[ ```scala *type F[A] = `A => X` def map[A, B](f: A => B)(fa: F[A]): F[B] = fa andThen f ``` ] --- ## Breaking Function .center[data:image/s3,"s3://crabby-images/6a8cc/6a8cc960aae33113f15bb20d122d86f7ac8f5fbe" alt="Overview"] .diff-rm[ ```scala type F[A] = A => X def map[A, B](f: A => B)(fa: F[A]): F[B] = * `fa andThen f` ``` ] --- ## Breaking Function .center[data:image/s3,"s3://crabby-images/09566/0956647b3eb14ea4aaeb3ed3cea80f51490ce555" alt="Overview"] .diff-rm[ ```scala type F[A] = A => X * *`def map[A, B](f: A => B)(fa: F[A]): F[B] =` * `fa andThen f` ``` ] --- ## Breaking Function .center[data:image/s3,"s3://crabby-images/a4826/a4826cff09460a8d9703227853a1e802de6718b1" alt="Overview"] ```scala type F[A] = A => X ``` --- ## Breaking Function .center[data:image/s3,"s3://crabby-images/7a1ff/7a1ffadc7208128d68f3ebd86106772a459b773e" alt="Overview"] .diff-rm[ ```scala *type F[A] = A => `X` ``` ] --- ## Breaking Function .center[data:image/s3,"s3://crabby-images/c9999/c9999764c0339ef2c88317deefc66fd69e665d6b" alt="Overview"] .diff-add[ ```scala *type F[A] = A => `Boolean` ``` ] --- ## Breaking Function .center[data:image/s3,"s3://crabby-images/1c1d4/1c1d4072f647162dc8a6ec09471004f8d8f9549c" alt="Overview"] .diff-rm[ ```scala *type `F[A]` = A => Boolean ``` ] --- ## Predicate .center[data:image/s3,"s3://crabby-images/e3da3/e3da3d4bb7b5b6d711987b76bca9add50fe6c951" alt="Overview"] .diff-add[ ```scala *type `Predicate[A]` = A => Boolean ``` ] --- ## Predicate .center[data:image/s3,"s3://crabby-images/1044b/1044b0337f049d9a49b0028f45a9728d58584fdc" alt="Overview"] ```scala type Predicate[A] = A => Boolean ``` --- ## Key takeaways A function parameterised on its: -- * output type is a `Functor`. -- * input type is *not* a `Functor`. --- class: center, middle # Interlude: product types --- ## Products .center[data:image/s3,"s3://crabby-images/43ece/43ece0d02edefdc6a8af5d3011c593d160aead8d" alt="Product type"] ```scala case class `×`[A, B](first: A, second: B) ``` --- ## Products .center[data:image/s3,"s3://crabby-images/43ece/43ece0d02edefdc6a8af5d3011c593d160aead8d" alt="Product type"] ```scala case class ×[A, B](`first: A`, second: B) ``` --- ## Products .center[data:image/s3,"s3://crabby-images/43ece/43ece0d02edefdc6a8af5d3011c593d160aead8d" alt="Product type"] ```scala case class ×[A, B](first: A, `second: B`) ``` --- ## Product introduction .center[data:image/s3,"s3://crabby-images/b2c0a/b2c0a8ef804089f799c6fe4dc26b18497c6020c1" alt="Product type"] ```scala def build[A, B](`a: A`, `b: B`): A×B = a×b ``` --- ## Product introduction .center[data:image/s3,"s3://crabby-images/ee9c6/ee9c607386bf52c8b311be012c64e86a7cd8843c" alt="Product type"] ```scala def build[A, B](a: A, b: B): `A×B` = a×b ``` --- ## Product introduction .center[data:image/s3,"s3://crabby-images/40b35/40b3555e9d7a50844d529ec98a8e6ad6e656142f" alt="Product type"] ```scala def build[A, B](a: A, b: B): A×B = `a×b` ``` --- ## Product elimination .center[data:image/s3,"s3://crabby-images/5a8a7/5a8a7f17bc1445834b305bd9283e86edb001ec8b" alt="Product type"] ```scala def print[A, B](ab: `A×B`) = val a×b = ab println(s"$a and $b") ``` --- ## Product elimination .center[data:image/s3,"s3://crabby-images/7d3e9/7d3e99d6e49adcfca02047aa5e15b2129fa6a9aa" alt="Product type"] ```scala def print[`A`, `B`](ab: A×B) = val a×b = ab println(s"$a and $b") ``` --- ## Product elimination .center[data:image/s3,"s3://crabby-images/e23de/e23de6c013c097dc266dfa787878f84315ab4139" alt="Product type"] ```scala def print[A, B](ab: A×B) = val `a×b` = ab println(s"$a and $b") ``` --- ## Products are functors .center[data:image/s3,"s3://crabby-images/a9da3/a9da316be62a401c799acb3329766e378a121296" alt="Overview"] ```scala type F[A] = `A×X` def map[A, B](f: A => B)(fa: F[A]): F[B] = ??? ``` --- ## Products are functors .center[data:image/s3,"s3://crabby-images/1fc96/1fc96aa4147c91b8d05c8ab482d0d6b9458fa8f4" alt="Overview"] ```scala type F[A] = A×X def map[A, B](`f: A => B`)(fa: F[A]): F[B] = ??? ``` --- ## Products are functors .center[data:image/s3,"s3://crabby-images/c84d2/c84d2499c5f0cba34f18c3f5fd18812c03f6c533" alt="Overview"] ```scala type F[A] = A×X def map[A, B](f: A => B)(`fa: F[A]`): F[B] = ??? ``` --- ## Products are functors .center[data:image/s3,"s3://crabby-images/6af17/6af179bdbaceba988ca0ad90a53196bc19663a66" alt="Overview"] ```scala type F[A] = A×X def map[A, B](f: A => B)(fa: F[A]): `F[B]` = ??? ``` --- ## Products are functors .center[data:image/s3,"s3://crabby-images/9fbb5/9fbb572a9cf26310ef2348c2612ced0a1cdebf28" alt="Overview"] .diff-rm[ ```scala type F[A] = A×X def map[A, B](f: A => B)(fa: F[A]): F[B] = * `???` ``` ] --- ## Products are functors .center[data:image/s3,"s3://crabby-images/9fbb5/9fbb572a9cf26310ef2348c2612ced0a1cdebf28" alt="Overview"] .diff-add[ ```scala type F[A] = A×X def map[A, B](f: A => B)(fa: F[A]): F[B] = * `(??? : B)×(??? : X)` ``` ] --- ## Products are functors .center[data:image/s3,"s3://crabby-images/b8052/b805293d345f62c444355612a762e9368ea4c46c" alt="Overview"] ```scala type F[A] = A×X def map[A, B](f: A => B)(`fa: F[A]`): F[B] = (??? : B)×(??? : X) ``` --- ## Products are functors .center[data:image/s3,"s3://crabby-images/1af56/1af5686ec489fdb7a8bc049687a10fbfa80204cd" alt="Overview"] .diff-add[ ```scala type F[A] = A×X def map[A, B](f: A => B)(fa: F[A]): F[B] = * `val (a×x) = fa` * (??? : B)×(??? : X) ``` ] --- ## Products are functors .center[data:image/s3,"s3://crabby-images/9ef2d/9ef2d1bd0f3920d79c2ec20364cf8a562b3ae2bc" alt="Overview"] ```scala type F[A] = A×X def map[A, B](`f: A => B`)(fa: F[A]): F[B] = val (`a`×x) = fa (??? : B)×(??? : X) ``` --- ## Products are functors .center[data:image/s3,"s3://crabby-images/9ef2d/9ef2d1bd0f3920d79c2ec20364cf8a562b3ae2bc" alt="Overview"] .diff-add[ ```scala type F[A] = A×X def map[A, B](f: A => B)(fa: F[A]): F[B] = val (a×x) = fa * `val b = f(a)` (??? : B)×(??? : X) ``` ] --- ## Products are functors .center[data:image/s3,"s3://crabby-images/6341c/6341cf94885633c634452b805d99ac627ec94220" alt="Overview"] ```scala type F[A] = A×X def map[A, B](f: A => B)(fa: F[A]): F[B] = val (a×`x`) = fa val `b` = f(a) (??? : B)×(??? : X) ``` --- ## Products are functors .center[data:image/s3,"s3://crabby-images/e4eef/e4eef74e5d31f2b3d5c9204018b4be5a28323ed8" alt="Overview"] .diff-rm[ ```scala type F[A] = A×X def map[A, B](f: A => B)(fa: F[A]): F[B] = val (a×x) = fa val b = f(a) * `(??? : B)`×`(??? : X)` ``` ] --- ## Products are functors .center[data:image/s3,"s3://crabby-images/e4eef/e4eef74e5d31f2b3d5c9204018b4be5a28323ed8" alt="Overview"] .diff-add[ ```scala type F[A] = A×X def map[A, B](f: A => B)(fa: F[A]): F[B] = val (a×x) = fa val b = f(a) * `b`×`x` ``` ] --- ## Products are functors .center[data:image/s3,"s3://crabby-images/7807b/7807b076612fc27f6eeb3c954a70122c22d81faa" alt="Overview"] ```scala type F[A] = A×X def map[A, B](f: A => B)(fa: F[A]): F[B] = val (a×x) = fa val b = f(a) b×x ``` --- ## Key takeaways A binary parametric product type is: -- * a `Functor`. --- class: center, middle # Functor but not Apply --- ## Apply .center[data:image/s3,"s3://crabby-images/fee18/fee18347fe198888ad0b0afe03c493798130b434" alt="Overview"] --- ## Apply .center[data:image/s3,"s3://crabby-images/314e1/314e10834f13fe886d48e07bc2e690534abfc804" alt="Overview"] --- ## Apply .center[data:image/s3,"s3://crabby-images/9ba7e/9ba7ef815cd89b00363a27931c9a30d6a621bac1" alt="Apply"] ```scala def map2[A, B, C](f: (A, B) => C)(`fa: F[A], fb: F[B]`): F[C] ``` --- ## Apply .center[data:image/s3,"s3://crabby-images/88e9f/88e9fb1089f6a049526b7da91547ee465f9bcb28" alt="Apply"] ```scala def map2[A, B, C](f: (A, B) => C)(fa: F[A], fb: F[B]): `F[C]` ``` --- ## Apply .center[data:image/s3,"s3://crabby-images/3957f/3957f9a9b0b2fd7fb4d14a16f9949349881a0ded" alt="Apply"] ```scala def map2[A, B, C](`f: (A, B) => C`)(fa: F[A], fb: F[B]): F[C] ``` --- ## Apply .center[data:image/s3,"s3://crabby-images/5ad63/5ad6315a243d392b6827cb25b442b4b5e46c7542" alt="Apply"] ```scala def `map2`[A, B, C](f: (A, B) => C)(fa: F[A], fb: F[B]): F[C] ``` --- ## Apply .center[data:image/s3,"s3://crabby-images/14751/14751d852c8a255b484628bc33103576f3c47806" alt="Apply"] ```scala def map2[A, B, C](f: (A, B) => C)(fa: F[A], fb: F[B]): F[C] ``` --- ## Product .center[data:image/s3,"s3://crabby-images/27cb5/27cb5df94c76dc7b7437714413544462dc53563e" alt="Overview"] ```scala type F[A] = `A×X` def map2[A, B, C](f: (A, B) => C)(fa: F[A], fb: F[B]): F[C] = ??? ``` --- ## Product .center[data:image/s3,"s3://crabby-images/1dc40/1dc406e1344adbb49891cb605e02efdd8c799fd6" alt="Overview"] ```scala type F[A] = A×X def map2[A, B, C](`f: (A, B) => C`)(fa: F[A], fb: F[B]): F[C] = ??? ``` --- ## Product .center[data:image/s3,"s3://crabby-images/e9f95/e9f9541d2aa5e23c2c792d0c6ff66e7b986164b2" alt="Overview"] ```scala type F[A] = A×X def map2[A, B, C](f: (A, B) => C)(`fa: F[A], fb: F[B]`): F[C] = ??? ``` --- ## Product .center[data:image/s3,"s3://crabby-images/25488/254887901c1b2c63715b7e49de26da150ce9113f" alt="Overview"] ```scala type F[A] = A×X def map2[A, B, C](f: (A, B) => C)(fa: F[A], fb: F[B]): `F[C]` = ??? ``` --- ## Product .center[data:image/s3,"s3://crabby-images/25488/254887901c1b2c63715b7e49de26da150ce9113f" alt="Overview"] .diff-rm[ ```scala type F[A] = A×X def map2[A, B, C](f: (A, B) => C)(fa: F[A], fb: F[B]): F[C] = * `???` ``` ] --- ## Product .center[data:image/s3,"s3://crabby-images/25488/254887901c1b2c63715b7e49de26da150ce9113f" alt="Overview"] .diff-add[ ```scala type F[A] = A×X def map2[A, B, C](f: (A, B) => C)(fa: F[A], fb: F[B]): F[C] = * `(??? : C)×(??? : X)` ``` ] --- ## Product .center[data:image/s3,"s3://crabby-images/e9f95/e9f9541d2aa5e23c2c792d0c6ff66e7b986164b2" alt="Overview"] ```scala type F[A] = A×X def map2[A, B, C](f: (A, B) => C)(`fa: F[A], fb: F[B]`): F[C] = (??? : C)×(??? : X) ``` --- ## Product .center[data:image/s3,"s3://crabby-images/60da7/60da742c15d36bdc03df6612dd1dedb4d4bcd555" alt="Overview"] ```scala type F[A] = A×X def map2[A, B, C](f: (A, B) => C)(`fa: F[A]`, `fb: F[B]`): F[C] = (??? : C)×(??? : X) ``` --- ## Product .center[data:image/s3,"s3://crabby-images/a3182/a318291ed08859a3e754c14b730b136fdc3c594b" alt="Overview"] .diff-add[ ```scala type F[A] = A×X def map2[A, B, C](f: (A, B) => C)(fa: F[A], fb: F[B]): F[C] = * `val (a×x1) = fa` * `val (b×x2) = fb` * (??? : C)×(??? : X) ``` ] --- ## Product .center[data:image/s3,"s3://crabby-images/c2e19/c2e194e8ab4fc00c39aad71f2893b2432b973695" alt="Overview"] ```scala type F[A] = A×X def map2[A, B, C](f: (A, B) => C)(fa: F[A], fb: F[B]): F[C] = val (a×`x1`) = fa val (b×`x2`) = fb (??? : C)×(??? : X) ``` --- ## Product .center[data:image/s3,"s3://crabby-images/d779e/d779e8333bdcbc19afefc3c5d2e67c113d345f43" alt="Overview"] ```scala type F[A] = A×X def map2[A, B, C](f: (A, B) => C)(fa: F[A], fb: F[B]): F[C] = val (`a`×x1) = fa val (`b`×x2) = fb (??? : C)×(??? : X) ``` --- ## Product .center[data:image/s3,"s3://crabby-images/7b9c5/7b9c51098d363358989d65d77fa93c7294035ac8" alt="Overview"] ```scala type F[A] = A×X def map2[A, B, C](`f: (A, B) => C`)(fa: F[A], fb: F[B]): F[C] = val (`a`×x1) = fa val (`b`×x2) = fb (??? : C)×(??? : X) ``` --- ## Product .center[data:image/s3,"s3://crabby-images/7b9c5/7b9c51098d363358989d65d77fa93c7294035ac8" alt="Overview"] .diff-add[ ```scala type F[A] = A×X def map2[A, B, C](f: (A, B) => C)(fa: F[A], fb: F[B]): F[C] = val (a×x1) = fa val (b×x2) = fb * `val c = f(a, b)` (??? : C)×(??? : X) ``` ] --- ## Product .center[data:image/s3,"s3://crabby-images/57625/57625a5c854db848469462dddabea31b7de684b4" alt="Overview"] ```scala type F[A] = A×X def map2[A, B, C](f: (A, B) => C)(fa: F[A], fb: F[B]): `F[C]` = val (a×x1) = fa val (b×x2) = fb val c = f(a, b) (`??? : C`)×(`??? : X`) ``` --- ## Product .center[data:image/s3,"s3://crabby-images/332c5/332c5efc43ef1876704e61e84ad4508dd9a227f2" alt="Overview"] ```scala type F[A] = A×X def map2[A, B, C](f: (A, B) => C)(fa: F[A], fb: F[B]): F[C] = val (a×`x1`) = fa val (b×`x2`) = fb val c = f(a, b) (??? : C)×(`??? : X`) ``` --- ## Product .center[data:image/s3,"s3://crabby-images/4a14c/4a14c6ce191cfa77caa84b8efbdbbd389b5b364e" alt="Overview"] .diff-add[ ```scala type F[A] = A×X def map2[A, B, C](f: (A, B) => C)(fa: F[A], fb: F[B]): F[C] = val (a×x1) = fa val (b×x2) = fb val c = f(a, b) (??? : C)×(??? : X) * *`def combine(lhs: X, rhs: X): X =` * `???` ``` ] --- ## Product .center[data:image/s3,"s3://crabby-images/4a14c/4a14c6ce191cfa77caa84b8efbdbbd389b5b364e" alt="Overview"] .diff-add[ ```scala type F[A] = A×X def map2[A, B, C](f: (A, B) => C)(fa: F[A], fb: F[B]): F[C] = val (a×x1) = fa val (b×x2) = fb val c = f(a, b) * `val x3 = combine(x1, x2)` (??? : C)×(??? : X) def combine(lhs: X, rhs: X): X = ??? ``` ] --- ## Product .center[data:image/s3,"s3://crabby-images/81fec/81fecc3ee9afe7762a47b66c10036466bc43c988" alt="Overview"] ```scala type F[A] = A×X def map2[A, B, C](f: (A, B) => C)(fa: F[A], fb: F[B]): F[C] = val (a×x1) = fa val (b×x2) = fb val `c` = f(a, b) val `x3` = combine(x1, x2) (??? : C)×(??? : X) def combine(lhs: X, rhs: X): X = ??? ``` --- ## Product .center[data:image/s3,"s3://crabby-images/f5236/f52360e47c87fcaa4da299b6dc31401dfc5c0b2a" alt="Overview"] .diff-rm[ ```scala type F[A] = A×X def map2[A, B, C](f: (A, B) => C)(fa: F[A], fb: F[B]): F[C] = val (a×x1) = fa val (b×x2) = fb val c = f(a, b) val x3 = combine(x1, x2) * `(??? : C)`×`(??? : X)` def combine(lhs: X, rhs: X): X = ??? ``` ] --- ## Product .center[data:image/s3,"s3://crabby-images/f5236/f52360e47c87fcaa4da299b6dc31401dfc5c0b2a" alt="Overview"] .diff-add[ ```scala type F[A] = A×X def map2[A, B, C](f: (A, B) => C)(fa: F[A], fb: F[B]): F[C] = val (a×x1) = fa val (b×x2) = fb val c = f(a, b) val x3 = combine(x1, x2) * `c`×`x3` def combine(lhs: X, rhs: X): X = ??? ``` ] --- ## Product .center[data:image/s3,"s3://crabby-images/4069e/4069e9e81f3bfab6a3f0440465b73b73106b0210" alt="Overview"] ```scala type F[A] = A×X def map2[A, B, C](f: (A, B) => C)(fa: F[A], fb: F[B]): F[C] = val (a×x1) = fa val (b×x2) = fb val c = f(a, b) val x3 = combine(x1, x2) c×x3 def combine(lhs: X, rhs: X): X = ??? ``` --- ## Breaking Product ```scala enum Label: case Front case Back case Bug case Feature ``` --- ## Breaking Product .center[data:image/s3,"s3://crabby-images/f220f/f220f457a1a57d61e1fa0eb73f26256c77f6ec7b" alt="Overview"] .diff-rm[ ```scala *type F[A] = A×`X` def map2[A, B, C](f: (A, B) => C)(fa: F[A], fb: F[B]): F[C] = val (a×x1) = fa val (b×x2) = fb val c = f(a, b) val x3 = combine(x1, x2) c×x3 *def combine(lhs: `X`, rhs: `X`): `X` = ??? ``` ] --- ## Breaking Product .center[data:image/s3,"s3://crabby-images/ce09f/ce09ffb95f8e35b0cd4f0b3916d64829fe00a2ab" alt="Overview"] .diff-add[ ```scala *type F[A] = A×`Label` def map2[A, B, C](f: (A, B) => C)(fa: F[A], fb: F[B]): F[C] = val (a×x1) = fa val (b×x2) = fb val c = f(a, b) val x3 = combine(x1, x2) c×x3 *def combine(lhs: `Label`, rhs: `Label`): X = ??? ``` ] --- ## Breaking Product .center[data:image/s3,"s3://crabby-images/57369/57369a4e35964d3e64937677a9493b912e4b6eb5" alt="Overview"] .diff-rm[ ```scala type F[A] = A×Label def map2[A, B, C](f: (A, B) => C)(fa: F[A], fb: F[B]): F[C] = val (a×x1) = fa val (b×x2) = fb val c = f(a, b) val x3 = combine(x1, x2) c×x3 * *`def combine(lhs: Label, rhs: Label): Label =` * `???` ``` ] --- ## Breaking Product .center[data:image/s3,"s3://crabby-images/57369/57369a4e35964d3e64937677a9493b912e4b6eb5" alt="Overview"] .diff-rm[ ```scala type F[A] = A×Label def map2[A, B, C](f: (A, B) => C)(fa: F[A], fb: F[B]): F[C] = val (a×x1) = fa val (b×x2) = fb val c = f(a, b) * `val x3 = combine(x1, x2)` c×x3 * *`def combine(lhs: Label, rhs: Label): Label =` * `???` ``` ] --- ## Breaking Product .center[data:image/s3,"s3://crabby-images/939bd/939bd86e4af5b0140ea1f972f81ee9c43e64fed2" alt="Overview"] .diff-rm[ ```scala type F[A] = A×Label def map2[A, B, C](f: (A, B) => C)(fa: F[A], fb: F[B]): F[C] = val (a×x1) = fa val (b×x2) = fb val c = f(a, b) * `val x3 = combine(x1, x2)` * `c×x3` * *`def combine(lhs: Label, rhs: Label): Label =` * `???` ``` ] --- ## Breaking Product .center[data:image/s3,"s3://crabby-images/939bd/939bd86e4af5b0140ea1f972f81ee9c43e64fed2" alt="Overview"] .diff-rm[ ```scala type F[A] = A×Label * *`def map2[A, B, C](f: (A, B) => C)(fa: F[A], fb: F[B]): F[C] =` * `val (a×x1) = fa` * `val (b×x2) = fb` * `val c = f(a, b)` * `val x3 = combine(x1, x2)` * * `c×x3` * *`def combine(lhs: Label, rhs: Label): Label =` * `???` ``` ] --- ## Breaking Product .center[data:image/s3,"s3://crabby-images/55575/55575c423f24a0bb8cc488deb6d1e20177f50a59" alt="Overview"] ```scala type F[A] = A×Label ``` --- ## Breaking Product .center[data:image/s3,"s3://crabby-images/4afec/4afec45c73b13214b06d6256da6e33514883b077" alt="Overview"] .diff-rm[ ```scala type F[A] = A×Label ``` ] --- ## Breaking Product .center[data:image/s3,"s3://crabby-images/c4214/c42143636a4d8316dda88bc30fcca6f754f9135c" alt="Overview"] .diff-rm[ ```scala *type `F[A]` = A×Label ``` ] --- ## Labelled data .center[data:image/s3,"s3://crabby-images/433d1/433d1a8038885fcf73b570bc6305c97a960a995c" alt="Overview"] .diff-add[ ```scala *type `Labelled[A]` = A×Label ``` ] --- ## Labelled data .center[data:image/s3,"s3://crabby-images/020c5/020c57ced0bd444bd45a5b8c07eb362c4f67bb70" alt="Overview"] ```scala type Labelled[A] = A×Label ``` --- ## Key takeaways A binary parametric product type is: * a `Functor`. -- * not an `Apply` if the other member is not a `Semigroup`. --- class: center, middle # Apply but not Applicative --- ## Applicative .center[data:image/s3,"s3://crabby-images/281cc/281ccc614020e2f70f9c7a7f95adb23abbfd4919" alt="Overview"] --- ## Applicative .center[data:image/s3,"s3://crabby-images/f72db/f72dbba12b507f8529206ec2a5753c27ee1ab933" alt="Overview"] --- ## Applicative .center[data:image/s3,"s3://crabby-images/5cc06/5cc06f3ac18095ecfd9421aab8ff8feb5d385615" alt="Applicative"] ```scala def pure[A](`a: A`): F[A] ``` --- ## Applicative .center[data:image/s3,"s3://crabby-images/e0c03/e0c0314a5d7741fec65f2b59f0eed69ee686d10a" alt="Applicative"] ```scala def pure[A](a: A): `F[A]` ``` --- ## Applicative .center[data:image/s3,"s3://crabby-images/a0954/a0954eb122148d3395b7b3c7509e5301e1f3deab" alt="Applicative"] ```scala def `pure`[A](a: A): F[A] ``` --- ## Product .center[data:image/s3,"s3://crabby-images/51811/51811a8b1997f528a0da6d3117e918bbad807505" alt="Overview"] ```scala type F[A] = `A×X` def pure[A](a: A): F[A] = ??? ``` --- ## Product .center[data:image/s3,"s3://crabby-images/5fafc/5fafc186a84028833d03eb1feb439bd21b43409b" alt="Overview"] ```scala type F[A] = A×X def pure[A](`a: A`): F[A] = ??? ``` --- ## Product .center[data:image/s3,"s3://crabby-images/9c4bf/9c4bfa33f20a9d304f27550caf711bb811f3ea1a" alt="Overview"] ```scala type F[A] = A×X def pure[A](a: A): `F[A]` = ??? ``` --- ## Product .center[data:image/s3,"s3://crabby-images/9c4bf/9c4bfa33f20a9d304f27550caf711bb811f3ea1a" alt="Overview"] .diff-rm[ ```scala type F[A] = A×X def pure[A](a: A): F[A] = * `???` ``` ] --- ## Product .center[data:image/s3,"s3://crabby-images/9c4bf/9c4bfa33f20a9d304f27550caf711bb811f3ea1a" alt="Overview"] .diff-add[ ```scala type F[A] = A×X def pure[A](a: A): F[A] = * `(??? : A)×(??? : X)` ``` ] --- ## Product .center[data:image/s3,"s3://crabby-images/5fafc/5fafc186a84028833d03eb1feb439bd21b43409b" alt="Overview"] ```scala type F[A] = A×X def pure[A](`a: A`): F[A] = (??? : A)×(??? : X) ``` --- ## Product .center[data:image/s3,"s3://crabby-images/caf82/caf822d373273cee72f9b39a119c45a1806f334d" alt="Overview"] ```scala type F[A] = A×X def pure[A](a: A): F[A] = (??? : A)×`(??? : X)` ``` --- ## Product .center[data:image/s3,"s3://crabby-images/caf82/caf822d373273cee72f9b39a119c45a1806f334d" alt="Overview"] .diff-add[ ```scala type F[A] = A×X def pure[A](a: A): F[A] = (??? : A)×(??? : X) * *`def empty: X =` * `???` ``` ] --- ## Product .center[data:image/s3,"s3://crabby-images/4377d/4377d738c6dc835a79614cb9fe67876534f48b04" alt="Overview"] .diff-rm[ ```scala type F[A] = A×X def pure[A](a: A): F[A] = * `(??? : A)`×`(??? : X)` def empty: X = ??? ``` ] --- ## Product .center[data:image/s3,"s3://crabby-images/4377d/4377d738c6dc835a79614cb9fe67876534f48b04" alt="Overview"] .diff-add[ ```scala type F[A] = A×X def pure[A](a: A): F[A] = * `a`×`empty` def empty: X = ??? ``` ] --- ## Product .center[data:image/s3,"s3://crabby-images/4377d/4377d738c6dc835a79614cb9fe67876534f48b04" alt="Overview"] ```scala type F[A] = A×X def pure[A](a: A): F[A] = a×empty def empty: X = ??? ``` --- ## Breaking Product .center[data:image/s3,"s3://crabby-images/d72f2/d72f2e957b79459b598f86dfd3d90529a32bd86b" alt="Overview"] .diff-rm[ ```scala *type F[A] = A×`X` def pure[A](a: A): F[A] = a×empty *def empty: `X` = ??? ``` ] --- ## Breaking Product .center[data:image/s3,"s3://crabby-images/488c3/488c310d7969647f2a83b4f0fa36d6dde18b1764" alt="Overview"] .diff-add[ ```scala *type F[A] = A×`PosInt` def pure[A](a: A): F[A] = a×empty *def empty: `PosInt` = ??? ``` ] --- ## Breaking Product .center[data:image/s3,"s3://crabby-images/43e06/43e06e759e0a9412debc28752004d288279d8d59" alt="Overview"] .diff-rm[ ```scala type F[A] = A×PosInt def pure[A](a: A): F[A] = a×empty * *`def empty: PosInt =` * `???` ``` ] --- ## Breaking Product .center[data:image/s3,"s3://crabby-images/43e06/43e06e759e0a9412debc28752004d288279d8d59" alt="Overview"] .diff-rm[ ```scala type F[A] = A×PosInt def pure[A](a: A): F[A] = * a×`empty` * *`def empty: PosInt =` * `???` ``` ] --- ## Breaking Product .center[data:image/s3,"s3://crabby-images/3632c/3632ced68d89b8333c9e86ed3e15bc6e09756880" alt="Overview"] .diff-rm[ ```scala type F[A] = A×PosInt * *`def pure[A](a: A): F[A] =` * a×`empty` * *`def empty: PosInt =` * `???` ``` ] --- ## Breaking Product .center[data:image/s3,"s3://crabby-images/4048e/4048e88d953636cf654dd1495d246f543c385baa" alt="Overview"] ```scala type F[A] = A×PosInt ``` --- ## Breaking Product .center[data:image/s3,"s3://crabby-images/c914f/c914f710b5e01f2a2d1f9ff71946c8d2413c52c4" alt="Overview"] .diff-rm[ ```scala *type `F[A]` = A×PosInt ``` ] --- ## Weighted data .center[data:image/s3,"s3://crabby-images/34935/34935ce74ba78e57e98d525b9dacdb7878b0019c" alt="Overview"] .diff-add[ ```scala *type `Weighted[A]` = A×PosInt ``` ] --- ## Weighted data .center[data:image/s3,"s3://crabby-images/7a37e/7a37e03b698339314e86fb616826b172affc04c4" alt="Overview"] ```scala type Weighted[A] = A×PosInt ``` --- ## Key takeaways A binary parametric product type is: * a `Functor`. * not an `Apply` if the other member is not a `Semigroup`. -- * not an `Applicative` if the other member is not a `Monoid`. --- class: center, middle # Apply but not FlatMap --- ## FlatMap .center[data:image/s3,"s3://crabby-images/ca9b4/ca9b419519562d20000ca08b18d500ee7b31417c" alt="Overview"] --- ## FlatMap .center[data:image/s3,"s3://crabby-images/1902b/1902bcd2a188475d0216b4365b074ab5991551ad" alt="Overview"] --- ## FlatMap .center[data:image/s3,"s3://crabby-images/75a24/75a24ad9323af9abf546f128fd697a303e67d73b" alt="Overview"] ```scala def flatMap[B](f: A => F[B])(`fa: F[A]`): F[B] ``` --- ## FlatMap .center[data:image/s3,"s3://crabby-images/8d725/8d725ec04bedaeb563156116e765e057fa68399f" alt="Overview"] ```scala def flatMap[B](f: A => F[B])(fa: F[A]): `F[B]` ``` --- ## FlatMap .center[data:image/s3,"s3://crabby-images/9fda8/9fda8781987ecd3bda4cd1254f04b678c2b77e0c" alt="Overview"] ```scala def flatMap[B](`f: A => F[B]`)(fa: F[A]): F[B] ``` --- ## FlatMap .center[data:image/s3,"s3://crabby-images/00e1b/00e1bc1574675e2926986117cee7ba6ab903b2d7" alt="Overview"] ```scala def `flatMap`[B](f: A => F[B])(fa: F[A]): F[B] ``` --- ## FlatMap .center[data:image/s3,"s3://crabby-images/b57dc/b57dc4726a37941e7c48ff2bd35bdfdce4a7ac4f" alt="Overview"] ```scala def flatMap[B](f: A => F[B])(fa: F[A]): F[B] ``` --- ## Product .center[data:image/s3,"s3://crabby-images/b70e2/b70e29f68a3e7aed93e3a391f8d928f83a4e1522" alt="Overview"] ```scala type F[A] = `A×X` def flatMap[A, B](f: A => F[B])(fa: F[A]): F[B] = ??? ``` --- ## Product .center[data:image/s3,"s3://crabby-images/d1c33/d1c334224f4869d0a987288dd9c39cda8cf0ac49" alt="Overview"] ```scala type F[A] = A×X def flatMap[A, B](f: A => F[B])(`fa: F[A]`): F[B] = ??? ``` --- ## Product .center[data:image/s3,"s3://crabby-images/b47e2/b47e2689ff9a88ef8363d0b087550b182c773be8" alt="Overview"] ```scala type F[A] = A×X def flatMap[A, B](`f: A => F[B]`)(fa: F[A]): F[B] = ??? ``` --- ## Product .center[data:image/s3,"s3://crabby-images/83f20/83f200df3c5bdcde282db7f156e33c089e3c97c0" alt="Overview"] ```scala type F[A] = A×X def flatMap[A, B](f: A => F[B])(fa: F[A]): `F[B]` = ??? ``` --- ## Product .center[data:image/s3,"s3://crabby-images/83f20/83f200df3c5bdcde282db7f156e33c089e3c97c0" alt="Overview"] .diff-rm[ ```scala type F[A] = A×X def flatMap[A, B](f: A => F[B])(fa: F[A]): F[B] = * `???` ``` ] --- ## Product .center[data:image/s3,"s3://crabby-images/83f20/83f200df3c5bdcde282db7f156e33c089e3c97c0" alt="Overview"] .diff-add[ ```scala type F[A] = A×X def flatMap[A, B](f: A => F[B])(fa: F[A]): F[B] = * `(??? : B)×(??? : X)` ``` ] --- ## Product .center[data:image/s3,"s3://crabby-images/d1c33/d1c334224f4869d0a987288dd9c39cda8cf0ac49" alt="Overview"] ```scala type F[A] = A×X def flatMap[A, B](f: A => F[B])(`fa: F[A]`): F[B] = (??? : B)×(??? : X) ``` --- ## Product .center[data:image/s3,"s3://crabby-images/1ce35/1ce35bbb120599e4670bfdd81d9f599c32f31ab2" alt="Overview"] .diff-add[ ```scala type F[A] = A×X def flatMap[A, B](f: A => F[B])(fa: F[A]): F[B] = * `val (a×x1) = fa` * (??? : B)×(??? : X) ``` ] --- ## Product .center[data:image/s3,"s3://crabby-images/416df/416df31a6d25c1c9ae8b8ad8c777245717ea1f10" alt="Overview"] ```scala type F[A] = A×X def flatMap[A, B](`f: A => F[B]`)(fa: F[A]): F[B] = val (`a`×x1) = fa (??? : B)×(??? : X) ``` --- ## Product .center[data:image/s3,"s3://crabby-images/416df/416df31a6d25c1c9ae8b8ad8c777245717ea1f10" alt="Overview"] .diff-add[ ```scala type F[A] = A×X def flatMap[A, B](f: A => F[B])(fa: F[A]): F[B] = val (a×x1) = fa * `val fb = f(a)` (??? : B)×(??? : X) ``` ] --- ## Product .center[data:image/s3,"s3://crabby-images/8574d/8574d2aa780e819239ceb98dba821daf2cc89be0" alt="Overview"] ```scala type F[A] = A×X def flatMap[A, B](f: A => F[B])(fa: F[A]): F[B] = val (a×`x1`) = fa val fb = f(a) (??? : B)×(??? : X) ``` --- ## Product .center[data:image/s3,"s3://crabby-images/c8f68/c8f683c7a48256e83e8c0604dedea66a971c0c6b" alt="Overview"] ```scala type F[A] = A×X def flatMap[A, B](f: A => F[B])(fa: F[A]): F[B] = val (a×x1) = fa val `fb` = f(a) (??? : B)×(??? : X) ``` --- ## Product .center[data:image/s3,"s3://crabby-images/43c46/43c46054986fe4f49a65894e93e6f89fc6af0f17" alt="Overview"] ```scala type F[A] = A×X def flatMap[A, B](f: A => F[B])(fa: F[A]): F[B] = val (a×x1) = fa val fb = f(a) (??? : B)×(??? : X) ``` --- ## Product .center[data:image/s3,"s3://crabby-images/ad8c8/ad8c81df57335c07887a6c6607688868482f79dc" alt="Overview"] .diff-rm[ ```scala type F[A] = A×X def flatMap[A, B](f: A => F[B])(fa: F[A]): F[B] = val (a×x1) = fa * val `fb` = f(a) (??? : B)×(??? : X) ``` ] --- ## Product .center[data:image/s3,"s3://crabby-images/30e55/30e55eb29075714453fa7416c952218b0904940a" alt="Overview"] .diff-add[ ```scala type F[A] = A×X def flatMap[A, B](f: A => F[B])(fa: F[A]): F[B] = val (a×x1) = fa * val `(b×x2)` = f(a) (??? : B)×(??? : X) ``` ] --- ## Product .center[data:image/s3,"s3://crabby-images/013cc/013cccee67bcc3eb6f5ac940567971667733fd72" alt="Overview"] .diff-add[ ```scala type F[A] = A×X def flatMap[A, B](f: A => F[B])(fa: F[A]): F[B] = val (a×x1) = fa val (b×x2) = f(a) (??? : B)×(??? : X) * *`def combine(lhs: X, rhs: X): X =` * `???` ``` ] --- ## Product .center[data:image/s3,"s3://crabby-images/013cc/013cccee67bcc3eb6f5ac940567971667733fd72" alt="Overview"] .diff-add[ ```scala type F[A] = A×X def flatMap[A, B](f: A => F[B])(fa: F[A]): F[B] = val (a×x1) = fa val (b×x2) = f(a) * `val x3 = combine(x1, x2)` (??? : B)×(??? : X) def combine(lhs: X, rhs: X): X = ??? ``` ] --- ## Product .center[data:image/s3,"s3://crabby-images/24a7f/24a7f24f50f90b34fd45e2fe48d2616820e6111a" alt="Overview"] ```scala type F[A] = A×X def flatMap[A, B](f: A => F[B])(fa: F[A]): F[B] = val (a×x1) = fa val (`b`×x2) = f(a) val `x3` = combine(x1, x2) (??? : B)×(??? : X) def combine(lhs: X, rhs: X): X = ??? ``` --- ## Product .center[data:image/s3,"s3://crabby-images/156b0/156b050f62e8834962064ba9f654d06674a0f74d" alt="Overview"] .diff-rm[ ```scala type F[A] = A×X def flatMap[A, B](f: A => F[B])(fa: F[A]): F[B] = val (a×x1) = fa val (b×x2) = f(a) val x3 = combine(x1, x2) * `(??? : B)`×`(??? : X)` def combine(lhs: X, rhs: X): X = ??? ``` ] --- ## Product .center[data:image/s3,"s3://crabby-images/156b0/156b050f62e8834962064ba9f654d06674a0f74d" alt="Overview"] .diff-add[ ```scala type F[A] = A×X def flatMap[A, B](f: A => F[B])(fa: F[A]): F[B] = val (a×x1) = fa val (b×x2) = f(a) val x3 = combine(x1, x2) * `b`×`x3` def combine(lhs: X, rhs: X): X = ??? ``` ] --- ## Product .center[data:image/s3,"s3://crabby-images/12607/12607f94b74537115423a0c193e029a0172a2d5b" alt="Overview"] ```scala type F[A] = A×X def flatMap[A, B](f: A => F[B])(fa: F[A]): F[B] = val (a×x1) = fa val (b×x2) = f(a) val x3 = combine(x1, x2) b×x3 def combine(lhs: X, rhs: X): X = ??? ``` --- ## Breaking Product .center[data:image/s3,"s3://crabby-images/4384f/4384f22986b8da3f486574ca80ee7f804ace77bb" alt="Overview"] ```scala type F[A] = A×X def flatMap[A, B](f: A => F[B])(fa: F[A]): F[B] = val (a×x1) = fa val (b×x2) = f(a) val x3 = combine(x1, x2) `b`×x3 def combine(lhs: X, rhs: X): X = ??? ``` --- ## Breaking Product .center[data:image/s3,"s3://crabby-images/51573/515737e8f2ee7571e8e1f81f484bc22c14f69e43" alt="Overview"] ```scala type F[A] = A×X def flatMap[A, B](f: A => F[B])(fa: F[A]): F[B] = val (a×x1) = fa val `(b×x2)` = f(a) val x3 = combine(x1, x2) b×x3 def combine(lhs: X, rhs: X): X = ??? ``` --- ## Breaking Product .center[data:image/s3,"s3://crabby-images/c5681/c5681c9cde153c7b5fcf53e3fb50743508cf1fa7" alt="Overview"] ```scala type F[A] = A×X def flatMap[A, B](f: A => F[B])(fa: F[A]): F[B] = val (a×x1) = fa val (b×x2) = `f(a)` val x3 = combine(x1, x2) b×x3 def combine(lhs: X, rhs: X): X = ??? ``` --- ## Breaking Product .center[data:image/s3,"s3://crabby-images/7b413/7b41392c5ce5c12ae829eb4af9145056cfdd49b8" alt="Overview"] ```scala type F[A] = A×X def flatMap[A, B](f: A => F[B])(fa: F[A]): F[B] = val `(a×x1)` = fa val (b×x2) = f(a) val x3 = combine(x1, x2) b×x3 def combine(lhs: X, rhs: X): X = ??? ``` --- ## Breaking Product .center[data:image/s3,"s3://crabby-images/7b413/7b41392c5ce5c12ae829eb4af9145056cfdd49b8" alt="Overview"] ```scala type F[A] = `A`×X def flatMap[A, B](f: A => F[B])(fa: F[A]): F[B] = val (a×x1) = fa val (b×x2) = f(a) val x3 = combine(x1, x2) b×x3 def combine(lhs: X, rhs: X): X = ??? ``` --- ## Breaking Product .center[data:image/s3,"s3://crabby-images/7b413/7b41392c5ce5c12ae829eb4af9145056cfdd49b8" alt="Overview"] .diff-rm[ ```scala *type F[A] = `A×`X def flatMap[A, B](f: A => F[B])(fa: F[A]): F[B] = val (a×x1) = fa val (b×x2) = f(a) val x3 = combine(x1, x2) b×x3 def combine(lhs: X, rhs: X): X = ??? ``` ] --- ## Breaking Product .center[data:image/s3,"s3://crabby-images/7b413/7b41392c5ce5c12ae829eb4af9145056cfdd49b8" alt="Overview"] .diff-rm[ ```scala *type F[A] = `A×`X def flatMap[A, B](f: A => F[B])(fa: F[A]): F[B] = * val (`a`×x1) = fa val (b×x2) = f(a) val x3 = combine(x1, x2) b×x3 def combine(lhs: X, rhs: X): X = ??? ``` ] --- ## Breaking Product .center[data:image/s3,"s3://crabby-images/7b413/7b41392c5ce5c12ae829eb4af9145056cfdd49b8" alt="Overview"] .diff-rm[ ```scala *type F[A] = `A×`X def flatMap[A, B](f: A => F[B])(fa: F[A]): F[B] = * val (`a`×x1) = fa * val (b×x2) = `f(a)` val x3 = combine(x1, x2) b×x3 def combine(lhs: X, rhs: X): X = ??? ``` ] --- ## Breaking Product .center[data:image/s3,"s3://crabby-images/5deea/5deea9604b9d7969bb0c68a48f2ad46738dfe158" alt="Overview"] .diff-rm[ ```scala *type F[A] = `A×`X def flatMap[A, B](f: A => F[B])(fa: F[A]): F[B] = * val (`a`×x1) = fa * `val (b×x2) = f(a)` val x3 = combine(x1, x2) b×x3 def combine(lhs: X, rhs: X): X = ??? ``` ] --- ## Breaking Product .center[data:image/s3,"s3://crabby-images/d653a/d653a6efbab27ee11e076610d5d47ffd90a37bf4" alt="Overview"] .diff-rm[ ```scala *type F[A] = `A×`X * *`def flatMap[A, B](f: A => F[B])(fa: F[A]): F[B] =` * `val (a×x1) = fa` * `val (b×x2) = f(a)` * `val x3 = combine(x1, x2)` * * `b×x3` def combine(lhs: X, rhs: X): X = ??? ``` ] --- ## Breaking Product .center[data:image/s3,"s3://crabby-images/897b2/897b294a05f455c407f3e19fbfccacd071421db6" alt="Overview"] ```scala type F[A] = X def combine(lhs: X, rhs: X): X = ??? ``` --- ## Breaking Product .center[data:image/s3,"s3://crabby-images/a2976/a29762e4409fde1ac3a5adb4b157de9e16448f08" alt="Overview"] .diff-rm[ ```scala *type F[A] = `X` *def combine(lhs: `X`, rhs: `X`): `X` = ??? ``` ] --- ## Breaking Product .center[data:image/s3,"s3://crabby-images/89b8c/89b8cbe94b30ecc7b4799fb9cc4ddede3497e2d8" alt="Overview"] .diff-add[ ```scala *type F[A] = `PosInt` *def combine(lhs: `PosInt`, rhs: `PosInt`): `PosInt` = ??? ``` ] --- ## Breaking Product .center[data:image/s3,"s3://crabby-images/bd049/bd0493724d2dd1e7615bd0fbfb2cf59393ab2129" alt="Overview"] ```scala type F[A] = PosInt def combine(lhs: PosInt, rhs: PosInt): PosInt = ??? ``` --- ## Breaking Product .center[data:image/s3,"s3://crabby-images/954cc/954cc28949502aff96b2ee93b9339aa4105ccad4" alt="Overview"] .diff-rm[ ```scala *type `F[A]` = PosInt def combine(lhs: PosInt, rhs: PosInt): PosInt = ??? ``` ] --- ## Weight .center[data:image/s3,"s3://crabby-images/e33b1/e33b1fa7cfb78775c75082e30b637abb0ee56cb8" alt="Overview"] .diff-add[ ```scala *type `Weight[A]` = PosInt def combine(lhs: PosInt, rhs: PosInt): PosInt = ??? ``` ] --- ## Weight .center[data:image/s3,"s3://crabby-images/d3e57/d3e5710fcfcb2b62b04ae1c19920b81fd55ff8ca" alt="Overview"] ```scala type Weight[A] = PosInt def combine(lhs: PosInt, rhs: PosInt): PosInt = ??? ``` --- ## Key takeaways A binary parametric product type is: * a `Functor`. * not an `Apply` if the other member is not a `Semigroup`. * not an `Applicative` if the other member is not a `Monoid`. -- * not a `FlatMap` if an `Apply` with a phantom type. --- class: center, middle # Applicative but not Monad --- ## Monad .center[data:image/s3,"s3://crabby-images/43f3c/43f3c827e22f1788678f4a2dbd3e63fea488d0d3" alt="Overview"] --- ## Monad .center[data:image/s3,"s3://crabby-images/97ae1/97ae1b331140dd5a7f1f41194c1679061d65a2cb" alt="Overview"] --- ## Flagged data .center[data:image/s3,"s3://crabby-images/437a4/437a4478d45bfb9d2297c58ef33e51380b873c2e" alt="Overview"] ```scala type `Flagged[A]` = A×Boolean ``` --- ## Flagged data .center[data:image/s3,"s3://crabby-images/f7596/f75967166b058dbfc589e62df8d484c60526c0ed" alt="Overview"] ```scala type Flagged[A] = `A`×Boolean ``` --- ## Flagged data .center[data:image/s3,"s3://crabby-images/17a57/17a572f1c5e341724915eea01aca003d8a63eb35" alt="Overview"] ```scala type Flagged[A] = A×`Boolean` ``` --- ## Flagged data .center[data:image/s3,"s3://crabby-images/e02e0/e02e0e86dac374c20fd65bec354ff534c3f8f116" alt="Overview"] .diff-add[ ```scala type Flagged[A] = A×Boolean *`def pure[A](a: A): Flagged[A] =` * `a×false` ``` ] --- ## Flagged data .center[data:image/s3,"s3://crabby-images/e02e0/e02e0e86dac374c20fd65bec354ff534c3f8f116" alt="Overview"] ```scala type Flagged[A] = A×Boolean def pure[A](a: A): Flagged[A] = a×`false` ``` --- ## Flagged data .center[data:image/s3,"s3://crabby-images/9d628/9d628184010541bbe5d924ec4a967dee3689d3d1" alt="Overview"] .diff-add[ ```scala type Flagged[A] = A×Boolean def pure[A](a: A): Flagged[A] = a×false * *`def flatMap[A, B](f: A => Flagged[B])(fa: Flagged[A]): Flagged[B] =` * `val (a×x1) = fa` * `val (b×x2) = f(a)` * `val x3 = x1 || x2` * * `b×x3` ``` ] --- ## Flagged data .center[data:image/s3,"s3://crabby-images/58b19/58b19e553ad988044422078cffed7e8190b1c983" alt="Overview"] ```scala type Flagged[A] = A×Boolean def pure[A](a: A): Flagged[A] = a×false def flatMap[A, B](f: A => Flagged[B])(fa: Flagged[A]): Flagged[B] = val (a×x1) = fa val (b×x2) = f(a) val x3 = x1 `||` x2 b×x3 ``` --- ## Flagged data .center[data:image/s3,"s3://crabby-images/9d628/9d628184010541bbe5d924ec4a967dee3689d3d1" alt="Overview"] ```scala type Flagged[A] = A×Boolean def pure[A](a: A): Flagged[A] = a×false def flatMap[A, B](f: A => Flagged[B])(fa: Flagged[A]): Flagged[B] = val (a×x1) = fa val (b×x2) = f(a) val x3 = x1 || x2 b×x3 ``` --- ## Breaking flagged data .center[data:image/s3,"s3://crabby-images/84048/840488dc4e59872451ae964b9be7998bca8f8609" alt="Overview"] .diff-rm[ ```scala *type Flagged[A] = `A×`Boolean def pure[A](a: A): Flagged[A] = * `a×`false def flatMap[A, B](f: A => Flagged[B])(fa: Flagged[A]): Flagged[B] = * val (`a×`x1) = fa * val (`b×`x2) = f(a) val x3 = x1 || x2 * `b×`x3 ``` ] --- ## Breaking flagged data .center[data:image/s3,"s3://crabby-images/3a7d7/3a7d72cd88d95741a0f883e969167de83144188d" alt="Overview"] .diff-rm[ ```scala *type Flagged[A] = `A×`Boolean def pure[A](a: A): Flagged[A] = * `a×`false * *`def flatMap[A, B](f: A => Flagged[B])(fa: Flagged[A]): Flagged[B] =` * `val (a×x1) = fa` * `val (b×x2) = f(a)` * `val x3 = x1 || x2` * * `b×`x3 ``` ] --- ## Breaking flagged data .center[data:image/s3,"s3://crabby-images/d88c4/d88c499fd14a73e35ab73c757eb8f83817aabd6b" alt="Overview"] ```scala type Flagged[A] = Boolean def pure[A](a: A): Flagged[A] = false ``` --- ## Breaking flagged data .center[data:image/s3,"s3://crabby-images/422b1/422b1113f7236658423c2168e8abfa3f5793bd7e" alt="Overview"] ```scala type Flagged[A] = Boolean def pure[A](a: A): Flagged[A] = false ``` --- ## Breaking flagged data .center[data:image/s3,"s3://crabby-images/86588/86588456912d253a837dbde4643d4f09c3a92569" alt="Overview"] .diff-rm[ ```scala *type `Flagged`[A] = Boolean *def pure[A](a: A): `Flagged`[A] = false ``` ] --- ## Flag .center[data:image/s3,"s3://crabby-images/f2f48/f2f48835050dac3035d1515321c1cd3b60db4b64" alt="Overview"] .diff-add[ ```scala *type `Flag`[A] = Boolean *def pure[A](a: A): `Flag`[A] = false ``` ] --- ## Flag .center[data:image/s3,"s3://crabby-images/832c6/832c60ac661462aebe8e91ebc1d718e3d61cffc5" alt="Overview"] ```scala type Flag[A] = Boolean def pure[A](a: A): Flag[A] = false ``` --- ## Key takeaways A binary parametric product type is: * a `Functor`. * not an `Apply` if the other member is not a `Semigroup`. * not an `Applicative` if the other member is not a `Monoid`. * not a `FlatMap` if an `Apply` with a phantom type. -- * not a `Monad` if an `Applicative` with a phantom type. --- class: center, middle # FlatMap but not Monad --- ## Monad .center[data:image/s3,"s3://crabby-images/462cd/462cd603be3b61025747b75a644f49d9727ba7bc" alt="Overview"] --- ## Monad .center[data:image/s3,"s3://crabby-images/5afd8/5afd86637f8ff0e11843efaa94c40bed78f64ca7" alt="Overview"] --- ## Flagged data .center[data:image/s3,"s3://crabby-images/9d628/9d628184010541bbe5d924ec4a967dee3689d3d1" alt="Overview"] ```scala type Flagged[A] = A×Boolean def pure[A](a: A): Flagged[A] = a×false def flatMap[A, B](f: A => Flagged[B])(fa: Flagged[A]): Flagged[B] = val (a×x1) = fa val (b×x2) = f(a) val x3 = x1 || x2 b×x3 ``` --- ## Breaking flagged data .center[data:image/s3,"s3://crabby-images/1819e/1819e9eb7288043328e3d3fa4ebce0aef551c6d1" alt="Overview"] .diff-rm[ ```scala *type Flagged[A] = A×`Boolean` def pure[A](a: A): Flagged[A] = a×false def flatMap[A, B](f: A => Flagged[B])(fa: Flagged[A]): Flagged[B] = val (a×x1) = fa val (b×x2) = f(a) val x3 = x1 || x2 b×x3 ``` ] --- ## Breaking flagged data .center[data:image/s3,"s3://crabby-images/5a416/5a4166ad89a43925ee886dd980d63f2ed4b51b94" alt="Overview"] .diff-add[ ```scala *type Flagged[A] = A×`PosInt` def pure[A](a: A): Flagged[A] = a×false def flatMap[A, B](f: A => Flagged[B])(fa: Flagged[A]): Flagged[B] = val (a×x1) = fa val (b×x2) = f(a) val x3 = x1 || x2 b×x3 ``` ] --- ## Breaking flagged data .center[data:image/s3,"s3://crabby-images/a8d52/a8d52bf21b0059b7d8cfa9563a7ba9644168cce5" alt="Overview"] .diff-rm[ ```scala type Flagged[A] = A×PosInt def pure[A](a: A): Flagged[A] = a×false def flatMap[A, B](f: A => Flagged[B])(fa: Flagged[A]): Flagged[B] = val (a×x1) = fa val (b×x2) = f(a) * val x3 = `x1 || x2` b×x3 ``` ] --- ## Breaking flagged data .center[data:image/s3,"s3://crabby-images/b8a64/b8a64890ba84e389b29a0965ea49ed71db13404c" alt="Overview"] .diff-add[ ```scala type Flagged[A] = A×PosInt def pure[A](a: A): Flagged[A] = a×false def flatMap[A, B](f: A => Flagged[B])(fa: Flagged[A]): Flagged[B] = val (a×x1) = fa val (b×x2) = f(a) * val x3 = `x1 + x2` b×x3 ``` ] --- ## Breaking flagged data .center[data:image/s3,"s3://crabby-images/d2d99/d2d99edb3d97f139a53ee17ab0f24524e9356f12" alt="Overview"] .diff-add[ ```scala type Flagged[A] = A×PosInt def pure[A](a: A): Flagged[A] = a×false def flatMap[A, B](f: A => Flagged[B])(fa: Flagged[A]): Flagged[B] = val (a×x1) = fa val (b×x2) = f(a) val x3 = x1 + x2 b×x3 ``` ] --- ## Breaking flagged data .center[data:image/s3,"s3://crabby-images/634b7/634b779416034ab44caa90982dc8d94bd020a268" alt="Overview"] .diff-add[ ```scala type Flagged[A] = A×PosInt def pure[A](a: A): Flagged[A] = a×false def flatMap[A, B](f: A => Flagged[B])(fa: Flagged[A]): Flagged[B] = val (a×x1) = fa val (b×x2) = f(a) val x3 = x1 + x2 b×x3 ``` ] --- ## Breaking flagged data .center[data:image/s3,"s3://crabby-images/1a8ae/1a8aee1e619928dea6ddf274b44c2c4245ab4c69" alt="Overview"] .diff-rm[ ```scala type Flagged[A] = A×PosInt def pure[A](a: A): Flagged[A] = * a×`false` def flatMap[A, B](f: A => Flagged[B])(fa: Flagged[A]): Flagged[B] = val (a×x1) = fa val (b×x2) = f(a) val x3 = x1 + x2 b×x3 ``` ] --- ## Breaking flagged data .center[data:image/s3,"s3://crabby-images/634b7/634b779416034ab44caa90982dc8d94bd020a268" alt="Overview"] .diff-rm[ ```scala type Flagged[A] = A×PosInt * *`def pure[A](a: A): Flagged[A] =` * `a×false` def flatMap[A, B](f: A => Flagged[B])(fa: Flagged[A]): Flagged[B] = val (a×x1) = fa val (b×x2) = f(a) val x3 = x1 + x2 b×x3 ``` ] --- ## Breaking flagged data .center[data:image/s3,"s3://crabby-images/1c38f/1c38fd2229b4ba391b60868d5ccd971cf2c81e6a" alt="Overview"] ```scala type Flagged[A] = A×PosInt def flatMap[A, B](f: A => Flagged[B])(fa: Flagged[A]): Flagged[B] = val (a×x1) = fa val (b×x2) = f(a) val x3 = x1 + x2 b×x3 ``` --- ## Breaking flagged data .center[data:image/s3,"s3://crabby-images/856ff/856ff9c8c15123a145e91cbf2c97667546f02f10" alt="Overview"] .diff-rm[ ```scala *type `Flagged`[A] = A×PosInt *def flatMap[A, B](f: A => `Flagged`[B])(fa: `Flagged`[A])`: Flagged[B]` = val (a×x1) = fa val (b×x2) = f(a) val x3 = x1 + x2 b×x3 ``` ] --- ## Weighted data .center[data:image/s3,"s3://crabby-images/affb5/affb5eb706faf005f020e693ca3ee15f5ddab620" alt="Overview"] .diff-add[ ```scala *type `Weighted`[A] = A×PosInt *def flatMap[A, B](f: A => `Weighted`[B])(fa: `Weighted`[A]) = val (a×x1) = fa val (b×x2) = f(a) val x3 = x1 + x2 b×x3 ``` ] --- ## Weighted data .center[data:image/s3,"s3://crabby-images/7a37e/7a37e03b698339314e86fb616826b172affc04c4" alt="Overview"] ```scala type Weighted[A] = A×PosInt def flatMap[A, B](f: A => Weighted[B])(fa: Weighted[A]) = val (a×x1) = fa val (b×x2) = f(a) val x3 = x1 + x2 b×x3 ``` --- ## Key takeaways A binary parametric product type is: * a `Functor`. * not an `Apply` if the other member is not a `Semigroup`. * not an `Applicative` if the other member is not a `Monoid`. * not a `FlatMap` if an `Apply` with a phantom type. * not a `Monad` if an `Applicative` with a phantom type. -- * not a `Monad` if a `FlatMap` and the other member is not a `Monoid`. ??? And with this, I've shown you all the counter examples I wanted to show you. To summarize, we've seen, roughly, that: * contravariant functors were not functors. * apply and applicative have a deep connection with semigroups and monoids - which is not exactly new, this was front and center in McBride's paper on applicative functors, but re-discovering this connection through counter example made it feel more concrete to me. * monads and flatmaps are about having a value - which sounds a little weird on the face of it, but really is just another way of saying that monads are about chaining functions. I hope this was, if not new information, at least an entertaining twist on old information. I certainly was quite entertaining to work on and figure out for myself. And with that, I'd be happy to answer any question you might have. --- 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/things-that-are-things/