-
Notifications
You must be signed in to change notification settings - Fork 96
/
Copy pathClassesAndTraits.scala
129 lines (102 loc) · 4.79 KB
/
ClassesAndTraits.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
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
package com.evolutiongaming.bootcamp.basics
object ClassesAndTraits {
// You can follow your progress using the tests in `ClassesAndTraitsSpec`:
// sbt "testOnly com.evolutiongaming.bootcamp.basics.ClassesAndTraitsSpec"
// Classes in Scala are blueprints for creating object instances. They can contain methods, values,
// variables, types, objects, traits, and classes which are collectively called members.
class MutableUser(var login: String, var balance: Double) {
def addMoney(amount: Double): Unit = changeBalance(amount)
def takeMoney(amount: Double): Unit = changeBalance(-amount)
private def changeBalance(amount: Double): Unit =
balance += amount
override def toString: String =
s"User($login, $balance)"
}
// Singleton objects are defined using `object`.
// It is a class that has exactly one instance.
// An object with the same name as a class is called a companion object.
//
// Use it to contain methods and values related to this trait or class, but that aren't
// specific to instances of this trait or class.
//
// User specific methods `apply` and `unapply` to construct and deconstruct classes
object MutableUser {
def apply(login: String): MutableUser = new MutableUser(login, 0)
def unapply(user: MutableUser): Option[(String, Double)] = Some(user.login, user.balance)
}
val mutableUser = MutableUser("potter")
mutableUser.login // "potter"
mutableUser.balance // 0.0
// Question. Is MutableUser a good design? Why or why not?
// Traits define a common interface that classes conform to. They are similar to Java's interfaces.
// A trait can be thought of as a contract that defines the capabilities and behaviour of a component.
// Subtyping
// Where a given trait is required, a subtype of the trait can be used instead.
// Classes and singleton objects can extend traits.
//
// This allows "programming to the interface" approach where you depend on traits instead of their
// specific implementations (classes or objects).
//
// This makes code more reusable and testable.
trait HasBalance {
def balance: Double
}
// Exercise. Implement a method that will allow aggregating all entities with balances into one
// that will have the sum of all balances inside.
//
def totalBalance(accounts: List[HasBalance]): HasBalance =
new HasBalance {
def balance: Double = ???
}
trait Account extends HasBalance {
// def addMoney(amount: Double)
// def takeMoney(amount: Double)
}
sealed trait User {
def login: String
}
final case class RegularUser(login: String, balance: Double) extends User with Account
case object Admin extends User {
val login: String = "admin"
}
// Case Classes
//
// Case classes are like regular classes, but with extra features which make them good for modelling
// immutable data. They have all the functionality of regular classes, but the compiler generates additional
// code, such as:
// - Case class constructor parameters are public `val` fields, publicly accessible
// - `apply` method is created in the companion object, so you don't need to use `new` to create a new
// instance of the class
// - `unapply` method which allows you to use case classes in `match` expressions (pattern matching)
// - a `copy` method is generated
// - `equals` and `hashCode` methods are generated, which let you compare objects & use them in collections
// - `toString` method is created for easier debugging purposes
// - extends Serializable
//
// Case object provides default equals, hashCode, unapply, toString methods, extends Serializable
//
// When declaring a case class, make it final. Otherwise someone may decide to inherit from it
// case-to-case inheritance is prohibited
// calls .apply method
val user = RegularUser("potter", 200)
// calls .unapply method
val RegularUser(login, balance) = user
val updatedUser = user.copy(balance = 1000)
println(updatedUser.toString) // RegularUser(potter,1000)
// Let's go back to the `Account` trait, which contains two methods for changing the balance:
//
// def addMoney(amount: Double)
// def takeMoney(amount: Double)
//
// How should we implement these methods for all heirs at once?
// What should be the return type of these methods in `Account`? In `RegularUser`?
// Generic classes and type parameters
// In a similar way as we saw with polymorphic methods, classes and traits can also take type parameters.
// For example, you can define a `Stack[A]` which works with any type of element `A`.
// Question. Do you agree with how the stack is modelled here? What would you do differently?
final case class Stack[A](elements: List[A] = Nil) {
def push(x: A): Stack[A] = ???
def peek: A = ???
def pop: (A, Stack[A]) = ???
}
}