Skip to content

Commit

Permalink
Finish course 5. Introduce GADTs before type families.
Browse files Browse the repository at this point in the history
  • Loading branch information
smelc committed Oct 11, 2024
1 parent e5c6e59 commit 18b3156
Show file tree
Hide file tree
Showing 3 changed files with 133 additions and 9 deletions.
9 changes: 5 additions & 4 deletions slides/Course05.hs
Original file line number Diff line number Diff line change
Expand Up @@ -56,11 +56,12 @@ class (Monoid w, Monad m) => MonadWriter w m | m -> w where

writerExample :: MonadWriter String m => m (Maybe Int)
writerExample = do
a <- pure $ Just 1 -- Could be an IO action
tell ("Producing a" :: String)
b <- pure $ Just 2
a <- pure (Just 1) -- Could be an IO action
tell "Producing a"
b <- pure Nothing
tell "Producing b"
return $ (+) <$> a <*> b
let r = (+) <$> a <*> b
return r

class Monad m => MonadState s m | m -> s where
-- | Return the state
Expand Down
90 changes: 86 additions & 4 deletions slides/course-05.md
Original file line number Diff line number Diff line change
Expand Up @@ -233,25 +233,107 @@ writerExample = do
tell "Producing a"
b <- pure Nothing
tell "Producing b"
return $ (+) <$> a <*> b
let r = (+) <$> a <*> b
return r
```

---

# Generalized Algebraic Datatypes (GADTs)

The problem:

<!-- exdown-skip -->
```hs
data Expr a b =
IntExpr Int
| BoolExpr Bool
| PairExpr a b
| IfThenElseExpr Bool (Expr a b) (Expr a b)

eval :: Expr a b -> ?
```

<center>
🤮
</center>

--

GADTs to the rescue!

<!-- exdown-skip: 7 -->
```hs
data Expr t where
IntExpr :: Int -> Expr Int
BoolExpr :: Bool -> Expr Bool
Pair :: Expr a -> Expr b -> Expr (a, b)
IfThenElse :: Expr Bool -> Expr a -> Expr a -> Expr a

eval :: Expr a -> a
```

<center>
😎
</center>

---

# Type Families

* Type families are type-level functions
* Functions that takes types as inputs and produce types

<center>
Type families' main usage is to work with GADTs
</center>

--

```hs
data Mix = MkMix { file :: FilePath, hash :: Int }

-- Phantom types
data File
data Hash

data Role a where
FileRole :: Role File
HashRole :: Role Hash

type family DataKind a where
DataKind File = FilePath
DataKind Hash = Int

-- | @getData m r@ can return either a value of type @FilePath@
-- or a value of type @Int@!
getData :: Mix -> Role a -> DataKind a
getData mix = \case
FileRole -> mix.file
HashRole -> mix.hash
```

---

# Recap

TBD
* Monads
* Generic sequencing operation, supporting interruption
* Safely bring back sequencing programming to functional programming 👑
* GADTs
* Existential types within data constructors
* Fine-grained typing of different constructors
* Type families
* Uniform treatment of non-uniform types

---

# Recommended reading

* [MonadWriter](https://hackage.haskell.org/package/mtl-2.3.1/docs/Control-Monad-Writer-CPS.html#t:MonadWriter)
* [MonadState](https://hackage.haskell.org/package/mtl-2.3.1/docs/Control-Monad-State-Class.html#t:MonadState)
* on Hackage:
* [MonadWriter](https://hackage.haskell.org/package/mtl-2.3.1/docs/Control-Monad-Writer-CPS.html#t:MonadWriter)
* [MonadState](https://hackage.haskell.org/package/mtl-2.3.1/docs/Control-Monad-State-Class.html#t:MonadState)
* Type families: https://serokell.io/blog/type-families-haskell

---

Expand Down
43 changes: 42 additions & 1 deletion tps/Scratch.hs
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,8 @@
{-# LANGUAGE AllowAmbiguousTypes #-}
{-# OPTIONS_GHC -Wno-noncanonical-monad-instances #-}
{-# LANGUAGE InstanceSigs #-}
{-# LANGUAGE GADTs #-}
{-# LANGUAGE TypeFamilies #-}
-- | Module for live trial and error
module Scratch where

Expand Down Expand Up @@ -153,4 +155,43 @@ monadWriterExample = do
tell "Producing a"
b <- pure $ Nothing
tell "Producing b"
return $ (+) <$> a <*> b
return $ (+) <$> a <*> b

data Expr t where
IntExpr :: Int -> Expr Int
BoolExpr :: Bool -> Expr Bool
Pair :: Expr a -> Expr b -> Expr (a, b)
IfThenElse :: Expr Bool -> Expr a -> Expr a -> Expr a


eval :: Expr a -> a
eval = \case
IntExpr x -> x
BoolExpr x -> x
Pair a b -> (eval a, eval b)
IfThenElse cond a b -> if eval cond then eval a else eval b

-- data Expr a b =
-- IntExpr Int
-- | BoolExpr Bool
-- | PairExpr a b
-- | IfThenElseExpr Bool (Expr a b) (Expr a b)

data Mix = MkMix { file :: FilePath, hash :: Int }

-- Phantom types
data File
data Hash

data Role a where
FileRole :: Role File
HashRole :: Role Hash

type family DataKind a where
DataKind File = FilePath
DataKind Hash = Int

getData :: Mix -> Role a -> DataKind a
getData mix = \case
FileRole -> mix.file
HashRole -> mix.hash

0 comments on commit 18b3156

Please sign in to comment.