我目前正在尝试在运行时动态创建Akka Stream图形定义。这个想法是用户将能够以交互方式定义流并将它们附加到现有/正在运行的BroadcastHubs
。这意味着我不知道在编译时将使用哪些流甚至多少流。
不幸的是,我正在努力进行泛型/类型擦除。坦率地说,我甚至不确定我在JVM上尝试做什么。
我有一个函数会返回代表两个连接Flow
的Akka Streams Flows
。它使用Scala的TypeTags
来绕过类型擦除。如果第一个流的输出类型与第二个流的输入类型相同,则可以成功连接。这很好用。
import akka.NotUsed
import akka.stream.FlowShape
import akka.stream.scaladsl.GraphDSL.Implicits._
import akka.stream.scaladsl.{Flow, GraphDSL}
import scala.reflect.runtime.universe._
import scala.util.{Failure, Success, Try}
def connect[A: TypeTag, B: TypeTag, C: TypeTag, D: TypeTag](a: Flow[A, B, NotUsed],
b: Flow[C, D, NotUsed]): Try[Flow[A, D, NotUsed]] = {
Try {
if (typeOf[B] =:= typeOf[C]) {
val c = b.asInstanceOf[Flow[B, D, NotUsed]]
Flow.fromGraph {
GraphDSL.create(a, c)((m1, m2) => NotUsed.getInstance()) { implicit b =>
(s1, s2) =>
s1 ~> s2
FlowShape(s1.in, s2.out)
}
}
}
else
throw new RuntimeException(s"Connection failed. Incompatible types: ${typeOf[B]} and ${typeOf[C]}")
}
}
因此,如果我有Flow[A,B]
和Flow[C,D]
,则假设B和C属于同一类型,结果将为Flow[A,D]
。
我还有尝试将List
Flows
合并/缩减为单Flow
的功能。让我们假设此列表是从文件或Web请求中的流定义列表派生的。
def merge(fcs: List[Flow[_, _, NotUsed]]): Try[Option[Flow[_, _, NotUsed]]] = {
fcs match {
case Nil => Success(None)
case h :: Nil => Success(Some(h))
case h :: t =>
val n = t.head
connect(h, n) match {
case Success(fc) => merge(fc :: t)
case Failure(e) => Failure(e)
}
}
}
不幸的是,由于Flows
存储在List
内,由于标准Lists
上的类型删除,我丢失了所有类型信息,因此无法连接Flows
1}}在运行时。这是一个例子:
def flowIdentity[A]() = Flow.fromFunction[A, A](x => x)
def flowI2S() = Flow.fromFunction[Int, String](_.toString)
val a = flowIdentity[Int]()
val b = flowIdentity[Int]()
val c = flowI2S()
val d = flowIdentity[String]()
val fcs: List[Flow[_, _, NotUsed]] = List(a, b, c, d)
val y = merge(fcs)
这导致例外:
Failure(java.lang.RuntimeException: Connection failed. Incompatible types _$4 and _$3)
我一直在调查Miles Sabin的Shapeless,并认为我可以使用HLists
来保留类型信息。不幸的是,这似乎只有在编译时知道列表的各个类型和长度时才有效。如果我将特定HList
向上转换为HList
,则看起来我再次丢失了类型信息。
val fcs: HList = a :: b :: c :: d :: HNil
所以我的问题是......这甚至可能吗?有没有办法用Shapeless generics魔术做到这一点(最好不需要使用特定的非存在类型提取器)?我想尽可能找到通用的解决方案,任何帮助都会受到赞赏。
谢谢!
答案 0 :(得分:-1)
您已经注意到,它不起作用的原因是该列表会删除您拥有的类型。因此,这是不可能的。 如果您知道所有可用作中间类型的类型,则可以通过添加解析函数来解决。添加这样的功能还将简化您的连接方法。我将添加一个代码段。我希望一切都会清楚。
def flowIdentity[A]() = Flow.fromFunction[A, A](x => x)
def flowI2S() = Flow.fromFunction[Int, String](_.toString)
def main(args: Array[String]): Unit = {
val idInt1 = flowIdentity[Int]()
val idInt2 = flowIdentity[Int]()
val int2String = flowI2S()
val idString = flowIdentity[String]()
val fcs = List(idInt1, idInt2, int2String, idString)
val source = Source(1 to 10)
val mergedGraph = merge(fcs).get.asInstanceOf[Flow[Int, String, NotUsed]]
source.via(mergedGraph).to(Sink.foreach(println)).run()
}
def merge(fcs: List[Flow[_, _, NotUsed]]): Option[Flow[_, _, NotUsed]] = {
fcs match {
case Nil => None
case h :: Nil => Some(h)
case h :: t =>
val n = t.head
val fc = resolveConnect(h, n)
merge(fc :: t.tail)
}
}
def resolveConnect(a: Flow[_, _, NotUsed], b: Flow[_, _, NotUsed]): Flow[_, _, NotUsed] = {
if (a.isInstanceOf[Flow[_, Int, NotUsed]] && b.isInstanceOf[Flow[Int, _, NotUsed]]) {
connectInt(a.asInstanceOf[Flow[_, Int, NotUsed]], b.asInstanceOf[Flow[Int, _, NotUsed]])
} else if (a.isInstanceOf[Flow[_, String, NotUsed]] && b.isInstanceOf[Flow[String, _, NotUsed]]) {
connectString(a.asInstanceOf[Flow[_, String, NotUsed]], b.asInstanceOf[Flow[String, _, NotUsed]])
} else {
throw new UnsupportedOperationException
}
}
def connectInt(a: Flow[_, Int, NotUsed], b: Flow[Int, _, NotUsed]): Flow[_, _, NotUsed] = {
a.via(b)
}
def connectString(a: Flow[_, String, NotUsed], b: Flow[String, _, NotUsed]): Flow[_, _, NotUsed] = {
a.via(b)
}
p.s
那里还隐藏着一个无尽循环的错误。调用合并递归时,应该删除第一个元素,因为它已经合并到主流中。