-
Notifications
You must be signed in to change notification settings - Fork 13
/
Copy pathLensFeature.scala
140 lines (120 loc) · 4.6 KB
/
LensFeature.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
130
131
132
133
134
135
136
137
138
139
140
package app.impl.scalaz
import org.junit.Test
import scalaz.Lens
/**
* In scala when we manage immutable class and we want to "modify" a value of this class, we create a new class since
* the other it´s immutable and we copy every value from one to another but the new value which it´s set in the new class.
* When we have multiple nested classes the thing get complicated and verbose a little bit.
*
* That´s when Scalaz Lens comes to help us. Basically the Lens pattern it's meant to create "links of copy" like if the
* class and the nested classes would be a chain, where every link it's able to modify a value of that chain.
*
* Then the idea is that you can reuse those links and compose chains together to modify values of the chain.
*/
class LensFeature {
case class Name(value: String)
case class Person(name: Name, age: Int)
case class Man(person: Person)
case class Me(man: Man)
//### Original value ######
val pablo = Me(Man(Person(Name("Pablo"), 36)))
val john = Me(Man(Person(Name("Johny"), 30)))
//##############
//### LENS ####
//##############
/**
* Here we define our lens in order to be reused in future copy from our immutable class.
* We define a link in the chain per len, between the class that contains the value to copy [ContainerClass]
* and the value to be copy T. Lens[ContainerClass, T]
* In here we have a len which will copy the class Name inside the class Person [Person, Name]
*/
val nameLen: Lens[Person, Name] = Lens.lensu[Person, Name](
(person, _name) => person.copy(name = _name), _.name
)
val ageLen: Lens[Person, Int] = Lens.lensu[Person, Int](
(person, _age) => person.copy(age = _age), _.age
)
val personLen: Lens[Man, Person] = Lens.lensu[Man, Person](
(man, _person) => man.copy(person = _person), _.person
)
val manLen: Lens[Me, Man] = Lens.lensu[Me, Man](
(me, _man) => me.copy(man = _man), _.man
)
/**
* This is an example how the copy of one value from one case class it would be.
*/
@Test
def normalCopy(): Unit = {
println(pablo)
val paul = pablo
.copy(man = pablo.man
.copy(person = pablo.man.person
.copy(name = pablo.man.person.name
.copy("Paul"), pablo.man.person.age)))
println(paul)
}
/**
* Here instead we use lens for the Name class, as you can see it´s less verbose and most important the already created lens
* are reusable.
* Here we combine lens as links to create a chain [val chain = link >=> link >=> link] from the origin to the link that contains
* the attribute to change.
* Once that we have the chain of lens we use it passing the original instance and the value that we want to modify from the
* original instance.
* This lens internally will make all the verbose copies and it will return a new instance with the copied values and the new one
* that you want to modify.
*/
@Test
def copyNameWithLens(): Unit = {
println(pablo)
val lensForName = manLen >=> personLen >=> nameLen
val paul = lensForName.set(pablo, Name("Paul"))
println(paul)
}
@Test
def copyAgeWithLens(): Unit = {
println(pablo)
val lensForAge = manLen >=> personLen >=> ageLen
val pabloWithAge = lensForAge.set(pablo, 29)
println(pabloWithAge)
}
/**
* In this example we see that we can use multiple chains of lens in one instance at the same time, to return
* a new instance with all changes at once.
*/
@Test
def copyAgeAndNameWithLens(): Unit = {
println(pablo)
val lensForPerson = manLen >=> personLen
val lensForAge = lensForPerson >=> ageLen
val lensForName = lensForPerson >=> nameLen
val paulWithAge = lensForAge.set(lensForName.set(pablo, Name("Paul")), 29)
println(paulWithAge)
}
@Test
def copyNameWithLensAndGetAllLevels(): Unit = {
println(pablo)
val lensForName = manLen >=> personLen >=> nameLen
val paul = lensForName.set(pablo, Name("Paul"))
val man = lensForName.set(pablo, Name("Paul")).man
val person = lensForName.set(pablo, Name("Paul")).man.person
val name = lensForName.set(pablo, Name("Paul")).man.person.name
println(paul)
println(man)
println(person)
println(name)
}
/**
* You can also not only set value and create new class but get values from the existing one.
*/
@Test
def getWithLens(): Unit = {
println(pablo)
val lensForPerson = manLen >=> personLen
val lensForAge = lensForPerson >=> ageLen
val lensForName = lensForPerson >=> nameLen
val age = lensForAge.get(pablo)
val name = lensForName.get(john)
println(age)
println(name)
}
}