首页 > 学院 > 开发设计 > 正文

在Eclipse 3.1体验J2SE 5.0新特性四(图)

2019-11-18 13:01:25
字体:
来源:转载
供稿:网友

  3.Documented:这个注释(Annotation)将作为public API的一部分。
  
  4.Inherited : 假设注释(Annotation)定义的时候使用了Inherited,那么假如这个注释(Annotation)修饰某个class,这个类的子类也被这个注释(Annotation)所修饰。
  
  2.3注释的应用
  
  下面各小节显示了在哪些情况下可以使用注释以及如何使用注释。
  
  2.3.1动态查找注释
  
  当我们定义好了注释以后,我们可以开发一些分析工具来解释这些注释。这里通常要用到java的反射特性。比如说我们希望找到某个对象/方法/域使用了哪些注释,或者获得某个特定的注释,或者判定是否使用某个特定的注释, 我们可以参考下面这个例子。
  
  这个例子中定义了两个注释:TODO和TOFORMATE。在MyCalculator类中,TODO用来修饰方法calculateRate,而TOFORMATE用来修饰类变量concurrency和debitDate。而在类TestCalculator的main函数中,通过Java反射特性,我们查找到使用这些注释的类变量和方法。清单12-清单15分别显示这些类的定义。
  
  清单12 TODO注释的定义
  
  @Target({ElementType.METHOD})
  @Retention(RetentionPolicy.RUNTIME)
  
  public @interface TODO {
  int PRiority() default 0;
  }
  
  清单13 TOFORMATE的定义
  
  @Target({ElementType.FIELD})
  @Retention(RetentionPolicy.RUNTIME)
  
  public @interface TOFORMATE {
  
  }
  
  清单14 使用注释的类MyCalculator
  
  public class MyCalculator {
  boolean isReady;
  @TOFORMATE double concurrency;
  @TOFORMATE Date debitDate;
  public MyCalculator() {
  super();
  }
  
  @TODO
  public void calculateRate(){
  System.out.println("Calculating...");
  }
  }
  
  清单15动态查找注释
  
  public class TestCalculator {
  public static void main(String[] args) {
  MyCalculator cal = new MyCalculator();
  cal.calculateRate();
  try {
  Class c = cal.getClass();
  Method[] methods = c.getDeclaredMethods();
  
  for (Method m: methods) {
  // 判定这个方法有没有使用TODO
  if (m.isAnnotationPresent(TODO.class))
  System.out.println("Method "+m.getName()+": the TODO is present");
  }
  
  Field[] fields = c.getDeclaredFields();
  for (Field f : fields) {
  // 判定这个域有没有使用TOFORMATE
  if (f.isAnnotationPresent(TOFORMATE.class))
  System.out.println
  ("Field "+f.getName()+": the TOFORMATE is present");
  }
  } catch (Exception exc) {
  exc.printStackTrace();
  }
  }
  }
  
  下面我们来运行这个例子,这个例子的运行结果如图10所示。
  
  运行结果和我们先前的定义是一致的。在运行时,我们可以获得注释使用的相关信息。
  
 在Eclipse 3.1体验J2SE 5.0新特性四(图)(图一)

  
图6 运行结果

  
  在我们介绍了什么是注释以后,你可能会想知道注释可以应用到什么地方呢?使用注释有什么好处呢?在下面的小节中我们将介绍一个稍复杂的例子。从这个例子中,你将体会到注释所以提供的强大的描述机制(declarative programming)。
  
  2.3.2 使用注释替代Visitor模式
  
  在J2SE 5.0以前,我们在设计应用的时候,我们经常会使用Visitor这个设计模式。Visitor这个模式一般是用于为我们已经设计好了一组类添加方法,而不需要担心改变定义好的类。比如说我们已经定义了好了一组类结构,但是我们希望将这些类的对象部分数据输出到某种格式的文件中。
  
  Vistor模式的实现
  
  使用Vistor模式,首先我们在Employee这个类中加入eXPort方法,export方法如图7所示。Export方法接受Exporter对象作为参数,并在方法体中调用exporter对象的visit()方法。
  
在Eclipse 3.1体验J2SE 5.0新特性四(图)(图二)

  
图7 使用Vistor模式实现格式输出

  
  在这里我们定义了一个Exporter抽象类,我们可以通过继续Exporter类,重写其visit方法来实现不同格式的文件输出。
  
  图7种给出visit方法的实现是一个简单的例子。假如要实现输出成xml格式的,可以定义Exporter子类:XMLExporter。假如希望输出成文本的可以定义TXTExporter。但是这样做不够灵活的地方在于,假如Employee加入其他的域变量,那么相应的visitor类也需要进行修改。这就违反了面向对象Open for Extension, close for Modification的原则。
  
  使用注释替代Vistor模式
  
  使用注释(Annotation),也可以完成数据输出的功能。首先定义一个新的注释类型:@Exportable。然后定义一个抽象的解释器ExportableGenerator,将Employee 对象传入解释器。
  
  在解释器中,查找哪些域使用了Exportable这个注释(Annotation),将这些域(Field)按照一定格式输出。图12给出了Exportable注释的定义。
  
  清单16注释Exportable的定义
  
  @Target({ElementType.FIELD})
  @Retention(RetentionPolicy.RUNTIME)
  @Inherited
  public @interface Exportable {
  }
  清单17-清单20中给出了包含数据的这些类的定义以及这些类是如何使用注释Exportable的。 图18定义了Main函数,使用ExporterGenerator来产生输出文件。清单21给出了使用注释来实现这一功能的两个类:ExporterGenerator和TXTExporterGenerator。
  
  其中ExporterGenerator定义了一个基本的框架。而TXTExporterGenerator继续了ExporterGenerator,并且重写了outputField方法,在这个方法中实现了特定格式的输出。用户可以继续这个ExporterGenerator,并且实现其中的抽象方法来定义自己期望的格式。
  
  清单17 Employee的类定义
  
  public abstract class Employee {
  public abstract String getName();
  public abstract String getEmpNo();
  public Employee() {
  super();
  }
  }
  
  清单18 Regular的类定义
  
  public class Regular extends Employee{
  @Exportable String name;
  @Exportable String address;
  @Exportable String title;
  @Exportable String phone;
  @Exportable String location;
  @Exportable Date onboardDate;
  @Exportable ArrayList team;
  String empNo;
  
  public Regular(String name, String address, String title, String phone,
  String location, Date date) {
  super();
  this.name = name;
  this.address = address;
  this.title = title;
  this.phone = phone;
  this.location = location;
  onboardDate = date;
  team = new ArrayList();
  }
  
  public void addMemeber(Employee e){
  team.add(e);
  }
  
  @Override
  public String getName() {
  // TODO Auto-generated method stub
  return name;
  }
  }

发表评论 共有条评论
用户名: 密码:
验证码: 匿名发表