Dmitry Petrashko
git ls-tree -r -z --name-only HEAD -- |egrep -z -Z -E '\.(scala)$'\
| xargs -0 -n1 git blame --line-porcelain |grep "^author "|sort|uniq -c|sort -nr
author | commits | additions | deletions | blame lines |
Martin Odersky | 2,280 | 146,133 | 85,391 | 57,284 |
Dmitry Petrashko | 628 | 158,249 | 50,201 | 83,929 |
Samuel Gruetter | 19 | 62,324 | 12,510 | 36,615 |
Ondrej Lhotak | 29 | 2,009 | 340 | 1,874 |
Guillaume Martres | 77 | 2,197 | 1,190 | 1,171 |
Vera Salvisberg | 17 | 1,109 | 364 | 819 |
Jason Zaugg | 16 | 213 | 149 | 153 |
Alexander Myltsev | 5 | 275 | 65 | 144 |
Vlad Ureche | 13 | 255 | 77 | 51 |
similar to Scala2 with
def foo(arg: Serializable & Comparable): Unit
similar to Scala2 with
trait A {
type T = Int
}
trait B {
type T = Double
}
(A with B) # T == Int
(A & B) # T == Int & Double
what is the difference?
with
is ordered, &
is not
A with B is different from B with A. & is not
Scala2 equivalent: none
class A(x: Int)
class B(x: Int)
class C(x: Array[Int])
object Test{
def foo(x: A | B): Int = x.x
def bar(x: A | C): Int | Array[Int] = x.x
}
object DaysOfTheWeek{
object Mon
object Tue
object Wed
object Thu
object Fri
object Sat
object Sun
type Weekend = Sat.type | Sun.type
type Workweek = Mon.type | Tue.type | Wed.type | Thu.type | Fri.type
type All = Weekend | Workweek
}
object Literals{
val fortyTwo: 42 = 42
val `2`: 2 = 2
val fortyFour: 44 = fortyTwo + `2`
val text: "text" = "text"
def id[T](a: T) = a
val two: 2 = id(`2`)
}
=> T*
means Function0[Seq[T]]
object DaysOfTheWeek{
def foo(a: => Any*) = ()
def bar(a: => Any*) = foo(a : _*)
def baz(a: => Seq[Any]) = foo(a : _*)
bar(???, ???)
baz(Seq(???, ???))
}
trait A(val i: Int)
trait A(val i: Int)
trait A1 extends A(1)
trait A2 extends A(2)
object O extends A1 with A2 {
// is i == 1 or i == 2
// how many times was it initialized?
}
trait A(val i: Int)
trait B extends A
class C extends B with A(1)
object O extends A(2) with B
object A {
lazy val a0 = B.b
lazy val a1 = 17
}
object B {
lazy val b = A.a1
}
object A {
lazy val a0 = B.b // <-- Thread 1
lazy val a1 = 17
}
object B {
lazy val b = A.a1 // <-- Thread 2
}
object A {
lazy val a0 = B.b // <-- Thread 1, grabs lock
lazy val a1 = 17
}
object B {
lazy val b = A.a1 // <-- Thread 2
}
object A {
lazy val a0 = B.b // <-- Thread 1, grabs lock
lazy val a1 = 17
}
object B {
lazy val b = A.a1 // <-- Thread 2, grabs lock
}
object A {
lazy val a0 = B.b // <-- Thread 1, grabs lock
lazy val a1 = 17 // Potential deadlock
}
object B {
lazy val b = A.a1 // <-- Thread 2, grabs lock
}
object C {
lazy val a0 = { /* long computation with IO */ } // <-- Thread 1
lazy val a1 = 42 // <-- Thread 2 will be waiting for Thread 1 to finish
}
Dotty lazy values:
object Test {
def foo() { // procedure syntax
}
}
object Test {
def foo(): Unit = { // procedure syntax
}
}
/* Classes and objects (but note, not traits) inheriting the `DelayedInit`
* marker trait will have their initialization code rewritten as follows:
* `code` becomes `delayedInit(code)`
*/
trait Helper extends DelayedInit {
def delayedInit(body: => Unit) = {
println("dummy text, printed before initialization of C")
body // evaluates the initialization code of C
}
}
class C extends Helper {
println("this is the initialization code of C")
}
new C
// dummy text, printed before initialization of C
// this is the initialization code of C
trait Helper(body: => Unit) {
println("dummy text, printed before initialization of C")
body // evaluates the initialization code of C
}
class C extends Helper(println("this is the initialization code of C"))
new C
// dummy text, printed before initialization of C
// this is the initialization code of C
Java 8 is coming
Proposed by scala.meta.
Interchange format that stores:
Scalac is a big cake:
Dotty has several small cakes:
def typed(tree: untpd.Tree, pt: Type = WildcardType)(implicit ctx: Context): Tree
def tryEither[T](op: Context => T)(fallBack: (Context) => T)(implicit ctx: Context) = {
val nestedCtx = ctx.fresh.setNewTyperState
val result = op(nestedCtx)
if (nestedCtx.reporter.hasErrors)
fallBack(ctx)
else {
nestedCtx.typerState.commit()
result
}
}
They change meaning depending on phase of compilation:
class C {
def id[T](t: T) = t
}
/** A transformer transforms denotations at a given phase */
trait DenotTransformer extends Phase {
/** The transformation method */
def transform(ref: SingleDenotation)(implicit ctx: Context): SingleDenotation
}
phase name id description
---------- -- -----------
parser 1 parse source into ASTs, perform simple desugaring
namer 2 resolve names, attach symbols to named trees
packageobjects 3 load package objects
typer 4 the meat and potatoes: type the trees
patmat 5 translate match expressions
superaccessors 6 add super accessors in traits and nested classes
extmethods 7 add extension methods for inline classes
pickler 8 serialize symbol tables
refchecks 9 reference/override checking, translate nested objects
uncurry 10 uncurry, translate function values to anonymous classes
tailcalls 11 replace tail calls by jumps
specialize 12 @specialized-driven class and method specialization
explicitouter 13 this refs to outer pointers
erasure 14 erase types, add interfaces for traits
posterasure 15 clean up erased inline classes
lazyvals 16 allocate bitmaps, translate lazy vals into lazified defs
lambdalift 17 move nested functions to top level
constructors 18 move field definitions into constructors
flatten 19 eliminate inner classes
mixin 20 mixin composition
cleanup 21 platform-specific cleanups, generate reflective calls
delambdafy 22 remove lambdas
icode 23 generate portable intermediate code
jvm 24 generate JVM bytecode
terminal 25 the last phase during a compilation run
Why? Macro phases cost a lot
Half of phases do `something` additionally
Under Y-check:all after each phase:
Examples of postconditions: