请原谅这个问题的长度。
我经常需要在代码的一层创建一些上下文信息,并在其他地方使用这些信息。我通常发现自己使用隐式参数:
def foo(params)(implicit cx: MyContextType) = ...
implicit val context = makeContext()
foo(params)
这可行,但需要大量传递隐式参数,在干预函数布局后污染层的方法签名,即使他们自己不关心它。
def foo(params)(implicit cx: MyContextType) = ... bar() ...
def bar(params)(implicit cx: MyContextType) = ... qux() ...
def qux(params)(implicit cx: MyContextType) = ... ged() ...
def ged(params)(implicit cx: MyContextType) = ... mog() ...
def mog(params)(implicit cx: MyContextType) = cx.doStuff(params)
implicit val context = makeContext()
foo(params)
我觉得这种方法很难看,但它确实有一个优点:它是类型安全的。我确定mog
会收到正确类型的上下文对象,否则它不会编译。
如果我可以使用某种形式的“依赖注入”来定位相关上下文,它将减轻混乱。引号表明这与 Scala 中常见的依赖注入模式不同。
起点foo
和终点mog
可能存在于系统的不同层次。例如,foo
可能是用户登录控制器,并且mog
可能正在执行 SQL 访问。可能有很多用户同时登录,但 SQL 层只有一个实例。每次mog
由不同的用户调用,都需要不同的上下文。所以上下文不能被烘焙到接收对象中,你也不想以任何方式合并两个层(如蛋糕模式)。我也不想依赖像 Guice 或 Spring 这样的 DI/IoC 库。我发现它们很重,不太适合 Scala。
所以我认为我需要的是可以mog
在运行时为它检索正确的上下文对象的东西,有点像ThreadLocal
里面有一个堆栈:
def foo(params) = ...bar()...
def bar(params) = ...qux()...
def qux(params) = ...ged()...
def ged(params) = ...mog()...
def mog(params) = { val cx = retrieveContext(); cx.doStuff(params) }
val context = makeContext()
usingContext(context) { foo(params) }
但是一旦异步参与者参与到链中的任何地方,这种情况就会下降。你使用哪个actor库无关紧要,如果代码在不同的线程上运行,那么它会丢失ThreadLocal
.
那么......我错过了一个技巧吗?一种在 Scala 中上下文传递信息的方法,它不会污染干预方法签名,不会静态地将上下文烘焙到接收器中,并且仍然是类型安全的?