NVIDIA 初创加速计划,免费加速您的创业启动 了解详情
写点什么

John Heintz 谈如何向 Java 注解添加行为

  • 2008-08-15
  • 本文字数:1622 字

    阅读完需:约 5 分钟

将常见可重用的行为以定制注解的方式添加到 Java 应用中是一种非常棒的方式,但如果开发者在决定使用注解的时候能够遵循一些开发标准和最佳实践的话,那就更事半功倍了。比如说在计划使用注释之前,比较一下当前情况下是不是注释真的比内迁代码和接口更为贴切。New Aspects 的 John Heintz 在最近举行的 No Fluff Just Stuff( NFJS )俄亥俄州软件中心研讨会( COSS )上作了一个演讲,讨论了向Java 注解添加行为的多种设计技术。

讲中,John 简要地谈到了字节码和元编程技术,大体介绍了Java 应用中的元数据是如何在早期Java 版本中使用属性文件、XML 到现在逐渐发展到注解上来的过程。他也提到了制定 JSR-175 规范的目的在于为 Java 语言提供元数据工具,比如 Java 5 中就已经添加了注解。

定义注解与定义接口的方式相似(使用 @Interface 类型来声明注解)。注解不仅可以用在包级别上,也可以用在类成员(属性、方法、构造方法),甚至还可以用在方法的局部参数上。

保持策略(retention policy)是实现注解的重要一环,根据数据在内存中保持时间的长短,有三种类型的保持策略:

  • Runtime(一直保留数据,可以通过反射来访问)
  • Class(数据保留在字节码中,无法在运行时访问)
  • Source(编译器不保留该数据)

一些流行框架如 EJB3 Hibernate Spring Seam Struts 2 RIFE JAX-WS 的新版本都开始支持注解。John 论述了处理注解的三种方式:

生成器:
这种注解处理方式是通过读取源代码来产生新的源代码,或修改现存源代码及其它一些文件(XML、文档等等)来实现。生成器主要依赖于容器或其它编码约定,可以在任何保持策略下工作。

使用生成器的例子有注解处理工具(Annotation Processing Tool,即 APT )和处理器、 XDoclet 、Spoon(针对 Java 的扩展编译器)、 APT-Jelly (一个模板库)等。APT 不允许你修改源代码,但对产生辅助文件却有相当大的帮助(像 WSDL、文档)。

字节码转换:
注解处理器对带有注解的类文件进行解析,然后对类做一些设当的修改。此外,他们也能生成其他类型的文件(比如 XML 配置文件)。字节码转换器在离线状态(编译期)、在装载期都能运行,还能在运行时动态运行(使用 JVMTI API)。在 class 或 runtime 这样的保持策略下,它都能运行。

使用字节码转换器的例子有 AspectJ 、Spring、Hibernate、 CGLib Javassist ASM BCEL 等。

运行时反射:
这种方式使用反射 API 以编程的方式在运行阶段检查对象。它主要依赖于容器或其他编码约定,同时也需要 runtime 保持策略。

使用运行时反射的例子有:Java 5 及更新 Java 版本中的反射、 Commons Attributes 。 像 JUnit TestNG 这样的测试框架也使用运行时反射来处理注解。

John 采用一个 Java 示范应用向大家展示了如何结合 APT、Javassist、AspectJ 及反射(使用了 Java 动态代理)等多种设计技术来定制注解。他还展示了如何向类、属性和方法中添加行为,也谈到了如何使用 Aspects 来注解类型和方法。

此外,John 还提到了 Aspect Processing 的好处,如容易实现(需要好工具的支持)、语义细致——可以影响到方法调用,甚至是属性的访问(在这点上,和反射和 APT 不同),还能够集成多个类库的注解,并支持特定的领域抽象。他建议开发人员说,如果 Java 代码可以以接口的方式实现,那就用接口,不要去用注解。他还建议大家不要对所有东西都使用注解,因为 POJO(Plain Old Java Object)要比 HAJO(Heavily Annotated Java Object)好。在演讲中,他列举了其它一些最佳实践准则,比如:

  • 注解要添加在最高层次的抽象上(比如在类和方法这两个层次上,注解要尽量添加在类上面)
  • 在面对规则的时候,尽量使用合适的默认值,而且只对 exception 添加注解。比外,还要尽量少用参数。
  • 将注解作为“规约优于配置”法则的补充(如 Spring)
  • 使用反射来增强现有框架(如测试)
  • 利用 Javassist 和 CGLIB 框架来产生新的对象,对于底层来说,可以提高字节码转换效率
  • 使用 APT 来生成非代码文件

查看英文原文: John Heintz on Adding Behavior to Java Annotations

2008-08-15 13:591403
用户头像

发布了 88 篇内容, 共 258.6 次阅读, 收获喜欢 8 次。

关注

评论

发布
暂无评论
发现更多内容

Druid 加载 Kafka 数据后查询和清理数据

HoneyMoose

Go并发之CSP并发模型、协程并发

Regan Yue

协程 CSP Go 语言 8月日更

番外2. OpenCV 中摄像头捕获与视频处理与常见问题解决方案

梦想橡皮擦

8月日更

kubernetes入门:harbor个人私有仓库傻瓜式搭建

小鲍侃java

8月日更

JavaScript代码片段学设计模式

devpoint

设计模式 工厂模式 8月日更

【设计模式】桥接模式

Andy阿辉

编程 后端 设计模式 8月日更

Vue进阶(十四):config/index.js 配置文件详解

No Silver Bullet

Vue 8月日更 配置文件

Druid 使用 Kafka 数据加载教程——下载和启动 Kafka

HoneyMoose

Vue进阶(十三):MOCK

No Silver Bullet

Vue Mock 8月日更

软通动力 OpenHarmony 师资培训班隆重开启,聚焦高校教师赋能

开放原子开源基金会

oeasy教您玩转vim - 8 - # 追加文本

o

SpringCloud微服务日志经kafka缓冲写入到ELK

Rubble

kafka Spring Cloud ELK 8月日更

超全激活函数学习总结!!!

Shirakawa

神经网络 机器学习 算法 激活函数

oeasy教您玩转vim - 9 - # 换行插入

o

docker编排参数详解(docker-compose.yml配置文件编写)

xcbeyond

Docker 容器 8月日更

在线邮箱地址提取工具

入门小站

工具

模块4作业

青鸟飞鱼

oeasy教您玩转vim - 7 - # 从头插入

o

🏆【Java技术之旅】「编译器专题」重塑认识Java编译器的执行过程(消除数组边界检查+公共子表达式)!

洛神灬殇

Java 编译器 8月日更 JIT编译器

两个小女孩

箭上有毒

8月日更

架构训练营 模块4作业

sophiahuxh

Druid 使用 Kafka 将数据载入到 Kafka

HoneyMoose

Linux中buff-cache占用过高解决方案

入门小站

Linux

元数据管理服务分析报告

漫长的白日梦

数据湖 AWS 元数据

如何从 0 到 1 设计 B 端产品?

蒋川

后台开发 产品开发 后台 后台管理系统 tob产品

netty系列之:Event、Handler和Pipeline

程序那些事

Java Netty nio 程序那些事

spring的循环依赖

卢卡多多

spring aop 8月日更

Druid 加载 Kafka 数据时通过控制台来提交一个 supervisor

HoneyMoose

数据探索

南湖职业技术学院机械狗

Druid 加载 Kafka 数据时直接提交一个 supervisor

HoneyMoose

【Flutter 专题】75 图解基本 TabBar 标签导航栏 (一)

阿策小和尚

Flutter 小菜 0 基础学习 Flutter Android 小菜鸟 8月日更

John Heintz谈如何向Java注解添加行为_Java_Srini Penchikala_InfoQ精选文章