如何在 Play Framework 2.3 Java 项目中的 Scala 模板中使用有界泛型参数?
我目前有类似的东西:
@(entities: List[_ <: Entity], currentEntity: Entity)
<ul>
@for(entity <- entities) {
@if(currentEntity.equals(entity)) {
<li><strong>@entity</strong></li>
} else {
<li>@entity</li>
}
}
</ul>
但是,我可以在
entities
中使用不同类型的实体来调用它。和 currentEntity
- 这不好。我想做类似的事情:@[T <: Entity](entities: List[T], currentEntity: T)
...
但这给了我
Invalid '@' symbol
作为编译错误。
最佳答案
正如@m-z 指出的那样,它(尚)不受支持。但是您可能能够通过首先将参数编码到 View 对象中来获得您想要的类型安全(以另一个类家族为代价):
case class HighlightedListView[E <: Entity](entities:List[E], currentEntity:E)
现在在你的 Controller 中,加载一个新的
HighlightedListView
实例而不是直接将参数提供给模板: def foo = Action {
...
// Assuming some SubEntity exists, the compiler will enforce the typing:
val hlv = HighlightedListView[SubEntity](entities, currentEntity)
Ok(html.mytemplate(hlv))
}
正如评论所述,如果您的类型不对齐,编译器将拒绝。那么模板的类型可能会非常松散,因为我们知道我们是安全的:
@(hlv:HighlightedListView[_])
<ul>
@for(entity <- hlv.entities) {
@if(hlv.currentEntity.equals(entity)) {
<li><strong>@entity</strong></li>
} else {
<li>@entity</li>
}
}
</ul>
您甚至可以利用您的新
View
对象添加辅助方法,这可以使模板更易于阅读,并促进单元测试:case class HighlightedListView[E <: Entity](entities:List[E], currentEntity:E) {
def shouldHighlight(e:Any):Boolean = currentEntity.equals(e)
}
导致:
@if(hlv.shouldHighlight(entity)) {
<li><strong>@entity</strong></li>
} else {
<li>@entity</li>
}
关于scala - Play 框架模板中的有界泛型参数,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/29615655/