由于这个答案:https ://stackoverflow.com/a/10708026/694597 ,我想知道在使用通用控制器时如何返回特定控制器固有的视图。
1 回答
当您在控制器操作中渲染视图时,您只需调用模板引擎生成的普通函数:
public Application extends Controller {
public static Result index() {
return ok(views.html.index.render(42));
}
}
在这里,是具有类型render
的对象的方法。index
Template1<Integer, Html>
现在的问题是:如何编写能够调用特定于另一个控制器的视图的通用控制器?或者简单地说:如何抽象视图?
我看到了两种解决方案:控制反转和反射。
让我们看看如何在一个简单的用例上实现这两者。假设您有以下通用Shower<T>
类能够计算包含任何类型值的 HTML 表示的 HTTP 响应T
:
public class Shower<T> {
public Result show(T value) {
// TODO return an HTML representation of `value`
}
}
控制反转
要Shower<T>
使用控制反转来实现,我们只需要注入Template1<T, Html>
用于执行渲染的值:
public class Shower<T> {
public final Template1<T, Html> template;
public Shower(Template1<T, Html> template) {
this.template = template;
}
public Result show(T value) {
return ok(template.render(value));
}
}
要在控制器中使用它,请创建一个静态实例Shower<T>
并将其注入要使用的模板:
public class Application extends Controller {
public static Shower<Foo> foo = new Shower<Foo>(views.html.Foo.show.ref());
}
反射
您可能会发现它太样板而不必显式地注入要用于 的每个实例的模板Shower<T>
,因此您可能很想根据命名约定通过反射来检索它,例如,要显示 type 的值Foo
,只需查找一个对象show
在包中命名views.html.Foo
:
public class Shower<T> {
private final Class<T> clazz;
public Shower(Class<T> clazz) {
this.clazz = clazz;
}
public Result show(T value) throws Exception {
Class<?> object = Play.application().classLoader().loadClass("views.html." + clazz.getSimpleName() + ".show$");
Template1<T, Html> template = (Template1<T, Html>)object.getField("MODULE$").get(null);
return ok(template.render(value));
}
}
(这是使用反射访问 Scala 对象的方式)
您可以在控制器中按如下方式使用它:
public class Application extends Controller {
public static Shower<Foo> foo = new Shower<Foo>(Foo.class);
}
优点和缺点
基于反射的解决方案在调用站点上需要较少的样板,但它依赖于命名约定的事实使其更加脆弱。此外,此解决方案只会在运行时失败时才会失败,而第一个解决方案会在编译时向您显示您丢失的模板。最后但同样重要的是,基于反射的解决方案可能会由于反射而增加一些性能开销。