-
Notifications
You must be signed in to change notification settings - Fork 3
/
Typeclasses.scala
99 lines (80 loc) · 3.42 KB
/
Typeclasses.scala
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
package uk.co.tobyhobson
/**
* Simple example demonstrating the use of the type class pattern
* see https://www.tobyhobson.co.uk/type-classes-for-beginners/
*/
object Typeclasses {
import scala.reflect.ClassTag
// Our 3 types
case class Car(make: String)
case class Bus(colour: String)
case class Tank(numGuns: Int)
// Core type class
trait VehicleLike[A] {
def drive(a: A): String
}
// Type class implementations for cars, busses and tanks
implicit val vehicleLikeCar = new VehicleLike[Car] {
override def drive(car: Car): String = s"driving a ${car.make} car"
}
implicit val vehicleLikeBus = new VehicleLike[Bus] {
override def drive(bus: Bus): String = s"driving a ${bus.colour} bus"
}
implicit val vehicleLikeTank = new VehicleLike[Tank] {
override def drive(tank: Tank): String = s"driving a tank with ${tank.numGuns} guns"
}
// Implementation for a List of any type
// note how we need a type class implementation in scope for the underlying type
implicit def vehicleListList[A](implicit evidence: VehicleLike[A]) = new VehicleLike[List[A]] {
override def drive(aa: List[A]): String = {
s"List type class: ${System.lineSeparator()}" + aa.map(a => evidence.drive(a)).mkString(System.lineSeparator())
}
}
/**
* A variation on the List type class illustrating two new features:
*
* 1. Instead of specifying the evidence as an implicit parameter we use the A: VehicleLike and implicitly shortcuts
* 2. We use another type class called ClassTag to get the type of A at runtime
*
* Note: A: VehicleLike : ClassTag means "any A so long as there are Vehicle and ClassTag type classes in scope"
*
* @tparam A
* @return
*/
implicit def vehicleLikeOption[A: VehicleLike : ClassTag] = new VehicleLike[Option[A]] {
override def drive(aa: Option[A]): String = {
// Use the ClassTag type class to get the runtime class name of A
val className = implicitly[ClassTag[A]].runtimeClass.getSimpleName
aa.map(a =>
s"Option[$className] type class: " + implicitly[VehicleLike[A]].drive(a)
).getOrElse(
s"Option[$className] type class: Nothing to drive"
)
}
}
def driveAndReturn[A](vehicle: A)(implicit evidence: VehicleLike[A]): A = {
println(evidence.drive(vehicle)); vehicle
}
def driveAndReturn2[A: VehicleLike](vehicle: A): A = {
println(implicitly[VehicleLike[A]].drive(vehicle)); vehicle
}
def main(args: Array[String]): Unit = {
val cars = List(Car("ford"), Car("BMW"), Car("VM"))
val busses = List(Bus("red"), Bus("yellow"))
val tanks = List(Tank(1), Tank(2))
val drivenFord: Car = driveAndReturn(cars.head)
val drivenCars: List[Car] = driveAndReturn(cars)
val drivenBus: Bus = driveAndReturn(busses.head)
val drivenBusses: List[Bus] = driveAndReturn(busses)
val drivenTank: Tank = driveAndReturn(tanks.head)
val drivenTanks: List[Tank] = driveAndReturn(tanks)
// Option[Tank]
val drivenOptionalTank: Option[Tank] = driveAndReturn(tanks.headOption)
// List[Option[Tank]]
val drivenListOptionalTanks: List[Option[Tank]] = driveAndReturn(tanks.map(t => Some(t)) :+ None)
// Option[List[Tank]]
// Note how we need to tell the compiler to treat Some(tanks) as Option(tanks)
// because we have a type class defined for Option, not Some
val drivenOptionalListTanks: Option[List[Tank]] = driveAndReturn(Some(tanks): Option[List[Tank]])
}
}