一.引入
上文http://zy19982004.iteye.com/blog/1979208中,提到了注解类JyzTargetPackage可以定义为@Target(ElementType.PACKAGE),可是在被注解类里我无论怎么加,编译器都报错,于是引入了package-info.java这个文件。
二.创建package-info.java
- "I found that when you create a new package in eclispe there is a check box to check if you want a package-info.java."勾上就行了。
- 如果不幸的是你已经创建了这个包并在里面定义了很多类,而eclispe又是不能直接创建一个package-info.java文件的。只能在包对应文件夹里,手动创建一个package-info.java,写上包名,最后刷新eclispe即可。
三.package-info.java的作用
- "Package annotations must be in file package-info.java",package-info.java为我们提供了包注解的地方。JyzTargetPackage(http://zy19982004.iteye.com/blog/1979208)苦苦寻找终于找到地方了。
- 提供包级别的类(或接口),这些类(或接口)只有本包里才能访问,即使是子包也不能访问。
- 提供包的整体注释说明。
package-info.java
/**
* <b>package-info不是平常类,其作用有三个:</b><br>
* 1、为标注在包上Annotation提供便利;<br>
* 2、声明包的私有类和常量;<br>
* 3、提供包的整体注释说明。<br>
*
* @author [email protected]
*/
@JyzTargetPackage(version="1.0")
package com.jyz.study.jdk.annotation;
class PackageInfo{
public void common(){
System.out.println("sa");
}
}
class PackageInfoGeneric<T extends Throwable>{
private T obj;
public void set(T obj){
this.obj = obj;
}
public void common(){
System.out.println(obj + "sa");
}
}
interface packageInfoInteger{
public void test();
}
class PackageConstants{
public static final String ERROE_CODE = "100001";
}
TestPackageInfo.java
package com.jyz.study.jdk.annotation;
import java.io.IOException;
/**
* 测试package-info.java文件的作用
* 1、为标注在包上Annotation提供便利;<br>
* 2、声明包的私有类和常量;<br>
* @author [email protected]
*
*/
public class TestPackageInfo {
public static void main(String[] args) {
//1
Package p = Package.getPackage("com.jyz.study.jdk.annotation");
if(p != null && p.isAnnotationPresent(JyzTargetPackage.class)){
JyzTargetPackage nav = p.getAnnotation(JyzTargetPackage.class);
if(nav != null){
System.out.println("package version:" + nav.version());
}
}
//2
PackageInfo packageInfo = new PackageInfo();
packageInfo.common();
//泛型也能很好的工作,在pakcage-info.java里定义的类和普通类没什么区别
PackageInfoGeneric<Exception> packageInfoGeneric = new PackageInfoGeneric<Exception>();
packageInfoGeneric.set(new IOException("device io"));
packageInfoGeneric.common();
Sub sub = new Sub();
sub.test();
System.out.println(PackageConstants.ERROE_CODE);
}
}
class Sub implements packageInfoInteger{
@Override
public void test() {
System.out.println("sub");
}
}
console output:
package version:1.0
sa
java.io.IOException: device iosa
sub
100001
需要注意两点
- package-info.java里不能声明public class(或 interface)
- 刚开始p.isAnnotationPresent(JyzTargetPackage.class)返回false,后来找到原因JyzTargetPackage没有加上@Retention(RetentionPolicy.RUNTIME)。