Skip to content

Commit 1b25f65

Browse files
authored
Merge pull request #14531 from som-snytt/tweak/opaque-example
2 parents 482c67e + 93a7a3a commit 1b25f65

File tree

2 files changed

+54
-35
lines changed

2 files changed

+54
-35
lines changed

docs/_docs/reference/other-new-features/opaques.md

+24-12
Original file line numberDiff line numberDiff line change
@@ -33,9 +33,9 @@ end MyMath
3333

3434
This introduces `Logarithm` as a new abstract type, which is implemented as `Double`.
3535
The fact that `Logarithm` is the same as `Double` is only known in the scope where
36-
`Logarithm` is defined which in the above example corresponds to the object `MyMath`.
37-
Or in other words, within the scope it is treated as type alias, but this is opaque to the outside world
38-
where in consequence `Logarithm` is seen as an abstract type and has nothing to do with `Double`.
36+
`Logarithm` is defined, which in the above example corresponds to the object `MyMath`.
37+
Or in other words, within the scope, it is treated as a type alias, but this is opaque to the outside world
38+
where, in consequence, `Logarithm` is seen as an abstract type that has nothing to do with `Double`.
3939

4040
The public API of `Logarithm` consists of the `apply` and `safe` methods defined in the companion object.
4141
They convert from `Double`s to `Logarithm` values. Moreover, an operation `toDouble` that converts the other way, and operations `+` and `*` are defined as extension methods on `Logarithm` values.
@@ -70,13 +70,12 @@ object Access:
7070
opaque type PermissionChoice = Int
7171
opaque type Permission <: Permissions & PermissionChoice = Int
7272

73-
extension (x: Permissions)
74-
def & (y: Permissions): Permissions = x | y
7573
extension (x: PermissionChoice)
7674
def | (y: PermissionChoice): PermissionChoice = x | y
75+
extension (x: Permissions)
76+
def & (y: Permissions): Permissions = x | y
7777
extension (granted: Permissions)
7878
def is(required: Permissions) = (granted & required) == required
79-
extension (granted: Permissions)
8079
def isOneOf(required: PermissionChoice) = (granted & required) != 0
8180

8281
val NoPermission: Permission = 0
@@ -101,9 +100,12 @@ where `x | y` means "a permission in `x` *or* in `y` granted".
101100

102101
Note that inside the `Access` object, the `&` and `|` operators always resolve to the corresponding methods of `Int`,
103102
because members always take precedence over extension methods.
104-
Because of that, the `|` extension method in `Access` does not cause infinite recursion.
105-
Also, the definition of `ReadWrite` must use `|`,
106-
even though an equivalent definition outside `Access` would use `&`.
103+
For that reason, the `|` extension method in `Access` does not cause infinite recursion.
104+
105+
In particular, the definition of `ReadWrite` must use `|`, the bitwise operator for `Int`,
106+
even though client code outside `Access` would use `&`, the extension method on `Permissions`.
107+
The internal representations of `ReadWrite` and `ReadOrWrite` are identical, but this is not visible to the client,
108+
which is interested only in the semantics of `Permissions`, as in the example below.
107109

108110
All three opaque type aliases have the same underlying representation type `Int`. The
109111
`Permission` type has an upper bound `Permissions & PermissionChoice`. This makes
@@ -115,8 +117,11 @@ object User:
115117
import Access.*
116118

117119
case class Item(rights: Permissions)
120+
extension (item: Item)
121+
def +(other: Item): Item = Item(item.rights & other.rights)
118122

119123
val roItem = Item(Read) // OK, since Permission <: Permissions
124+
val woItem = Item(Write)
120125
val rwItem = Item(ReadWrite)
121126
val noItem = Item(NoPermission)
122127

@@ -128,11 +133,18 @@ object User:
128133

129134
assert(!noItem.rights.is(ReadWrite))
130135
assert(!noItem.rights.isOneOf(ReadOrWrite))
136+
137+
assert((roItem + woItem).rights.is(ReadWrite))
131138
end User
132139
```
133-
134-
On the other hand, the call `roItem.rights.isOneOf(ReadWrite)` would give a type error
135-
since `Permissions` and `PermissionChoice` are different, unrelated types outside `Access`.
140+
On the other hand, the call `roItem.rights.isOneOf(ReadWrite)` would give a type error:
141+
```scala
142+
assert(roItem.rights.isOneOf(ReadWrite))
143+
^^^^^^^^^
144+
Found: (Access.ReadWrite : Access.Permissions)
145+
Required: Access.PermissionChoice
146+
```
147+
`Permissions` and `PermissionChoice` are different, unrelated types outside `Access`.
136148

137149

138150
### Opaque Type Members on Classes

tests/pos/reference/opaque.scala

+30-23
Original file line numberDiff line numberDiff line change
@@ -1,77 +1,84 @@
1-
object Logarithms {
1+
object MyMath:
22

33
opaque type Logarithm = Double
44

5-
object Logarithm {
5+
object Logarithm:
6+
7+
// These are the two ways to lift to the Logarithm type
68

7-
// These are the ways to lift to the logarithm type
89
def apply(d: Double): Logarithm = math.log(d)
910

1011
def safe(d: Double): Option[Logarithm] =
11-
if (d > 0.0) Some(math.log(d)) else None
12-
}
12+
if d > 0.0 then Some(math.log(d)) else None
13+
14+
end Logarithm
1315

1416
// Extension methods define opaque types' public APIs
1517
extension (x: Logarithm)
1618
def toDouble: Double = math.exp(x)
1719
def + (y: Logarithm): Logarithm = Logarithm(math.exp(x) + math.exp(y))
1820
def * (y: Logarithm): Logarithm = x + y
19-
}
2021

21-
object LogTest {
22-
import Logarithms.*
22+
end MyMath
23+
24+
object LogTest:
25+
import MyMath.Logarithm
2326
import Predef.{any2stringadd as _, *}
2427

2528
val l = Logarithm(1.0)
2629
val l2 = Logarithm(2.0)
2730
val l3 = l * l2
2831
val l4 = l + l2
29-
}
3032

3133

32-
object Access {
34+
35+
object Access:
3336

3437
opaque type Permissions = Int
3538
opaque type PermissionChoice = Int
3639
opaque type Permission <: Permissions & PermissionChoice = Int
3740

38-
extension (x: Permissions)
39-
def & (y: Permissions): Permissions = x | y
4041
extension (x: PermissionChoice)
4142
def | (y: PermissionChoice): PermissionChoice = x | y
43+
extension (x: Permissions)
44+
def & (y: Permissions): Permissions = x | y
4245
extension (granted: Permissions)
4346
def is(required: Permissions) = (granted & required) == required
44-
extension (granted: Permissions)
4547
def isOneOf(required: PermissionChoice) = (granted & required) != 0
4648

4749
val NoPermission: Permission = 0
4850
val Read: Permission = 1
4951
val Write: Permission = 2
5052
val ReadWrite: Permissions = Read | Write
5153
val ReadOrWrite: PermissionChoice = Read | Write
52-
}
5354

54-
object User {
55+
end Access
56+
57+
object User:
5558
import Access.*
5659

5760
case class Item(rights: Permissions)
61+
extension (item: Item)
62+
def +(other: Item): Item = Item(item.rights & other.rights)
5863

5964
val roItem = Item(Read) // OK, since Permission <: Permissions
65+
val woItem = Item(Write)
6066
val rwItem = Item(ReadWrite)
6167
val noItem = Item(NoPermission)
6268

63-
assert( roItem.rights.is(ReadWrite) == false )
64-
assert( roItem.rights.isOneOf(ReadOrWrite) == true )
69+
assert(!roItem.rights.is(ReadWrite))
70+
assert(roItem.rights.isOneOf(ReadOrWrite))
6571

66-
assert( rwItem.rights.is(ReadWrite) == true )
67-
assert( rwItem.rights.isOneOf(ReadOrWrite) == true )
72+
assert(rwItem.rights.is(ReadWrite))
73+
assert(rwItem.rights.isOneOf(ReadOrWrite))
6874

69-
assert( noItem.rights.is(ReadWrite) == false )
70-
assert( noItem.rights.isOneOf(ReadOrWrite) == false )
75+
assert(!noItem.rights.is(ReadWrite))
76+
assert(!noItem.rights.isOneOf(ReadOrWrite))
7177

78+
assert((roItem + woItem).rights.is(ReadWrite))
7279
// Would be a type error:
73-
// assert( roItem.rights.isOneOf(ReadWrite) == true )
74-
}
80+
// assert(roItem.rights.isOneOf(ReadWrite))
81+
end User
7582

7683
object o {
7784
opaque type T = Int

0 commit comments

Comments
 (0)