Повторное использование аннотаций на основе макросов Scala

Рассмотрим аннотацию на основе макросов Scala, такую ​​как @ memoise из macmemo. Для аннотации требуются два аргумента: максимальный размер кеша и время жизни, например,

@memoize(maxSize = 20000, expiresAfter = 2 hours)

Допустим, вы хотите создать аннотацию @cacheall, эквивалентную @memoize(maxSize = Int.MaxValue, expiresAfter = 100 days), чтобы уменьшить шаблон и иметь единую точку параметризации.

Есть ли стандартный шаблон для такого типа повторного использования? Очевидно,

class cacheall extends memoize(Int.MaxValue, 100 days)

не будет работать из-за синтаксического анализа аргументов во время компиляции в макросе.


person Sim    schedule 11.07.2019    source источник


Ответы (1)


Стандартный шаблон - сделать вашу аннотацию аннотацией макроса, которая при раскрытии включает необходимую аннотацию с необходимыми параметрами.

import scala.annotation.StaticAnnotation
import scala.language.experimental.macros
import scala.reflect.macros.blackbox

class cacheall extends StaticAnnotation {
  def macroTransform(annottees: Any*): Any = macro cacheallMacro.impl
}

object cacheallMacro {
  def impl(c: blackbox.Context)(annottees: c.Tree*): c.Tree = {
    import c.universe._

    val memoize = q"""
      new _root_.com.softwaremill.macmemo.memoize(
        _root_.scala.Int.MaxValue, {
          import _root_.scala.concurrent.duration._
          100.days 
      })"""

    annottees match {
      case q"${mods: Modifiers} def $tname[..$tparams](...$paramss): $tpt = $expr" :: _ =>
        q"${mods.mapAnnotations(memoize :: _)} def $tname[..$tparams](...$paramss): $tpt = $expr"
    }
  }
}
person Dmytro Mitin    schedule 11.07.2019