alias blame-lines="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 | 66,698 |
Dmitry Petrashko | 628 | 158,249 | 50,201 | 83,713 |
Samuel Gruetter | 19 | 62,324 | 12,510 | 36,615 |
Others | 144 | 5,315 | 2,108 | 4,161 |
Java | Scala |
|
|
45 msec | 872 msec |
| Java cycle body:
|
|
Scala cycle body:
|
def plus[T](a: T, b: T)(implicit num: Numeric[T]): T =
num.plus(a, b)
is compiled into
def plus(a: Object, b: Object, num: Numeric): Object =
num.plus(a, b)
How does plus(1, 1)
look like?
unbox(plus(box(1), box(1), intNum))
def plus(a: Int, b: Int, num: Numeric[Int]): Int =
num.plus(a, b)
def plus(a: Long, b: Long, num: Numeric[Long]): Long =
num.plus(a, b)
def plus(a: Double, b: Double, num: Numeric[Double]): Double =
num.plus(a, b)
...
Scala has 9 primitive types:
+ we need a reference type.
= 10 types total.
class T[A1, A2, ..., An]
How many combinations are there?
10 ^ n
class T[A1, A2, ..., An] {
def foo[B1, B2, ..., Bn] = ...
}
How many combinations are there?
10 ^ (n + m)
Dragos(2010): Ask your user!
trait Function3[-T1, -T2, -T3, +R]
Type-arguments and term-args are treated in the same way.
def foo[T](x: Seq[T], id: Int) = x(id)
val arr = (1 to 10000000).toArray
arr.reduce(_ + _).toDouble / arr.length
Java | Scala 2.11 | Scala 2.12.0-M3 | ScalaBlitz | Linker | ScalaJS |
45ms | 872ms | 680ms | 42ms | 68ms1 | 2440ms |
|
|
Fast | Maintainable |
|
|
x.length == 0
becomes
x.isEmpty
import dotty.linker._
@rewrites
object CollectionOptimization {
def isEmpty(x: Seq[Int]) =
Rewrite(from = x.length == 0,
to = x.isEmpty)
def bigIntShift(bi: BigInt, a: Int)(implicit evi: IsLiteral[a.type]) =
Rewrite(from = bi / a,
to =
if (Integer.bitCount(a) == 1)
bi >> Integer.numberOfTrailingZeros(a)
else bi / a
)
BigInt(10) / 2
becomes
if (Integer.bitCount(2) == 1)
BigInt(10) >> Integer.numberOfTrailingZeros(2)
else BigInt(10) / 2
List(1, 2, 3).filter(_ > 1).filter(_ > 2)
becomes
List(1, 2, 3).filter(x => x > 1 && x > 2)
def twoFilters(x: List[Int], a: Int => Boolean, b: Int => Boolean)
(implicit apure: IsPure[a.type]) =
Rewrite(from = x.filter(a).filter(b),
to = x.filter(x => a(x) && b(x)))
def parallelReduceLeft(ps: ParSeq[Int]) =
ps.reduceLeft(_ + _)
[warn] TestPar.scala:40: reduceLeft on a parallel collection makes no sense
[warn] List(1, 2, 3).toPar.reduceLeft(_ + _)
[warn] ^
def customFancyWarning(ps: ParSeq[Int], x: (Int, Int) => Int) =
Warn(pattern = ps.reduceLeft(x)),
msg = "reduceLeft on a parallel collection makes no sense")
import java.util.Logging._
def traceOpt(l: Logger, x: Any) =
Rewrite(from = l.info(x),
to = if (l.isLoggable(INFO)) l.log(TRACE, x)
)
Rewrite rules | Macros & Inline | |
Defined | Anywhere in the program & libraries | inside implementation |
Applied | Based on analysis | Syntax only |
Influences typing | No | Yes |
Works with Java methods | Yes | No |
Works across libraries | Yes | No |
Best for | simple declarative rules | full-blown metaprogramming |
def metaExample[T](x: List[T]) =
Rewrite(from = x.toString,
to = meta { /* entry point to interpreted scala.meta macros */})
Macros More in Eugene's talk tomorrow.
$ sbt test
+ Collections.twoFilters: OK, passed 100 tests.
+ Collections.isEmpty: OK, passed 100 tests.
+ Collections.twoDrops: OK, passed 100 tests.
+ Collections.mapFlatMap: OK, passed 100 tests.
+ Collections.mapMap: OK, passed 100 tests.
List.flatMap