在上期讨论中我们介绍了Scala Macros,它可以说是工具库编程人员不可或缺的编程手段,可以实现编译器在编译源代码时对源代码进行的修改、扩展和替换,如此可以对用户屏蔽工具库复杂的内部细节,使他们可以用简单的声明方式,通过编译器自动产生铺垫代码来实现工具库中各种复杂的类型、对象及方法函数的构建。虽然Def Macros可能具备超强的编程功能,但同时使用者也普遍认为它一直存有着一些严重的诟病:包括用法复杂、容易犯错、运算行为难以预测以及没用完善的集成开发环境工具(IDE)支持等。这些恶评主要是因为Def Macros和编译器scalac捆绑的太紧,使用者必须对编译器的内部运作原理和操作函数有比较深刻的了解。加之Def Macros向用户提供的api比较复杂且调用繁琐,其中比较致命的问题就是与scalac的紧密捆绑了:因为Def Macros还只是一项实验性功能,没有scala语言规范文件背书,肯定会面临升级换代。而且scala本身也面临着向2.12版本升级的情况,其中dotty就肯定是scalac的替代编译器。Scalameta是根据scala语言规范SIP-28-29-Inline-Macros由零重新设计的Macros编程工具库。主要目的就是为了解决Def Macros所存在的问题,而且Jetbrains的IntelliJ IDEA 2016.3 EAP对Scalameta已经有了比较好的支持,能为使用者带来更简单、安全的Macros编程工具。

我在介绍了Slick之后立即转入Scala Macros是有一些特别目的的。研究FRM Slick乃至学习泛函编程的初衷就是希望能为传统的OOP编程人员提供更简单易用的泛函库应用帮助,使他们无须对函数式编程模式有太深刻了解也能使用由函数式编程模式所开发的函数库。实现这个目标的主要方式就是Macros了。希望通过Macros的产生代码功能把函数库的泛函特性和模式屏蔽起来,让用户能用他们习惯的方式来定义函数库中的类型对象、调用库中的方法函数。

Macros功能实现方式(即编译时的源代码扩展compile time expansion)由两个主要部分组成:一是在调用时扩展(on call expansion),二是申明时扩展即注释(annotation)。这两种方式我们在上一篇讨论里都一一做了示范。通过测试发现,Scalameta v1.x只支持注释方式。这事动摇了我继续探讨的意愿:试想如果没了”Implicit Macros“,“Extractor Macros“这些模式,会损失多少理想有趣的编码方式。通过与Scalameta作者沟通后得知他们将会在Scalameta v2.x中开始支持全部两种模式,因此决定先介绍一下Scalameta v1.x,主要目的是让大家先熟悉了解Scalameta新的api和使用模式。我们可以把上次Def Macros的Macros Annotations示范例子在Scalameta里重新示范一遍来达到这样的目的。

虽然Scalameta是从头设计的,但是它还是保留了许多Def Macros的思想,特别是沿用了大部分scala-reflect的quasiquote模式。与Def Macros运算原理相同,Scalameta的Macros扩展也是基于AST(abstract syntax tree)由编译器运算产生的,因此Macros申明必须先完成编译,所以我们还是沿用了上一篇讨论中的build.sbt,保留项目结构,及demos对macros的这种依赖关系。

 name := "learn-scalameta"

 val commonSettings = Seq(
version := "1.0" ,
scalaVersion := "2.11.8",
scalacOptions ++= Seq("-deprecation", "-feature"),
resolvers += Resolver.sonatypeRepo("snapshots"),
addCompilerPlugin(
"org.scalameta" % "paradise" % "3.0.0-M5" cross CrossVersion.full),
scalacOptions += "-Xplugin-require:macroparadise" )
val macrosSettings = Seq(
libraryDependencies += "org.scalameta" %% "scalameta" % "1.3.0",
libraryDependencies += "org.scalatest" %% "scalatest" % "3.0.1" % "test"
)
lazy val root = (project in file(".")).aggregate(macros, demos) lazy val macros = project.in(file("macros")).
settings(commonSettings : _*).
settings(macrosSettings : _*) lazy val demos = project.in(file("demos")).settings(commonSettings : _*).dependsOn(macros)

下面我们先用一个最简单的例子来开始了解Scalameta Macros Annotations:

 object MacroAnnotDemo extends App {

   @Greetings object Greet {
def add(x: Int, y: Int) = println(x + y)
} Greet.sayHello("John")
Greet.add(,)
}

这里的注释@Greetings代表被注释对象Greet将会被扩展增加一个sayHello的函数。我们看看这个注释的实现方式:

 import scala.meta._

 class Greetings extends scala.annotation.StaticAnnotation {
inline def apply(defn: Any): Any = meta {
defn match {
case q"object $name {..$stats}" => {
q"""
object $name {
def sayHello(msg: String): Unit = println("Hello," + msg)
..$stats
}
"""
}
case _ => abort("annottee must be object!")
}
}
}

首先,我们看到这段源代码表达方式直接了许多:只需要import scala.meta,没有了blackbox、whitebox、universe这些imports。特别是避免了对blackbox.Context和whitebox.Context这些复杂运算域的人为判定。quasiquote的使用没有什么变化。直观上Macros编程简单了,实际上编写的Macros程序能更安全稳定的运行。

我们再重复演示方法注释(method annotation)的实现方法:

 class Benchmark extends scala.annotation.StaticAnnotation {
inline def apply(defn: Any): Any = meta {
defn match {
case q"..$mod def $name[..$tparams](...$args): $rtpe = $body" =>
q"""
..$mod def $name[..$tparams](...$args): $rtpe = {
val start = System.nanoTime()
val result = $body
val end = System.nanoTime()
println(${name.toString} + " elapsed time = " + (end - start) + "ns")
result
}
"""
case _ => abort("Fail to expand annotation Benchmark!")
}
}
}

还是固定格式。只是quasiquote的调用组合变化。用下面方法调用测试:

   @Benchmark
def calcPow(x: Double, y: Double) = {
val z = x + y
math.pow(z,z)
} println(calcPow(4.2, 8.9))

在下面这个例子里我们在注释对象中增加main方法(未extends App的对象):

 import scala.meta.Ctor.Call
class main extends scala.annotation.StaticAnnotation {
inline def apply(defn: Any): Any = meta {
def abortIfObjectAlreadyExtendsApp(ctorcalls: scala.collection.immutable.Seq[Call], objectName: Term) = {
val extendsAppAlready = ctorcalls.map(_.structure).contains(ctor"App()".structure)
if (extendsAppAlready){
abort(s"$objectName already extends App")
}
}
defn match {
case q"..$mods object $name extends $template" => template match {
case template"{ ..$stats1 } with ..$ctorcalls { $param => ..$stats2 }" =>
abortIfObjectAlreadyExtendsApp(ctorcalls, name)
val mainMethod = q"def main(args: Array[String]): Unit = { ..$stats2 }"
val newTemplate = template"{ ..$stats1 } with ..$ctorcalls { $param => $mainMethod }" q"..$mods object $name extends $newTemplate"
}
case _ => abort("@main can be annotation of object only")
}
}
}

下面这个是case class的注释示例:效果是添加一个从case class转Map的类型转换函数toMap:

 @compileTimeOnly("@Mappable not expanded")
class Mappable extends StaticAnnotation {
inline def apply(defn: Any): Any = meta {
defn match {
case q"..$mods class $tname[..$tparams] (...$paramss) extends $template" =>
template match {
case template"{ ..$stats } with ..$ctorcalls { $param => ..$body }" => {
val expr = paramss.flatten.map(p => q"${p.name.toString}").zip(paramss.flatten.map{
case param"..$mods $paramname: $atpeopt = $expropt" => paramname
}).map{case (q"$paramName", paramTree) => {
q"${Term.Name(paramName.toString)} -> ${Term.Name(paramTree.toString)}"
}} val resultMap = q"Map(..$expr)" val newBody = body :+ q"""def toMap: Map[String, Any] = $resultMap"""
val newTemplate = template"{ ..$stats } with ..$ctorcalls { $param => ..$newBody }" q"..$mods class $tname[..$tparams] (...$paramss) extends $newTemplate"
}
}
case _ => throw new Exception("@Mappable can be annotation of class only")
}
}
}

可以用下面的数据进行测试:

   @Mappable
case class Car(color: String, model: String, year: Int, owner: String){
def turnOnRadio = {
"playing"
}
} val newCarMap = Car("Silver", "Ford", , "John Doe").toMap
println(newCarMap)

在下面这个例子里示范了如何使用注释参数:

 import scala.util.Try
@compileTimeOnly("@RetryOnFailure not expanded")
class RetryOnFailure(repeat: Int) extends scala.annotation.StaticAnnotation {
inline def apply(defn: Any): Any = meta {
defn match {
case q"..$mods def $name[..$tparams](...$paramss): $tpeopt = $expr" => {
val q"new $_(${arg})" = this
val repeats = Try(arg.toString.toInt).getOrElse(abort(s"Retry on failure takes number as parameter")) val newCode =
q"""..$mods def $name[..$tparams](...$paramss): $tpeopt = {
import scala.util.Try for( a <- to $repeats){
val res = Try($expr)
if(res.isSuccess){
return res.get
}
} throw new Exception("Method fails after "+$repeats + " repeats")
}
"""
newCode
}
case _ => abort("@RetryOnFailure can be annotation of method only")
}
}
}

具体使用方法如下:

 object utils {
def methodThrowingException(random: Int): Unit = {
if(random% == ){
throw new Exception(s"throwing exception for ${random}")
}
}
}
import scala.util.Random
@RetryOnFailure() def failMethod[String](): Unit = {
val random = Random.nextInt()
println("Retrying...")
utils.methodThrowingException(random)
}

顺便也把上次的那个TalkingAnimal重新再写一下:

 class TalkingAnimal(voice: String) extends StaticAnnotation {
inline def apply(defn: Any): Any = meta {
defn match {
case q"..$mods class $tname[..$tparams] (...$paramss) extends $template" =>
template match {
case template"{ ..$stats } with ..$ctorcalls { $param => ..$body }" => {
val q"new $_(${arg})" = this
val sound = arg.toString()
val animalType = tname.toString()
val newBody = body :+
q""" def sayHello: Unit =
println("Hello, I'm a " + $animalType +
" and my name is " + name + " " + $sound+ "...")
"""
val newTemplate =template"{ ..$stats } with ..$ctorcalls { $param => ..$newBody }"
q"..$mods class $tname[..$tparams] (...$paramss) extends $newTemplate"
}
}
case _ => abort("Error: expanding TalkingAnimal!")
}
}
}

对比旧款Def Macros可以发现quasiquote的语法还是有变化的,比如拆分class定义就需要先拆出template。Scalameta重新定义了新的quasiquote,另外注释对象参数的运算方法也有所不同,这是因为Scalameta的AST新设计的表达结构。

测试运算如下:

   trait Animal {
val name: String
}
@TalkingAnimal("wangwang")
case class Dog(val name: String) extends Animal @TalkingAnimal("miaomiao")
case class Cat(val name: String) extends Animal //@TalkingAnimal("")
//case class Carrot(val name: String)
//Error:(12,2) Annotation TalkingAnimal only apply to Animal inherited! @TalingAnimal
Dog("Goldy").sayHello
Cat("Kitty").sayHello

下面是本次讨论中的完整示范源代码:

注释实现源代码:

 import scala.meta._
class Greetings extends scala.annotation.StaticAnnotation {
inline def apply(defn: Any): Any = meta {
defn match {
case q"object $name {..$stats}" => {
q"""
object $name {
def sayHello(msg: String): Unit = println("Hello," + msg)
..$stats
}
"""
}
case q"object $name extends $parent {..$stats}" => {
q"""
object $name extends $parent {
def sayHello(msg: String): Unit = println("Hello," + msg)
..$stats
}
"""
}
case _ => abort("annottee must be object!")
}
}
} class Benchmark extends scala.annotation.StaticAnnotation {
inline def apply(defn: Any): Any = meta {
defn match {
case q"..$mod def $name[..$tparams](...$args): $rtpe = $body" =>
q"""
..$mod def $name[..$tparams](...$args): $rtpe = {
val start = System.nanoTime()
val result = $body
val end = System.nanoTime()
println(${name.toString} + " elapsed time = " + (end - start) + "ns")
result
}
"""
case _ => abort("Fail to expand annotation Benchmark!")
}
}
} import scala.meta.Ctor.Call
class main extends scala.annotation.StaticAnnotation {
inline def apply(defn: Any): Any = meta {
def abortIfObjectAlreadyExtendsApp(ctorcalls: scala.collection.immutable.Seq[Call], objectName: Term) = {
val extendsAppAlready = ctorcalls.map(_.structure).contains(ctor"App()".structure)
if (extendsAppAlready){
abort(s"$objectName already extends App")
}
}
defn match {
case q"..$mods object $name extends $template" => template match {
case template"{ ..$stats1 } with ..$ctorcalls { $param => ..$stats2 }" =>
abortIfObjectAlreadyExtendsApp(ctorcalls, name)
val mainMethod = q"def main(args: Array[String]): Unit = { ..$stats2 }"
val newTemplate = template"{ ..$stats1 } with ..$ctorcalls { $param => $mainMethod }" q"..$mods object $name extends $newTemplate"
}
case _ => abort("@main can be annotation of object only")
}
}
}
import scala.annotation.{StaticAnnotation, compileTimeOnly}
@compileTimeOnly("@Mappable not expanded")
class Mappable extends StaticAnnotation {
inline def apply(defn: Any): Any = meta {
defn match {
case q"..$mods class $tname[..$tparams] (...$paramss) extends $template" =>
template match {
case template"{ ..$stats } with ..$ctorcalls { $param => ..$body }" => {
val expr = paramss.flatten.map(p => q"${p.name.toString}").zip(paramss.flatten.map{
case param"..$mods $paramname: $atpeopt = $expropt" => paramname
}).map{case (q"$paramName", paramTree) => {
q"${Term.Name(paramName.toString)} -> ${Term.Name(paramTree.toString)}"
}} val resultMap = q"Map(..$expr)" val newBody = body :+ q"""def toMap: Map[String, Any] = $resultMap"""
val newTemplate = template"{ ..$stats } with ..$ctorcalls { $param => ..$newBody }" q"..$mods class $tname[..$tparams] (...$paramss) extends $newTemplate"
}
}
case _ => throw new Exception("@Mappable can be annotation of class only")
}
}
}
import scala.util.Try
@compileTimeOnly("@RetryOnFailure not expanded")
class RetryOnFailure(repeat: Int) extends scala.annotation.StaticAnnotation {
inline def apply(defn: Any): Any = meta {
defn match {
case q"..$mods def $name[..$tparams](...$paramss): $tpeopt = $expr" => {
val q"new $_(${arg})" = this
val repeats = Try(arg.toString.toInt).getOrElse(abort(s"Retry on failure takes number as parameter")) val newCode =
q"""..$mods def $name[..$tparams](...$paramss): $tpeopt = {
import scala.util.Try for( a <- to $repeats){
val res = Try($expr)
if(res.isSuccess){
return res.get
}
} throw new Exception("Method fails after "+$repeats + " repeats")
}
"""
newCode
}
case _ => abort("@RetryOnFailure can be annotation of method only")
}
}
} class TalkingAnimal(voice: String) extends StaticAnnotation {
inline def apply(defn: Any): Any = meta {
defn match {
case q"..$mods class $tname[..$tparams] (...$paramss) extends $template" =>
template match {
case template"{ ..$stats } with ..$ctorcalls { $param => ..$body }" => {
val q"new $_(${arg})" = this
val sound = arg.toString()
val animalType = tname.toString()
val newBody = body :+
q""" def sayHello: Unit =
println("Hello, I'm a " + $animalType +
" and my name is " + name + " " + $sound+ "...")
"""
val newTemplate =template"{ ..$stats } with ..$ctorcalls { $param => ..$newBody }"
q"..$mods class $tname[..$tparams] (...$paramss) extends $newTemplate"
}
}
case _ => abort("Error: expanding TalkingAnimal!")
}
}
}

试运行代码:

 object MacroAnnotDemo extends App {
@Greetings object Greet {
def add(x: Int, y: Int) = println(x + y)
}
@Greetings object Hi extends AnyRef {} Greet.sayHello("John")
Greet.add(,)
Hi.sayHello("Susana Wang") @Benchmark
def calcPow(x: Double, y: Double) = {
val z = x + y
math.pow(z,z)
} println(calcPow(4.2, 8.9)) @Mappable
case class Car(color: String, model: String, year: Int, owner: String){
def turnOnRadio = {
"playing"
}
} val newCarMap = Car("Silver", "Ford", , "John Doe").toMap
println(newCarMap) object utils {
def methodThrowingException(random: Int): Unit = {
if(random% == ){
throw new Exception(s"throwing exception for ${random}")
}
}
}
import scala.util.Random
@RetryOnFailure() def failMethod[String](): Unit = {
val random = Random.nextInt()
println("Retrying...")
utils.methodThrowingException(random)
} trait Animal {
val name: String
}
@TalkingAnimal("wangwang")
case class Dog(val name: String) extends Animal @TalkingAnimal("miaomiao")
case class Cat(val name: String) extends Animal //@TalkingAnimal("")
//case class Carrot(val name: String)
//Error:(12,2) Annotation TalkingAnimal only apply to Animal inherited! @TalingAnimal
Dog("Goldy").sayHello
Cat("Kitty").sayHello }

Scala Macros - scalamela 1.x,inline-meta annotations的更多相关文章

  1. Scala Macros - 元编程 Metaprogramming with Def Macros

    Scala Macros对scala函数库编程人员来说是一项不可或缺的编程工具,可以通过它来解决一些用普通编程或者类层次编程(type level programming)都无法解决的问题,这是因为S ...

  2. 【Effective C++ 读书笔记】条款02: 尽量以 const, enum, inline 替换 #define

    条款02: 尽量以 const, enum, inline 替换 #define 这个条款或许可以改为“宁可以编译器替换预处理器”. 编译过程: .c文件--预处理-->.i文件--编译--&g ...

  3. Effective C++ 之 Item 2:尽量以 const, enum, inline 替换 #define

    Effective C++ Chapter 1. 让自己习惯C++(Accustoming Yourself to C++) Item 2. 尽量以 const, enum, inline 替换 #d ...

  4. block,inline和inline-block概念和区别

    总体概念 block和inline这两个概念是简略的说法,完整确切的说应该是 block-level elements (块级元素) 和 inline elements (内联元素).block元素通 ...

  5. scala中的面向对象定义类,构造函数,继承

    我们知道scala中一切皆为对象,函数也是对象,数字也是对象,它是一个比java还要面向对象的语言. 定义scala的简单类 class Point (val x:Int, val y:Int) 上面 ...

  6. block,inline和inlinke-block细节对比

    block,inline和inlinke-block细节对比 display:block block元素会独占一行,多个block元素会各自新起一行.默认情况下,block元素宽度自动填满其父元素宽度 ...

  7. block,inline和inline-block概念和区别(转)

    转自  http://www.cnblogs.com/KeithWang/p/3139517.html 总体概念 block和inline这两个概念是简略的说法,完整确切的说应该是 block-lev ...

  8. block,inline,inline-block的区别

    block: 英语翻译过来是“块”意思,就跟小时候玩过的积木方块一样,一块一块往上搭. inline: 英语翻译过来就是“内联”的意思,内联不好理解,我的理解就是行内元素: block和inline都 ...

  9. EC笔记,第一部分:2.尽量以const,enum,inline代替#define

    02.尽量以const,enum,inline代替#define 原因:编译前的预处理会替换宏,所以调试的时候找不到错误 1.const 尽量用const替代常量宏定义 两种特殊情况: (1).常量指 ...

随机推荐

  1. 获取 dhcp IP 过程分析 - 每天5分钟玩转 OpenStack(91)

    前面我们已经讨论了 DHCP agent 的配置以及 namespace 如何隔离 dnsmasq 服务,本节将以 cirros-vm1 为例分析获取 DHCP IP 的详细过程. 在创建 insta ...

  2. npm源切换

    版权声明:欢迎转载,请附加转载来源:一路博客(http://www.16boke.com)   目录(?)[+] 安装 使用 列出可选的源 切换 增加源 删除源 测试速度 许可 项目主页   我们介绍 ...

  3. Linux基础介绍【第五篇】

    linux权限位 Linux文件或目录的权限位是由9个权限位来控制,每三位为一组,它们分别是文件属主权限.属组权限.其他用户权限. r:read可读权限,对应数字4: w:write可写权限,对应数字 ...

  4. Linux C语言解析并显示.bmp格式图片

    /************************* *bmp.h文件 *************************/ #ifndef __BMP_H__ #define __BMP_H__ # ...

  5. 【微信SEO】公众号也能做排名?

    [写于2016年8月] 最近,微信团队发出一则公告,开放公众号运营者一年内更改公众号名一次,这对不少名字起的奇葩名字(包括dkplus)的公众号来说是一件好事. 为什么说是好事呢?公众号名字直接关联到 ...

  6. Merge Sorted Array

    Given two sorted integer arrays nums1 and nums2, merge nums2 into nums1 as one sorted array. Note:Yo ...

  7. 通过几个Hello World感受.NET Core全新的开发体验

    2016年6月27日,这是一个特殊的日子,微软全新的.NET开发平台.NET Core的RTM版本正式发布.我个人将.NET Core的核心特性归结为三点,它们的首字母组成一个非常好记的简称——COM ...

  8. 【腾讯优测干货分享】安卓专项测试之GPU测试探索

    本文来自于Dev Club 开发者社区,非经作者同意,请勿转载,原文地址:http://dev.qq.com/topic/57c7ffdc0569a1191bce8a63 作者:章未哲——腾讯SNG质 ...

  9. 【JavaScript吉光片羽】遭遇IE8

    最初对做兼容性的认知只停留在UI层面,但其实UI层面都还好,因为毕竟你可以直接看得见现象,更为重要的是在JavaScript层面,因为这个部分涉及到功能性,前者最多是体验性的问题.下面扯一下这几天遇到 ...

  10. 【css3笔记】---- 渐变的秘密

    <CSS揭秘>这本书非常不错,充满了干货和惊喜.以下主要是关于使用渐变做出来的一些效果的笔记.请用最新的现代浏览器观看. 首先要回顾下一个css语句: linear-gradient([ ...