gpt4 book ai didi

scala - 字符串插值和宏 : how to get the StringContext and expression locations

转载 作者:行者123 更新时间:2023-12-03 22:32:59 25 4
gpt4 key购买 nike

我正在尝试使用宏实现自定义字符串插值方法,我需要一些有关使用 API 的指导。

这是我想做的事情:

/** expected
* LocatedPieces(List(("\nHello ", Place("world"), Position()),
("\nHow are you, ", Name("Eric"), Position(...)))
*/
val locatedPieces: LocatedPieces =
s2"""
Hello $place

How are you, $name
"""

val place: Piece = Place("world")
val name: Piece = Name("Eric")

trait Piece
case class Place(p: String) extends Piece
case class Name(n: String) extends Piece

/** sequence of each interpolated Piece object with:
* the preceding text and its location
*/
case class LocatedPieces(located: Seq[(String, Piece, Position)])

implicit class s2pieces(sc: StringContext) {
def s2(parts: Piece*) = macro s2Impl
}

def impl(c: Context)(pieces: c.Expr[Piece]*): c.Expr[LocatedPieces] = {
// I want to build a LocatedPieces object with the positions for all
// the pieces + the pieces + the (sc: StringContext).parts
// with the method createLocatedPieces below
// ???
}

def createLocatedPieces(parts: Seq[String], pieces: Seq[Piece], positions: Seq[Position]):
LocatedPieces =
// zip the text parts, pieces and positions together to create a LocatedPieces object
???

我的问题是:
  • 如何访问 StringContext宏内的对象,以获取所有 StringContext.parts字符串?
  • 我怎样才能捕获每件的位置?
  • 如何调用createLocatedPieces上面的方法并具体化结果以获得宏调用的结果?
  • 最佳答案

    经过几个小时的努力,我找到了一个可运行的解决方案:

    object Macros {

    import scala.reflect.macros.Context
    import language.experimental.macros

    sealed trait Piece
    case class Place(str: String) extends Piece
    case class Name(str: String) extends Piece
    case class Pos(column: Int, line: Int)
    case class LocatedPieces(located: List[(String, Piece, Pos)])

    implicit class s2pieces(sc: StringContext) {
    def s2(pieces: Piece*) = macro s2impl
    }

    // pieces contain all the Piece instances passed inside of the string interpolation
    def s2impl(c: Context)(pieces: c.Expr[Piece]*): c.Expr[LocatedPieces] = {
    import c.universe.{ Name => _, _ }

    c.prefix.tree match {
    // access data of string interpolation
    case Apply(_, List(Apply(_, rawParts))) =>

    // helper methods
    def typeIdent[A : TypeTag] =
    Ident(typeTag[A].tpe.typeSymbol)

    def companionIdent[A : TypeTag] =
    Ident(typeTag[A].tpe.typeSymbol.companionSymbol)

    def identFromString(tpt: String) =
    Ident(c.mirror.staticModule(tpt))

    // We need to translate the data calculated inside of the macro to an AST
    // in order to write it back to the compiler.
    def toAST(any: Any) =
    Literal(Constant(any))

    def toPosAST(column: Tree, line: Tree) =
    Apply(
    Select(companionIdent[Pos], newTermName("apply")),
    List(column, line))

    def toTupleAST(t1: Tree, t2: Tree, t3: Tree) =
    Apply(
    TypeApply(
    Select(identFromString("scala.Tuple3"), newTermName("apply")),
    List(typeIdent[String], typeIdent[Piece], typeIdent[Pos])),
    List(t1, t2, t3))

    def toLocatedPiecesAST(located: Tree) =
    Apply(
    Select(companionIdent[LocatedPieces], newTermName("apply")),
    List(located))

    def toListAST(xs: List[Tree]) =
    Apply(
    TypeApply(
    Select(identFromString("scala.collection.immutable.List"), newTermName("apply")),
    List(AppliedTypeTree(
    typeIdent[Tuple3[String, Piece, Pos]],
    List(typeIdent[String], typeIdent[Piece], typeIdent[Pos])))),
    xs)

    // `parts` contain the strings a string interpolation is built of
    val parts = rawParts map { case Literal(Constant(const: String)) => const }
    // translate compiler positions to a data structure that can live outside of the compiler
    val positions = pieces.toList map (_.tree.pos) map (p => Pos(p.column, p.line))
    // discard last element of parts, `transpose` does not work otherwise
    // trim parts to discard unnecessary white space
    val data = List(parts.init map (_.trim), pieces.toList, positions).transpose
    // create an AST containing a List[(String, Piece, Pos)]
    val tupleAST = data map { case List(part: String, piece: c.Expr[_], Pos(column, line)) =>
    toTupleAST(toAST(part), piece.tree, toPosAST(toAST(column), toAST(line)))
    }
    // create an AST of `LocatedPieces`
    val locatedPiecesAST = toLocatedPiecesAST(toListAST(tupleAST))
    c.Expr(locatedPiecesAST)

    case _ =>
    c.abort(c.enclosingPosition, "invalid")
    }
    }
    }

    用法:
    object StringContextTest {
    val place: Piece = Place("world")
    val name: Piece = Name("Eric")
    val pieces = s2"""
    Hello $place
    How are you, $name?
    """
    pieces.located foreach println
    }

    结果:
    (Hello,Place(world),Pos(12,9))
    (How are you,,Name(Eric),Pos(19,10))

    我没想到要花这么多时间才能把所有东西放在一起,但那是一段有趣的时光。我希望代码符合您的要求。如果您需要有关特定事物如何运作的更多信息,请查看其他问题及其对 SO 的回答:
  • Information on how ASTs are constructed
  • How to work with TypeTag
  • How to use reify in the REPL to get information about the AST

  • 非常感谢 Travis Brown(见评论),我得到了一个更短的编译解决方案:
    object Macros {

    import scala.reflect.macros.Context
    import language.experimental.macros

    sealed trait Piece
    case class Place(str: String) extends Piece
    case class Name(str: String) extends Piece
    case class Pos(column: Int, line: Int)
    case class LocatedPieces(located: Seq[(String, Piece, Pos)])

    implicit class s2pieces(sc: StringContext) {
    def s2(pieces: Piece*) = macro s2impl
    }

    def s2impl(c: Context)(pieces: c.Expr[Piece]*): c.Expr[LocatedPieces] = {
    import c.universe.{ Name => _, _ }

    def toAST[A : TypeTag](xs: Tree*): Tree =
    Apply(
    Select(Ident(typeOf[A].typeSymbol.companionSymbol), newTermName("apply")),
    xs.toList)

    val parts = c.prefix.tree match {
    case Apply(_, List(Apply(_, rawParts))) =>
    rawParts zip (pieces map (_.tree)) map {
    case (Literal(Constant(rawPart: String)), piece) =>
    val line = c.literal(piece.pos.line).tree
    val column = c.literal(piece.pos.column).tree
    val part = c.literal(rawPart.trim).tree
    toAST[(_, _, _)](part, piece, toAST[Pos](line, column))
    }
    }
    c.Expr(toAST[LocatedPieces](toAST[Seq[_]](parts: _*)))
    }
    }

    它对冗长的 AST 结构进行了抽象,其逻辑略有不同但几乎相同。如果您在理解代码的工作原理上有困难,请先尝试理解第一个解决方案。它的作用更加明确。

    关于scala - 字符串插值和宏 : how to get the StringContext and expression locations,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/15329027/

    25 4 0
    Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
    广告合作:1813099741@qq.com 6ren.com