Я читал о том, как делать инъекцию зависимостей в scala с помощью шаблона торта. Думаю, я это понимаю, но я, должно быть, что-то пропустил, потому что я до сих пор не вижу смысла в этом! Почему предпочтительнее объявлять зависимости через собственные типы, а не только абстрактные поля?
Учитывая пример в Программирование Scala TwitterClientComponent
объявляет зависимости, подобные этому, с использованием шаблона cake:
//other trait declarations elided for clarity
...
trait TwitterClientComponent {
self: TwitterClientUIComponent with
TwitterLocalCacheComponent with
TwitterServiceComponent =>
val client: TwitterClient
class TwitterClient(val user: TwitterUserProfile) extends Tweeter {
def tweet(msg: String) = {
val twt = new Tweet(user, msg, new Date)
if (service.sendTweet(twt)) {
localCache.saveTweet(twt)
ui.showTweet(twt)
}
}
}
}
Как это лучше, чем объявлять зависимости как абстрактные поля, как показано ниже?
trait TwitterClient(val user: TwitterUserProfile) extends Tweeter {
//abstract fields instead of cake pattern self types
val service: TwitterService
val localCache: TwitterLocalCache
val ui: TwitterClientUI
def tweet(msg: String) = {
val twt = new Tweet(user, msg, new Date)
if (service.sendTweet(twt)) {
localCache.saveTweet(twt)
ui.showTweet(twt)
}
}
}
Глядя на время инстанцирования, которое происходит, когда на самом деле происходит DI (как я понимаю), я изо всех сил стараюсь увидеть преимущества пирога, особенно когда вы рассматриваете дополнительную клавиатуру, которую вы должны использовать для заявлений на торты (охватывающая черта )
//Please note, I have stripped out some implementation details from the
//referenced example to clarify the injection of implemented dependencies
//Cake dependencies injected:
trait TextClient
extends TwitterClientComponent
with TwitterClientUIComponent
with TwitterLocalCacheComponent
with TwitterServiceComponent {
// Dependency from TwitterClientComponent:
val client = new TwitterClient
// Dependency from TwitterClientUIComponent:
val ui = new TwitterClientUI
// Dependency from TwitterLocalCacheComponent:
val localCache = new TwitterLocalCache
// Dependency from TwitterServiceComponent
val service = new TwitterService
}
Теперь снова с абстрактными полями, более или менее одинаковыми!:
trait TextClient {
//first of all no need to mixin the components
// Dependency on TwitterClient:
val client = new TwitterClient
// Dependency on TwitterClientUI:
val ui = new TwitterClientUI
// Dependency on TwitterLocalCache:
val localCache = new TwitterLocalCache
// Dependency on TwitterService
val service = new TwitterService
}
Я уверен, что я должен что-то пропустить о превосходстве торта! Однако на данный момент я не вижу, что он предлагает, чтобы объявить зависимости каким-либо другим способом (конструктор, абстрактные поля).