主要介绍以下内容:

  • 如何弃用API

  • @deprecate Javadoc标签和@Deprecation注解在弃用的API中的角色

  • 用于生成弃用警告的详细规则

  • 在JDK 9中更新@Deprecation注解

  • JDK 9中的新的弃用警告

  • 如何使用@SuppressWarnings注解来抑制JDK 9中的不同类型的弃用警告

  • 如何使用jdeprscan静态分析工具来扫描编译的代码库,以查找已弃用的JDK API的用法

一. 什么是弃用

Java中的弃用是提供有关API生命周期的信息的一种方式。 可以弃用模块,包,类型,构造函数,方法,字段,参数和局部变量。 当弃用API时,要告诉其用户:

  • 不要使用API,因为它存在风险。

  • API已经迁移,因为存在API的更好的替代方案。

  • API已经迁移,因为API将在以后的版本中被删除。

二. 如何弃用API

JDK有两个用于弃用API的结构:

  • @deprecated Javadoc标签

  • @Deprecated注解

@deprecated Javadoc标签已添加到JDK 1.1中,它允许使用丰富的HTML文本格式功能指定关于弃用的详细信息。JDK 5.0中添加了java.lang.Deprecated注解类型,并且可以在已被弃用的API元素上使用。 在JDK 9之前,注解不包含任何元素。 它在运行时保留。

@deprecated标签和@Deprecated注解应该一起使用。 两者都应该存在或两者都不存在。 @Deprecation注解不允许指定弃用的描述,因此必须使用@deprecated标签来提供描述。

Tips
在API元素上使用@deprecated标签(而不是@Deprecated注解)会生成编译器警告。 在JDK 9之前,需要使用-Xlint:dep-ann编译器标志来查看这些警告。

下面包含FileCopier类的声明。 假设这个类作为类库迁移的一部分。 该类使用@Deprecation注解表示弃用。 它的Javadoc使用@deprecated标签来提供不推荐使用的详细信息,例如不推荐使用的时间,它的替换和删除通知。 在JDK 9之前,@Deprecated注解类型不包含任何元素,因此必须使用Javadoc中已弃用的API的@deprecated标签提供有关弃用的所有详细信息。 请注意,Javadoc中使用的@since标签表示FileCopier类自该库的版本1.2以来已经存在,而@deprecated标签表示该类自版本1.4以来已被弃用。

// FileCopier.javapackage com.jdojo.deprecation;import java.io.File;/**
 * The class consists of static methods that can be used to
 * copy files and directories.
 *
 * @deprecated Deprecated since 1.4. Not safe to use. Use the
 * <code>java.nio.file.Files</code> class instead. This class
 * will be removed in a future release of this library.
 *
 * @since 1.2
 */@Deprecatedpublic class FileCopier {    // No direct instantiation supported.
    private FileCopier() {
    }    /**
     * Copies the contents of src to dst.
     * @param src The source file
     * @param dst The destination file
     * @return true if the copy is successfully,
     * false otherwise.
     */
    public static boolean copy(File src, File dst) {        // More code goes here
        return true;
    }    // More methods go here}

Javadoc工具将@deprecated标签的内容移动到生成的Javadoc中的顶部,以引起读者的注意。 当不被弃用的代码使用不推荐使用的API时,编译器会生成警告。 请注意,使用@Deprecated注解标注API不会生成警告;但是,使用已经使用@Deprecated注解标注的API。 如果在类本身之外使用FileCopier类,则会收到关于使用不推荐使用的类的编译器警告。

三. JDK 9中@Deprecated注解的更新

假设编译了代码并将其部署到生产环境中。如果升级了JDK版本或包含旧应用程序使用的新的已弃用的API的库/框架,则不会收到任何警告,并且将错过从不推荐使用的API迁移的机会。必须重新编译代码以接收警告。没有任何扫描和分析编译代码(例如JAR文件)的工具,并报告使用已弃用的API。更坏的情况是,从旧版本中删除不推荐使用的API,而旧的编译代码会收到意外的运行时错误。当他们查看不赞成使用的元素Javadoc时,开发人员也感到困惑 —— 当API被废弃时,无法表达何种方式,以及在将来的版本中是否会删除已弃用的API。所有可以做的是在文本中将这些信息指定为@deprecated标签的一部分。 JDK 9尝试通过增强@Deprecated注解来解决这些问题。注解在JDK 9中已增加两个新元素:sinceforRemoval

在JDK 9之前,注解的声明如下:

@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target(value={CONSTRUCTOR, FIELD, LOCAL_VARIABLE, METHOD, PACKAGE, PARAMETER, TYPE})public @interface Deprecated {
}

在JDK 9中,弃用注解的声明更改为以下内容:

@Documented@Retention(RetentionPolicy.RUNTIME)@Target(value={CONSTRUCTOR, FIELD, LOCAL_VARIABLE, METHOD, PACKAGE, MODULE, PARAMETER, TYPE})public @interface Deprecated {    String since() default "";    boolean forRemoval() default false;
}

两个新元素都具有指定的默认值,因此注解的现有使用不会有问题。 since元素指定已注解的API元素已被弃用的版本。 它是一个字符串,将遵循与JDK版本方案相同的版本命名约定,例如“9”。 它默认为空字符串。 请注意,JDK 9没有向@Deprecated注解类型添加元素,以指定不推荐的描述。 这是由于两个原因:

  • 注解在运行时保留。 向注解添加描述性文本将添加到运行时内存。

  • 描述性文字不能只是纯文本。 例如,它需要提供一个链接来替代已弃用的API。 现有的@deprecated Javadoc标签已经提供了这个功能。

forRemoval元素表示注解的API元素在将来的版本中被删除,应该迁移API。 它默认为false。

Tips
元素上的@since Javadoc标签表示何时添加了API元素,而@Deprecated注解的since元素表示API元素已被弃用。

在JDK 9之前,弃用警告是基于API元素及其使用场景(use-site)上使用@Deprecated注解的问题,如下所示。 当在没有弃用的使用场景使用不推荐使用的API元素时,会发出警告。 如果声明及其使用场景都已弃用,则不会发出任何警告。 可以通过使用@SuppressWarnings("deprecation")注解标示用户场景来抑制弃用警告。

API Use-SiteAPI Declaration SiteAPI Declaration Site
EmptyNot DeprecatedDeprecated
Not DeprecatedNW
DeprecateNN
N = No warning,W = Warning

@Deprecation注解类型中添加forRemoval元素增加了多于五个用例。 当forRemoval设置为false时,不推荐使用API,则将这种弃用称为普通弃用,在这种情况下发出的警告称为普通弃用警告。 当forRemoval设置为true时,不推荐使用API,则将这种弃用称为终止弃用,并且在这种情况下发出的警告称为终止弃用警告或删除警告。

API Use-SiteAPI Declaration SiteAPI Declaration SiteAPI Declaration Site
EmptyNot DeprecatedOrdinarily DeprecatedTerminallyDeprecated
Not DeprecatedNOWRW
Ordinarily DeprecatedNNRW
Terminally DeprecatedNNRW
N = No warning,OW = Ordinary deprecation warning,RW = Removal deprecation warning

为了实现向后兼容,如果代码在JDK 8中生成了弃用警告,它将继续在JDK 9中生成普通的弃用警告。如果API已经被终止使用,其使用场景将生成删除警告,而不考虑使用场景状态。

在JDK 9中,在一个情况下发出的警告,其API和其使用场景都被最终弃用,这些警告需要一点解释。 API和使用它的代码都已被弃用,并且将来都会被删除,所以在这种情况下要发出警告是什么意思? 这样做是为了涵盖最终弃用的API及其使用场景在两个不同的代码库中并独立维护的情况。 如果使用场景代码库存活超过了API代码库,则用场景将会收到意外的运行时错误,因为它使用的API不再存在。用场景发出警告将提供一个机会,以防在用场景的代码去掉之前,来计划替代最终弃用的API。

四. 抑制弃用警告

介绍JDK 9中的removal警告已添加了一个新的用例来抑制弃用警告。 在JDK 9之前,可以通过使用@SuppressWarnings("deprecation")注解标示使用场景来抑制所有弃用警告。 考虑以下几种情况:

  • 在JDK 8中,弃用的API和使用场景会抑制弃用警告。

  • 在JDK 9中,API的弃用从普通的弃用变为最终弃用。

  • 由于JDK 8中抑制了弃用警告,所以在JDK 9中使用场景的编译没有问题。

  • API被删除,并且使用场景收到意外的运行时错误,而不会在之前接收到任何删除警告。

为了涵盖这种情况,当使用@SuppressWarnings("deprecation"),JDK 9不会抑制删除警告。 它只抑制普通的弃用警告。 要抑制删除警告,需要使用@SuppressWarnings("removal")。 如果要抑制普通和删除的弃用警告,则需要使用@SuppressWarnings({“deprecation”, "removal"})

五. 一个弃用API示例

在本节中,展示弃用API的所有用例,使用弃用使用的API,并通过一个简单的示例来抑制警告。 在该示例中,对方法标示为弃用的,并使用它们来生成编译时警告。 但是,不限于仅弃用方法。 对这些方法的注解可以更好地了解预期的行为。 下面包含一个Box类的代码。 该类包含三种方法 —— 没有弃用的方法,普通的弃用方法和最终弃用的方法。 编译Box类不会生成任何废弃警告,因为该类不使用任何已弃用的API,而是包含过时的API。

// Box.javapackage com.jdojo.deprecation;/**
 * This class is used to demonstrate how to deprecate APIs.
 */public class Box {    /**
     * Not deprecated
     */    
    public static void notDeprecated() {
        System.out.println("notDeprecated...");
    }    /**
     * Deprecated ordinarily.
     * @deprecated  Do not use it.
     */    
    @Deprecated(since="2")    public static void deprecatedOrdinarily() {
        System.out.println("deprecatedOrdinarily...");
    }    /**
     * Deprecated terminally.
     * @deprecated  It will be removed in a future release.
     *              Migrate your code.
     */    
    @Deprecated(since="2", forRemoval=true)    public static void deprecatedTerminally() {
        System.out.println("deprecatedTerminally...");
    }
}

下面包含BoxTest类的代码。 该类使用Box类的所有方法。 BoxTest类中的几种方法已经被普遍和最终弃用了。 m4X()的方法,其中X是数字,显示如何抑制弃用警告。

// BoxTest.javapackage com.jdojo.deprecation;public class BoxTest {    /**
     * API: Not deprecated
     * Use-site: Not deprecated
     * Deprecation warning: No warning
     */
    public static void m11() {
        Box.notDeprecated();
    }    /**
    * API: Ordinarily deprecated
    * Use-site: Not deprecated
    * Deprecation warning: No warning
    */
    public static void m12() {
        Box.deprecatedOrdinarily();
    }    /**
     * API: Terminally deprecated
     * Use-site: Not deprecated
     * Deprecation warning: Removal warning
     */
    public static void m13() {
        Box.deprecatedTerminally();
    }    /**
     * API: Not deprecated
     * Use-site: Ordinarily deprecated
     * Deprecation warning: No warning
     * @deprecated Dangerous to use.
     */
    @Deprecated(since="1.1")    public static void m21() {
        Box.notDeprecated();
    }    /**
    * API: Ordinarily deprecated
    * Use-site: Ordinarily deprecated
    * Deprecation warning: No warning
    * @deprecated Dangerous to use.
    */
    @Deprecated(since="1.1")    
    public static void m22() {
        Box.deprecatedOrdinarily();
    }    /**
     * API: Terminally deprecated
     * Use-site: Ordinarily deprecated
     * Deprecation warning: Removal warning
     * @deprecated Dangerous to use.
    */
    @Deprecated(since="1.1")    public static void m23() {
        Box.deprecatedTerminally();
    }    /**
     * API: Not deprecated
     * Use-site: Terminally deprecated
     * Deprecation warning: No warning
     * @deprecated Going away.
     */
    @Deprecated(since="1.1", forRemoval=true)    public static void m31() {
        Box.notDeprecated();
    }    /**
    * API: Ordinarily deprecated
    * Use-site: Terminally deprecated
    * Deprecation warning: No warning
    * @deprecated Going away.
    */
    @Deprecated(since="1.1", forRemoval=true)    public static void m32() {
        Box.deprecatedOrdinarily();
    }    /**
     * API: Terminally deprecated
     * Use-site: Terminally deprecated
     * Deprecation warning: Removal warning
     * @deprecated Going away.
    */
    @Deprecated(since="1.1", forRemoval=true)    public static void m33() {
        Box.deprecatedTerminally();
    }    /**
     * API: Ordinarily and Terminally deprecated
     * Use-site: Not deprecated
     * Deprecation warning: Ordinary and removal warnings
    */    
    public static void m41() {
        Box.deprecatedOrdinarily();
        Box.deprecatedTerminally();        
    }    /**
     * API: Ordinarily and Terminally deprecated
     * Use-site: Not deprecated
     * Deprecation warning: Ordinary warnings
    */    
    @SuppressWarnings("deprecation")    public static void m42() {
        Box.deprecatedOrdinarily();
        Box.deprecatedTerminally();        
    }    /**
     * API: Ordinarily and Terminally deprecated
     * Use-site: Not deprecated
     * Deprecation warning: Removal warnings
    */    
    @SuppressWarnings("removal")    public static void m43() {
        Box.deprecatedOrdinarily();
        Box.deprecatedTerminally();        
    }    /**
     * API: Ordinarily and Terminally deprecated
     * Use-site: Not deprecated
     * Deprecation warning: Removal warnings
    */    
    @SuppressWarnings({"deprecation", "removal"})    public static void m44() {
        Box.deprecatedOrdinarily();
        Box.deprecatedTerminally();        
    }
}

需要使用-Xlint:deprecation编译器标志来编译BoxTest类,因此编译会发出弃用警告。 请注意,以下命令在一行上输入,而不是两行。

C:\Java9Revealed\com.jdojo.deprecation\src>javac-Xlint:deprecation-d ..\build\classes com\jdojo\deprecation\BoxTest.java

输出结果为:

com\jdojo\deprecation\BoxTest.java:20: warning: [deprecation] deprecatedOrdinarily() in Box has been deprecated
        Box.deprecatedOrdinarily();
           ^com\jdojo\deprecation\BoxTest.java:29: war

http://www.cnblogs.com/IcanFixIt/p/7234054.html

延伸阅读

向着Java前进-Java培训,做最负责任的教育,学习改变命运,软件学习,再就业,大学生如何就业,帮大学生找到好工作,lphotoshop培训,电脑培训,电脑维修培训,移动软件开发培训,网站设计培训,网站建设培训向着Java前进