热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

EffectiveJava(35)--注解优先于命名模式(从零构建JUnit测试类)

在java1.5版本之前的代码中,一般使用命名模式表明哪些程序元素需要通过某种工具或框架进行特殊处理.但是它有严重的缺点–以Junit为例1.由于JUn

在 java1.5 版本之前的代码中 , 一般使用命名模式表明哪些程序元素需要通过某种工具或框架进行特殊处理 . 但是它有严重的缺点 – 以 Junit 为例
1.由于 JUnit 要求测试方法的开头必须为test , 所以类名的文字拼写会导致运行失败 , 但是编译器不会报错或提示
2.无法确保它们只用于相应的程序元素上
3.命名模式没有提供将参数值与程序元素关联起来的方法 例如想要支持只在抛出异常时才会运行成功的测试类
而注解则很好地解决了这些问题
例如我们自定义一个 JUnit 的 Test 注解

//元注解:注解注解的注解
@Retention(RetentionPolicy.RUNTIME) //它注明的注解应该在运行时保留
@Target(ElementType.METHOD) //他注明的注解表明Test注解只在方法声明中才是合法的
public @interface Test {
}

由于Test注解没有参数 , 只是”标注”被注解的元素 , 所以它被称作标记注解 接下来我们测试我们自定义的Test注解 , 在没写测试方法之前 , 我们可以通过上面的解释猜到一下测试方法的运行结果

    public class Sample {
@Test
public static void t1() {
// 运行成功或失败
}
@Test
public static void t2() {
//抛出异常
throw new RuntimeException("BOOM");
}
@Test
public void t3() {
//运行成功或失败
}
@Test
public static void t4() {
//抛出异常
throw new RuntimeException("Crash");
}
}

这是由于Test注解只能被用作无参的静态方法标注
接下来我们完成测试方法 , 检验我们的猜测

public class RunTests {
public static void main(String[] args) {
int tests = 0;
int passed = 0;
try {
Class testClass = Class.forName(args[0]);
for (Method m : testClass.getDeclaredMethods()) {
//isAnnotationPresent告知该工具要运行哪些方法
if (m.isAnnotationPresent(Test.class)) {
tests++;
try {
//反射式的运行所有标注了Test的方法
m.invoke(null);
passed++;
//如果测试方法抛出异常,反射机制就会将他封装在InvocationTargetException并打印报告 如t2 t4
} catch (InvocationTargetException e) {
Throwable exc = e.getCause();
System.out.println(m + "failed: " + exc);
} catch (Exception e) {
System.out.println("Invalid @Test: " + m);
}
}
}
System.out.printf("Passed: %d,Failed: %d%n", passed, tests - passed);
} catch (Exception e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}

控制台输出

    Invalid @Test:public static void Sample.t1()
public static void Sample.t2() failed:RuntimeException:BOOM
public static void Sample.t4() failed:RuntimeException:Crash
passed:1,Failed:3

结果正如我们所料
那么可以不可利用注解忽略异常组 , 使程序在抛出指定异常时依旧执行成功呢 ?让我们来测试一下
注解方法 –

    @Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface ExceptionTest {
//Classextends Exception>某个扩展Exception的类的Class对象 ; value:注解中的方法
Classextends
Exception>[] value();
}
Sample --
public class Sample2 {
@ExceptionTest({ArithmeticException.class,NullPointExcepition.class})
public static void t1() {
int i = 0;
i = i / 0;
}
}
main方法 --
public class RunTests {
public static void main(String[] args) {
int tests = 0;
int passed = 0;
try {
Class testClass = Class.forName(args[0]);
for (Method m : testClass.getDeclaredMethods()) {

//isAnnotationPresent告知该工具要运行哪些方法
if (m.isAnnotationPresent(Test.class)) {
tests++;
try {
//反射式的运行所有标注了Test的方法
m.invoke(null);
System.out.printf("测试 %s 失败:没有注解这个异常%n",m);
//如果测试方法抛出异常,反射机制就会将他封装在InvocationTargetException并打印报告
} catch (InvocationTargetException e) {
//提取注解参数的值 , 并用它检验该测试抛出的异常是否为正确的类型
Throwable exc = e.getCause();
Classextends Exception>[] excTypes = m.getAnnotation(ExceptionTest.class).value();
int oldPassed = passed;
for(Classextends
Exception> excType :excTypes){

if(excType.isInstance(exc)){
passed++;
break;
}
}
if(passed == oldPassed){
System.out.printf("测试%s失败:%s %n",m,exc);
}
} catch (Exception e) {
System.out.println("Invalid @Test: " + m);
}
}
}
System.out.printf("Passed: %d,Failed: %d%n", passed, tests - passed);
} catch (Exception e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}

以上的例子不揭露了注解的冰山一角 , 但它鲜明了表达了一个观点 , 既然有了注解 , 就不必再用命名模式了

总结:除了特定的程序员之外 , 大多数程序员都不必定义注解类型 . 但是所有的程序员都应该使用Java平台所提供的预定义的注解类型 . 还要考虑 IDE 或者静态分析工具所提供的任何注解 . 这种注解可以提升由这些工具所提供的诊断信息的质量 . 但是要注意这些注解还没有表转化 , 因此如果变换工具或者形成标准 , 就需要做更多地工作 .


推荐阅读
  • 本文介绍了Java高并发程序设计中线程安全的概念与synchronized关键字的使用。通过一个计数器的例子,演示了多线程同时对变量进行累加操作时可能出现的问题。最终值会小于预期的原因是因为两个线程同时对变量进行写入时,其中一个线程的结果会覆盖另一个线程的结果。为了解决这个问题,可以使用synchronized关键字来保证线程安全。 ... [详细]
  • Java太阳系小游戏分析和源码详解
    本文介绍了一个基于Java的太阳系小游戏的分析和源码详解。通过对面向对象的知识的学习和实践,作者实现了太阳系各行星绕太阳转的效果。文章详细介绍了游戏的设计思路和源码结构,包括工具类、常量、图片加载、面板等。通过这个小游戏的制作,读者可以巩固和应用所学的知识,如类的继承、方法的重载与重写、多态和封装等。 ... [详细]
  • Iamtryingtomakeaclassthatwillreadatextfileofnamesintoanarray,thenreturnthatarra ... [详细]
  • 本文讨论了一个关于cuowu类的问题,作者在使用cuowu类时遇到了错误提示和使用AdjustmentListener的问题。文章提供了16个解决方案,并给出了两个可能导致错误的原因。 ... [详细]
  • 从零学Java(10)之方法详解,喷打野你真的没我6!
    本文介绍了从零学Java系列中的第10篇文章,详解了Java中的方法。同时讨论了打野过程中喷打野的影响,以及金色打野刀对经济的增加和线上队友经济的影响。指出喷打野会导致线上经济的消减和影响队伍的团结。 ... [详细]
  • [大整数乘法] java代码实现
    本文介绍了使用java代码实现大整数乘法的过程,同时也涉及到大整数加法和大整数减法的计算方法。通过分治算法来提高计算效率,并对算法的时间复杂度进行了研究。详细代码实现请参考文章链接。 ... [详细]
  • Android源码深入理解JNI技术的概述和应用
    本文介绍了Android源码中的JNI技术,包括概述和应用。JNI是Java Native Interface的缩写,是一种技术,可以实现Java程序调用Native语言写的函数,以及Native程序调用Java层的函数。在Android平台上,JNI充当了连接Java世界和Native世界的桥梁。本文通过分析Android源码中的相关文件和位置,深入探讨了JNI技术在Android开发中的重要性和应用场景。 ... [详细]
  • Java学习笔记之面向对象编程(OOP)
    本文介绍了Java学习笔记中的面向对象编程(OOP)内容,包括OOP的三大特性(封装、继承、多态)和五大原则(单一职责原则、开放封闭原则、里式替换原则、依赖倒置原则)。通过学习OOP,可以提高代码复用性、拓展性和安全性。 ... [详细]
  • 本文介绍了iOS数据库Sqlite的SQL语句分类和常见约束关键字。SQL语句分为DDL、DML和DQL三种类型,其中DDL语句用于定义、删除和修改数据表,关键字包括create、drop和alter。常见约束关键字包括if not exists、if exists、primary key、autoincrement、not null和default。此外,还介绍了常见的数据库数据类型,包括integer、text和real。 ... [详细]
  • Android系统源码分析Zygote和SystemServer启动过程详解
    本文详细解析了Android系统源码中Zygote和SystemServer的启动过程。首先介绍了系统framework层启动的内容,帮助理解四大组件的启动和管理过程。接着介绍了AMS、PMS等系统服务的作用和调用方式。然后详细分析了Zygote的启动过程,解释了Zygote在Android启动过程中的决定作用。最后通过时序图展示了整个过程。 ... [详细]
  • 本文介绍了Java中Currency类的getInstance()方法,该方法用于检索给定货币代码的该货币的实例。文章详细解释了方法的语法、参数、返回值和异常,并提供了一个示例程序来说明该方法的工作原理。 ... [详细]
  • Java容器中的compareto方法排序原理解析
    本文从源码解析Java容器中的compareto方法的排序原理,讲解了在使用数组存储数据时的限制以及存储效率的问题。同时提到了Redis的五大数据结构和list、set等知识点,回忆了作者大学时代的Java学习经历。文章以作者做的思维导图作为目录,展示了整个讲解过程。 ... [详细]
  • 本文讨论了如何优化解决hdu 1003 java题目的动态规划方法,通过分析加法规则和最大和的性质,提出了一种优化的思路。具体方法是,当从1加到n为负时,即sum(1,n)sum(n,s),可以继续加法计算。同时,还考虑了两种特殊情况:都是负数的情况和有0的情况。最后,通过使用Scanner类来获取输入数据。 ... [详细]
  • 本文介绍了在Java中gt、gtgt、gtgtgt和lt之间的区别。通过解释符号的含义和使用例子,帮助读者理解这些符号在二进制表示和移位操作中的作用。同时,文章还提到了负数的补码表示和移位操作的限制。 ... [详细]
  • Go GUIlxn/walk 学习3.菜单栏和工具栏的具体实现
    本文介绍了使用Go语言的GUI库lxn/walk实现菜单栏和工具栏的具体方法,包括消息窗口的产生、文件放置动作响应和提示框的应用。部分代码来自上一篇博客和lxn/walk官方示例。文章提供了学习GUI开发的实际案例和代码示例。 ... [详细]
author-avatar
萧海豚泳_756
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有