Scala: Self type annotations and structured types
A few days ago I tweeted that I didn’t really see the point in structured types in Scala…
Not sure I understand where you would use structural types in #scala instead of defining a method on a trait http://bit.ly/jgiW7b
…but today my colleague Uday came up with a cool way of combining self type annotations with structured types inside a trait we defined.
We had some code duplicated across two classes which looked roughly like this:
class OnePageType {
lazy val peopleNodes = root \\ "SomeNode" \ "SomeSubNode" \ "People" \ "Person"
private def fullName(personName: Node): String = // code to build person's name
lazy val people: String = peopleNodes.map(fullName).mkString(", ")
}
class AnotherPageType {
lazy val peopleNodes = root \\ "OtherNode" \ "OtherSubNode" \ "People" \ "Person"
private def fullName(personName: Node): String = // code to build person's name
lazy val people: String = peopleNodes.map(fullName).mkString(", ")
}
The first line is different but the other two are identical because the data is stored in exactly the same format once we get down to that level.
Since We want to keep the XPathish queries as descriptive as possible so that we don’t accidentally end up pulling the wrong elements onto the page, making those a bit looser wasn’t an option in this case.
Instead we pulled out a trait like so:
trait People {
self: {val peopleNodes: NodeSeq} =>
private def fullName(personName: Node): String = // code to build person's name
lazy val people: String = peopleNodes.map(fullName).mkString(", ")
}
Which we include in the classes like this:
class OnePageType extends People {}
class AnotherPageType extends People {}
What we’re done on line 2 of the People trait is to define a self annotation which says that we need a val of peopleNodes to be present on the classes in which the trait is mixed.
If a val of peopleNodes doesn’t exist then the class won’t compile!
In this case the structure type works quite well because we wouldn’t really want to pull out peopleNodes into a trait just to reference it as a self type annotation.
About the author
I'm currently working on short form content at ClickHouse. I publish short 5 minute videos showing how to solve data problems on YouTube @LearnDataWithMark. I previously worked on graph analytics at Neo4j, where I also co-authored the O'Reilly Graph Algorithms Book with Amy Hodler.