在本教程的第一部分中,我们介绍了创建,发布和调用模块的基本知识。 我们创建的模块并没有真正做很多事情,因此现在是时候使用Play的某些功能来扩展功能了。

1.拦截器

拦截器使您可以拦截对控制器的调用,并增强或阻止其行为。 在第一个示例应用程序中,我们向MyLogger添加了显式调用,以将消息记录到控制台。 如果我们扩大规模,并且您想在每个控制器方法调用中都使用这个非常有用的插件,那么您将编写大量样板代码。 拦截器使我们能够自动应用动作,从而减少样板。

1.1添加代码

在app目录中,创建一个名为action的新程序包。 在这里,我们将添加LogMe注释,以及将在存在注释时执行的LogMeAction。

此时,LogMe.java是一个非常简单的注释,不带任何参数

package actions;import play.mvc.With;import java.lang.annotation.Documented;
import java.lang.annotation.ElementType;
import java.lang.annotation.Inherited;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;/*** @author Steve Chaloner (steve@objectify.be)*/
@With(LogMeAction.class)
@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.METHOD, ElementType.TYPE})
@Inherited
@Documented
public @interface LogMe
{
}

看一下注释,然后您将看到(LogInAction.class)–这让Play知道遇到此注释时,它应该在实际目标之前执行LogInAction。

package actions;import play.mvc.Action;
import play.mvc.Http;
import play.mvc.Result;/*** @author Steve Chaloner (steve@objectify.be)*/
public class LogMeAction extends Action{@Overridepublic Result call(Http.Context context) throws Throwable{System.out.println("MyLogger: " + context.request().path());return delegate.call(context);}
}

这是非常优雅的东西–动作具有LogMe的通用参数类型,该参数类型可以访问提供给LogMe批注的任何参数。 这使您可以自定义动作的行为。 当我们添加一些额外的功能时,我们将看到这一点。 一旦完成您的代码(在本例中为System.out的另一个类),然后您将返回proxy.class(context)的结果以恢复正常的执行流程。 同时,如果将@LogMe添加到控制器方法,则操作的路径将记录到控制台; 如果将@LogMe添加到控制器,则对该控制器中任何方法的调用都会导致该路径记录到控制台。

1.2更新Build.scala

由于我们拥有新版本的mylogger,因此我们应该更改版本号。 打开项目/Build.scala并进行更改

val appVersion      = "1.0-SNAPSHOT"

val appVersion      = "1.1"

1.3确保已检测到您的项目更改

如果您已经在mylogger / project-code中运行Play控制台,则需要执行“重新加载”以获取对Build.scala的更改。 如果未打开控制台,请立即将其打开-更改将在启动时自动进行。

[mylogger] $ reload
[info] Loading project definition from C:\Temp\mylogger\project-code\project
[info] Set current project to mylogger (in build file:/C:/Temp/mylogger/project-code/)

1.4清理并发布

如前所述,在发布之前进行清理始终是一个好主意,以确保您不会推出任何不应存在的对象。

[mylogger] $ clean
[success] Total time: 0 s, completed Mar 19, 2012 9:17:25 PM
[mylogger] $ publish-local
[info] Packaging /tmp/mylogger/project-code/target/scala-2.9.1/mylogger_2.9.1-
1.1 -sources.jar ...
[info] Done packaging.
[info] Wrote /tmp/mylogger/project-code/target/scala-2.9.1/mylogger_2.9.1- 1.1
 .pom
[info] Updating {file:/tmp/mylogger/project-code/}mylogger...
[info] Done updating.
[info] :: delivering :: mylogger#mylogger_2.9.1;1.1 :: 1.1 :: release ::
Mon Mar 19 21:17:30 CET 2012
[info] Generating API documentation for main sources...
[info] Compiling 3 Java sources to /tmp/mylogger/project-code/target/scala-2.9.1
/classes...
[info]  delivering ivy file to /tmp/mylogger/project-code/target/scala-2.9.1
/ivy- 1.1 .xml
model contains 7 documentable templates
[info] API documentation generation successful.
[info] Packaging /tmp/mylogger/project-code/target/scala-2.9.1/mylogger_2.9.1- 1.1
 -javadoc.jar ...
[info] Done packaging.
[info] Packaging /tmp/mylogger/project-code/target/scala-2.9.1/mylogger_2.9.1- 1.1 .jar ...
[info] Done packaging.
[info]  published mylogger_2.9.1 to /home/steve/development/play/play-2.0/framework
/../repository/local/mylogger/mylogger_2.9.1/ 1.1 /poms/mylogger_2.9.1.pom
[info]  published mylogger_2.9.1 to /home/steve/development/play/play-2.0/framework
/../repository/local/mylogger/mylogger_2.9.1/ 1.1 /jars/mylogger_2.9.1.jar
[info]  published mylogger_2.9.1 to /home/steve/development/play/play-2.0/framework
/../repository/local/mylogger/mylogger_2.9.1/ 1.1 /srcs/mylogger_2.9.1-sources.jar
[info]  published mylogger_2.9.1 to /home/steve/development/play/play-2.0/framework
/../repository/local/mylogger/mylogger_2.9.1/ 1.1 /docs/mylogger_2.9.1-javadoc.jar
[info]  published ivy to /home/steve/development/play/play-2.0/framework/../repository
/local/mylogger/mylogger_2.9.1/ 1.1 /ivys/ivy.xml
[success] Total time: 3 s, completed Mar 19, 2012 9:17:31 PM

请注意,模块的版本在日志记录中已更改。 如果仍然看到1.0-SNAPSHOT,请确保在发布之前重新加载了项目!

1.5更新示例应用程序

返回示例应用程序,在project / Build.scala中更改所需的模块版本

val appDependencies = Seq("mylogger" % "mylogger_2.9.1" % "1.1")

重新加载并运行“依赖项”以确保您具有正确的版本。 现在,您可以更新app / controllers / Application.java以使用以下新代码:

package controllers;import actions.LogMe;
import play.mvc.Controller;
import play.mvc.Result;
import views.html.index;@LogMe
public class Application extends Controller
{public static Result index(){return ok(index.render("Your new application is ready."));}
}

运行此示例,现在您将看到通过注释应用的MyLogger输出。

2.添加了拦截器参数

仅记录请求的路径并不是特别有用或令人兴奋。 如果应该为每种控制器或控制器方法提供特定的日志消息怎么办? 在这种情况下,我们需要添加一些参数。

2.1更改注释签名

上载action / LogMe.java以使用value()参数-这是默认的注释参数,因此在使用时无需显式命名。 该值默认为空字符串,因此如果此处不存在此消息,则可以在操作中提供标准消息。

public @interface LogMe
{String value() default "";
}

在操作中,将继承的配置字段键入通用参数(在本例中为LogMe),并提供对参数的访问。 更新call(Http.Context)方法以利用此方法。

public Result call(Http.Context context) throws Throwable
{String value = configuration.value();if (value == null || value.isEmpty()){value = context.request().path();}System.out.println("MyLogger: " + value);return delegate.call(context);
}

2.2发布更改

再次重复步骤1.2至1.4,这次将appVersion更改为1.2

2.3更新示例应用程序

像以前一样,在Build.scala中更新依赖项版本,重新加载并使用“ dependencies”进行确认。 现在,您可以向LogMe批注添加一条消息:

@LogMe("This is my log message")
public class Application extends Controller

运行该应用程序,现在您将在控制台中看到注释消息。

[info] play - Application started (Dev)
MyLogger: This is my log message

3.使拦截器交互

现在,您(希望)掌握了这个要诀,我们将加快一点。 在本节中,我们将研究拦截器如何相互交互。 Play首先将拦截器应用于方法,然后应用于控制器,因此,如果在方法和控制器级别都存在相同的注释,它将执行两次。 LogMe批注可以同时应用于类级别和方法级别,但是,如果您对整个控制器有一条通用的日志记录消息,但一种方法需要使用另一条消息,那该怎么办呢? 另外,我们只希望每次调用一个日志消息。 为此,我们可以使用传递给每个动作的上下文。

3.1更新模块

更新LogMeAction以使其了解以前的调用:

package actions;import play.mvc.Action;
import play.mvc.Http;
import play.mvc.Result;/*** @author Steve Chaloner (steve@objectify.be)*/
public class LogMeAction extends Action{public static final String ALREADY_LOGGED = "already-logged";@Overridepublic Result call(Http.Context context) throws Throwable{Result result;if (context.args.containsKey(ALREADY_LOGGED)){// skip the logging, just continue the executionresult = delegate.call(context);}else{// we're not using the value here, only the key, but this// mechanism can also be used to pass objectscontext.args.put(ALREADY_LOGGED, "");String value = configuration.value();if (value == null || value.isEmpty()){value = context.request().path();}System.out.println("MyLogger: " + value);result = delegate.call(context);}return result;}
}

更新版本号,清理,重新加载和本地发布。

3.2更新示例应用程序

这次,我们将在index方法中添加第二个注释。 这将覆盖控制器级别的注释。 因此,更新Build.scala中的依赖项编号,重新加载并运行。

package controllers;import actions.LogMe;
import play.mvc.Controller;
import play.mvc.Result;
import views.html.index;@LogMe("This is my log message")
public class Application extends Controller
{@LogMe("This is my method-specific log message")public static Result index(){return ok(index.render("Your new application is ready."));}
}

当您访问http:// localhost:9000时,现在您将在控制台中看到以下内容:

@LogMe("This is my log message")
[info] play - Application started (Dev)
MyLogger: This is my method-specific log message

4.又是啤酒时间

您现在拥有了支持参数化操作的基础结构。 请记住,很多东西都可以作为注释参数传递,但是-重要的是-并非所有东西都可以传递。 您可能需要为某些任务发挥创造力!

您可以在此处下载完整的源代码。

参考: Play 2的编写模块,第2部分: Objective博客上JCG合作伙伴 Steve Chaloner的拦截器 。

翻译自: https://www.javacodegeeks.com/2012/04/writing-modules-for-play-2-part-2.html

编写Play 2的模块,第2部分:拦截器相关推荐

  1. 谷粒商城笔记+踩坑(19)——订单模块构建、登录拦截器

    导航: 谷粒商城笔记+踩坑汇总篇 目录 1.页面环境搭建 1.1 动静分离 1.2 hosts添加域名映射 1.3 配置网关和nacos 1.4 引导类开启注册发现和feign客户端 1.5 thym ...

  2. play 拦截器_编写Play 2的模块,第2部分:拦截器

    play 拦截器 在本教程的第一部分中,我们介绍了创建,发布和调用模块的基本知识. 我们创建的模块并没有真正做很多事情,因此现在是时候使用Play的某些功能来扩展功能了. 1.拦截器 拦截器使您可以拦 ...

  3. 编写Play 2的模块,第1部分:使工作正常

    几周前,我迁移了Play! 框架 1.x版本的Deadbolt应用于Play 2平台,并且对缺少有关创建模块的信息感到惊讶. Play 1.x文档中详细介绍了该主题,这使得创建模块非常简单. 显然,需 ...

  4. (Xposed)编写第一个Xposed模块

    编写第一个Xposed模块 1.创建项目 在创建项目时,选择一个有MainActivity的项目,要是注册空项目还要自己去注册入口activity 2.添加Xposed提供的jar到lib目录 3.修 ...

  5. python反编译luac_Lua程序逆向之为Luac编写IDA Pro处理器模块

    上一篇讲解了如何加载一个Luac文件到IDA Pro当中,加载进入idb数据库的内容犹如切好洗净的食材,并不能粗暴的直接展示给用户,还需要IDA Pro中的处理器模块对内容进行下一步的反汇编渲染与指令 ...

  6. boost::fusion模块实现一个函数绑定器的测试程序

    boost::fusion模块实现一个函数绑定器的测试程序 实现功能 C++实现代码 实现功能 boost::fusion模块实现一个函数绑定器的测试程序 C++实现代码 #include <b ...

  7. Flume编写拦截器

    前提 已经安装了Kafka Flume Zookeeper 数据准备 事件日志有时间戳 启动日志是json数据 /tmp/logs/app-2020-10-15.log {"action&q ...

  8. 【Flume】(四)IDEA编写自定义拦截器

    IDEA编写自定义拦截器 IDEA中导入以下依赖: <!-- https://mvnrepository.com/artifact/org.apache.flume/flume-ng-core ...

  9. pythonDay06核心编程part2(类装饰器,元类,垃圾回收,内建属性(属性拦截器),内建函数(fliter,map,reduce,sorted),集合,functools,模块进阶)

    我若将死,给孩子留遗言,只留一句话:Repetition is the mother of all learning重复是学习之母.他们将来长大,学知识,技巧.爱情.事业.交流....倘若懂得行动的力 ...

最新文章

  1. ListBox滚动条置底
  2. C++ Primer 5th笔记(chap 16 模板和泛型编程)成员模板
  3. POJ1088 滑雪 记忆化搜索
  4. VisualBox中增大linux硬盘空间的方法
  5. easypoi 导入oracle,记一次由openjdk导致的poi错误(easyexcel)
  6. 远程mysql_java.sql.SQLException: null, message from server: Host 'xxx' is not allowed to connect
  7. 微软雅黑的问题(for silverlight)
  8. 各种文件后缀名与打开方式大全
  9. 1万美元FS-ISAC网络安全奖学金
  10. 迅雷9边下边播绑定第三方播放器
  11. 微信小程序showToast在真机中显示时间不可控制,显示时间短
  12. Windows 10打开蓝牙已关闭解决方法
  13. 瘟神的尾行 -- Rootkit技术发展史 (转载)
  14. 关于修改虚拟机内存,和开启 硬件加速
  15. cmnet和cmwap的访问
  16. DNS域名服务协议和其实现Bind应用
  17. 【智能合约实战】——入门级DAPP,没有想象中那么难
  18. 外贸开发信主题怎么写
  19. java 百度贴吧 爬虫,爬取百度贴吧帖子
  20. 零基础能学大数据开发吗 可以从哪些方面入手

热门文章

  1. 如何创建虚拟硬盘 + os 读取硬盘参数代码
  2. soapui 测试soap_使用SoapUI调用不同的安全WCF SOAP服务-基本身份验证,第二部分
  3. javaserver_什么是JavaServer Faces(JSF)
  4. 使用UriBuilder快速创建URI
  5. nashorn 性能_Nashorn简介
  6. Fluent Design单选按钮,复选框,选择框,Java菜单
  7. 展望Java的未来:空值类型
  8. hibernate 刷新_Hibernate事实:了解刷新操作顺序很重要
  9. 使用LocalDate,LocalTime和LocalDateTime
  10. Java中的异步等待