From 348c8e55f89498ee89a1c140d4db06761e2d74ec Mon Sep 17 00:00:00 2001 From: Jules Ivanic Date: Sat, 21 Oct 2023 19:40:37 +0400 Subject: [PATCH] Prefer mutable data structure to reduce memory usage in macros (#2932) * Prefer mutable data structure for speed * Prefer mutable data structure to reduce memory usage in macros (#2933) * Prefer mutable data structure to reduce memory usage in macros * Prefer mutable data structure to reduce memory usage in macros * Prefer mutable data structure to reduce memory usage in macros * Prefer mutable data structure to reduce memory usage in macros * Prefer ListBuffer --- .../io/getquill/idiom/ReifyStatement.scala | 119 +++++++++--------- .../scala/io/getquill/idiom/Statement.scala | 30 ++--- .../scala/io/getquill/util/Interleave.scala | 11 +- 3 files changed, 84 insertions(+), 76 deletions(-) diff --git a/src/main/scala/io/getquill/idiom/ReifyStatement.scala b/src/main/scala/io/getquill/idiom/ReifyStatement.scala index c088af6dca..32b6b11c8a 100644 --- a/src/main/scala/io/getquill/idiom/ReifyStatement.scala +++ b/src/main/scala/io/getquill/idiom/ReifyStatement.scala @@ -1,10 +1,11 @@ package io.getquill.idiom import io.getquill.ast._ -import io.getquill.util.Interleave import io.getquill.idiom.StatementInterpolator._ +import io.getquill.util.Interleave import scala.annotation.tailrec +import scala.collection.mutable.ListBuffer object ReifyStatement { @@ -15,10 +16,9 @@ object ReifyStatement { forProbing: Boolean ): (String, List[External]) = { val expanded = - forProbing match { - case true => statement - case false => expandLiftings(statement, emptySetContainsToken) - } + if (forProbing) statement + else expandLiftings(statement, emptySetContainsToken) + token2string(expanded, liftingPlaceholder) } @@ -26,49 +26,52 @@ object ReifyStatement { @tailrec def apply( workList: List[Token], - sqlResult: Seq[String], - liftingResult: Seq[External], + sqlResult: ListBuffer[String], + liftingResult: ListBuffer[External], liftingSize: Int - ): (String, List[External]) = workList match { - case Nil => sqlResult.reverse.mkString("") -> liftingResult.reverse.toList - case head :: tail => - head match { - case StringToken(s2) => apply(tail, s2 +: sqlResult, liftingResult, liftingSize) - case SetContainsToken(a, op, b) => apply(stmt"$a $op ($b)" +: tail, sqlResult, liftingResult, liftingSize) - case ScalarLiftToken(lift) => - apply(tail, liftingPlaceholder(liftingSize) +: sqlResult, lift +: liftingResult, liftingSize + 1) - case ScalarTagToken(tag) => - apply(tail, liftingPlaceholder(liftingSize) +: sqlResult, tag +: liftingResult, liftingSize + 1) - case Statement(tokens) => apply(tokens.foldRight(tail)(_ +: _), sqlResult, liftingResult, liftingSize) - case ValuesClauseToken(stmt) => apply(stmt +: tail, sqlResult, liftingResult, liftingSize) - case _: QuotationTagToken => - throw new UnsupportedOperationException("Quotation Tags must be resolved before a reification.") - } - } + ): (String, List[External]) = + workList match { + case Nil => sqlResult.mkString("") -> liftingResult.toList + case head :: tail => + head match { + case StringToken(s2) => apply(tail, sqlResult += s2, liftingResult, liftingSize) + case SetContainsToken(a, op, b) => apply(stmt"$a $op ($b)" +: tail, sqlResult, liftingResult, liftingSize) + case ScalarLiftToken(lift) => + apply(tail, sqlResult += liftingPlaceholder(liftingSize), liftingResult += lift, liftingSize + 1) + case ScalarTagToken(tag) => + apply(tail, sqlResult += liftingPlaceholder(liftingSize), liftingResult += tag, liftingSize + 1) + case Statement(tokens) => apply(tokens.foldRight(tail)(_ +: _), sqlResult, liftingResult, liftingSize) + case ValuesClauseToken(stmt) => apply(stmt +: tail, sqlResult, liftingResult, liftingSize) + case _: QuotationTagToken => + throw new UnsupportedOperationException("Quotation Tags must be resolved before a reification.") + } + } - apply(List(token), Seq(), Seq(), 0) + apply(List(token), ListBuffer.empty, ListBuffer.empty, 0) } - private def expandLiftings(statement: Statement, emptySetContainsToken: Token => Token) = + private def expandLiftings(statement: Statement, emptySetContainsToken: Token => Token): Statement = Statement { - statement.tokens.foldLeft(List.empty[Token]) { - case (tokens, SetContainsToken(a, op, ScalarLiftToken(lift: ScalarQueryLift))) => - lift.value.asInstanceOf[Iterable[Any]].toList match { - case Nil => tokens :+ emptySetContainsToken(a) - case values => - val liftings = values.map(v => - ScalarLiftToken(ScalarValueLift(lift.name, External.Source.Parser, v, lift.encoder, lift.quat)) - ) - val separators = List.fill(liftings.size - 1)(StringToken(", ")) - (tokens :+ stmt"$a $op (") ++ Interleave(liftings, separators) :+ StringToken(")") - } - case (tokens, token) => - tokens :+ token - } + statement.tokens + .foldLeft(List.empty[Token]) { + case (tokens, SetContainsToken(a, op, ScalarLiftToken(lift: ScalarQueryLift))) => + lift.value.asInstanceOf[Iterable[Any]].toList match { + case Nil => tokens :+ emptySetContainsToken(a) + case values => + val liftings = values.map(v => + ScalarLiftToken(ScalarValueLift(lift.name, External.Source.Parser, v, lift.encoder, lift.quat)) + ) + val separators = List.fill(liftings.size - 1)(StringToken(", ")) + (tokens :+ stmt"$a $op (") ++ Interleave(liftings, separators) :+ StringToken(")") + } + case (tokens, token) => + tokens :+ token + } } } object ReifyStatementWithInjectables { + import Tokens._ def apply[T]( liftingPlaceholder: Int => String, @@ -79,10 +82,9 @@ object ReifyStatementWithInjectables { injectables: List[(String, T => ScalarLift)] ): (String, List[External]) = { val expanded = - forProbing match { - case true => statement - case false => expandLiftings(statement, emptySetContainsToken, subBatch, injectables.toMap) - } + if (forProbing) statement + else expandLiftings(statement, emptySetContainsToken, subBatch, injectables.toMap) + val (query, externals) = token2string(expanded, liftingPlaceholder) (query, externals) } @@ -91,19 +93,19 @@ object ReifyStatementWithInjectables { @tailrec def apply( workList: List[Token], - sqlResult: Seq[String], - liftingResult: Seq[External], + sqlResult: ListBuffer[String], + liftingResult: ListBuffer[External], liftingSize: Int ): (String, List[External]) = workList match { - case Nil => sqlResult.reverse.mkString("") -> liftingResult.reverse.toList + case Nil => sqlResult.mkString("") -> liftingResult.toList case head :: tail => head match { - case StringToken(s2) => apply(tail, s2 +: sqlResult, liftingResult, liftingSize) + case StringToken(s2) => apply(tail, sqlResult += s2, liftingResult, liftingSize) case SetContainsToken(a, op, b) => apply(stmt"$a $op ($b)" +: tail, sqlResult, liftingResult, liftingSize) case ScalarLiftToken(lift) => - apply(tail, liftingPlaceholder(liftingSize) +: sqlResult, lift +: liftingResult, liftingSize + 1) + apply(tail, sqlResult += liftingPlaceholder(liftingSize), liftingResult += lift, liftingSize + 1) case ScalarTagToken(tag) => - apply(tail, liftingPlaceholder(liftingSize) +: sqlResult, tag +: liftingResult, liftingSize + 1) + apply(tail, sqlResult += liftingPlaceholder(liftingSize), liftingResult += tag, liftingSize + 1) case Statement(tokens) => apply(tokens.foldRight(tail)(_ +: _), sqlResult, liftingResult, liftingSize) case ValuesClauseToken(stmt) => apply(stmt +: tail, sqlResult, liftingResult, liftingSize) case _: QuotationTagToken => @@ -111,7 +113,7 @@ object ReifyStatementWithInjectables { } } - apply(List(token), Seq(), Seq(), 0) + apply(List(token), ListBuffer.empty, ListBuffer.empty, 0) } private def expandLiftings[T]( @@ -119,9 +121,9 @@ object ReifyStatementWithInjectables { emptySetContainsToken: Token => Token, subBatch: List[T], injectables: collection.Map[String, T => ScalarLift] - ) = { + ): Statement = { - def resolveInjectableValue(v: ScalarTagToken, value: T) = { + def resolveInjectableValue(v: ScalarTagToken, value: T): ScalarLiftToken = { val injectable = // Look up the right uuid:String to get the right for ((p:Person) => ScalarLift(p.)) injectables.get(v.tag.uid) match { @@ -170,8 +172,8 @@ object ReifyStatementWithInjectables { } case (tokens, valuesClause: ValuesClauseToken) => val pluggedClauses = subBatch.map(value => plugScalarTags(valuesClause, value)) - val separators = List.fill(pluggedClauses.size - 1)(StringToken(", ")) - (tokens ++ Interleave(pluggedClauses.toList, separators)) + val separators = List.fill(pluggedClauses.size - 1)(`, `) + tokens ++ Interleave(pluggedClauses, separators) case (tokens, SetContainsToken(a, op, ScalarLiftToken(lift: ScalarQueryLift))) => lift.value.asInstanceOf[Iterable[Any]].toList match { case Nil => tokens :+ emptySetContainsToken(a) @@ -179,8 +181,8 @@ object ReifyStatementWithInjectables { val liftings = values.map(v => ScalarLiftToken(ScalarValueLift(lift.name, External.Source.Parser, v, lift.encoder, lift.quat)) ) - val separators = List.fill(liftings.size - 1)(StringToken(", ")) - (tokens :+ stmt"$a $op (") ++ Interleave(liftings, separators) :+ StringToken(")") + val separators = List.fill(liftings.size - 1)(`, `) + (tokens :+ stmt"$a $op (") ++ Interleave(liftings, separators) :+ `)` } case (tokens, token) => tokens :+ token @@ -188,3 +190,8 @@ object ReifyStatementWithInjectables { } } } + +private[idiom] object Tokens { + val `, ` : StringToken = StringToken(", ") + val `)` : StringToken = StringToken(")") +} diff --git a/src/main/scala/io/getquill/idiom/Statement.scala b/src/main/scala/io/getquill/idiom/Statement.scala index fa4b573e29..08f50508e7 100644 --- a/src/main/scala/io/getquill/idiom/Statement.scala +++ b/src/main/scala/io/getquill/idiom/Statement.scala @@ -2,33 +2,33 @@ package io.getquill.idiom import io.getquill.ast._ -sealed trait Token +sealed trait Token extends Product with Serializable sealed trait TagToken extends Token -case class StringToken(string: String) extends Token { - override def toString = string +final case class StringToken(string: String) extends Token { + override def toString: String = string } -case class ScalarTagToken(tag: ScalarTag) extends TagToken { - override def toString = s"lift(${tag.uid})" +final case class ScalarTagToken(tag: ScalarTag) extends TagToken { + override def toString: String = s"lift(${tag.uid})" } -case class QuotationTagToken(tag: QuotationTag) extends TagToken { - override def toString = s"quoted(${tag.uid})" +final case class QuotationTagToken(tag: QuotationTag) extends TagToken { + override def toString: String = s"quoted(${tag.uid})" } -case class ScalarLiftToken(lift: ScalarLift) extends Token { - override def toString = s"lift(${lift.name})" +final case class ScalarLiftToken(lift: ScalarLift) extends Token { + override def toString: String = s"lift(${lift.name})" } -case class ValuesClauseToken(statement: Statement) extends Token { - override def toString = statement.toString +final case class ValuesClauseToken(statement: Statement) extends Token { + override def toString: String = statement.toString } -case class Statement(tokens: List[Token]) extends Token { - override def toString = tokens.mkString +final case class Statement(tokens: List[Token]) extends Token { + override def toString: String = tokens.mkString } -case class SetContainsToken(a: Token, op: Token, b: Token) extends Token { - override def toString = s"${a.toString} ${op.toString} (${b.toString})" +final case class SetContainsToken(a: Token, op: Token, b: Token) extends Token { + override def toString: String = s"${a.toString} ${op.toString} (${b.toString})" } diff --git a/src/main/scala/io/getquill/util/Interleave.scala b/src/main/scala/io/getquill/util/Interleave.scala index 14f0d8d328..4d329bad6b 100644 --- a/src/main/scala/io/getquill/util/Interleave.scala +++ b/src/main/scala/io/getquill/util/Interleave.scala @@ -1,17 +1,18 @@ package io.getquill.util import scala.annotation.tailrec +import scala.collection.mutable.ListBuffer object Interleave { def apply[T](l1: List[T], l2: List[T]): List[T] = - interleave(l1, l2, List.empty) + interleave(l1, l2, ListBuffer.empty) @tailrec - private[this] def interleave[T](l1: List[T], l2: List[T], acc: List[T]): List[T] = + private[this] def interleave[T](l1: List[T], l2: List[T], acc: ListBuffer[T]): List[T] = (l1, l2) match { - case (Nil, l2) => acc.reverse ++ l2 - case (l1, Nil) => acc.reverse ++ l1 - case (h1 :: t1, h2 :: t2) => interleave(t1, t2, h2 +: h1 +: acc) + case (Nil, l2) => (acc ++ l2).toList + case (l1, Nil) => (acc ++ l1).toList + case (h1 :: t1, h2 :: t2) => interleave(t1, t2, { acc += h1; acc += h2 }) } }