Java针对类、成员变量、方法,有很多修饰符,例如public、private、static、final、synchronized、abstract等,这些修饰符用来控制访问权限或其他特性。
本文就用成员变量(Field)来举例说明,类以及方法的修饰符获取与成员变量是一样的。
先看一个类
public class MyTest
{
public int a;
public static int b;
public static final int c = 0;
private int d;
}
通过反射,可以拿到这四个变量:
public static void main(String[] args) {
Class<?> clazz = MyTest.class;
Field[] fields = clazz.getDeclaredFields();//获取这个类所有的成员变量
for(Field field : fields) {
System.out.println(field.getName());
}
}
输出:
a
b
c
d
现在,我想知道每个变量的带有哪些修饰符,或者是不是包含某个修饰符。
先看下Member接口:Member表示一个类中的成员,包括成员变量、方法、构造方法三种实现,上面用到的Field就是Member的一种。
Java文档:
java.lang.reflect
接口 Member
所有已知实现类:
Constructor(构造方法), Field(成员变量), Method(方法)
Member接口有个方法:
int getModifiers()
作为整数返回由此 Member所表示的成员或构造方法的 Java语言修饰符。
同时再看java.lang.Class类中也有同样的一个方法:
int getModifiers()
返回此类或接口以整数编码的 Java语言修饰符。
这个方法就是返回一个int型的返回值,代表类、成员变量、方法的修饰符。
public static void main(String[]args) {
Class<?> clazz = MyTest.class;
Field[] fields = clazz.getDeclaredFields();
for(Field field : fields)
{
System.out.print(field.getName() +"->");
System.out.println(field.getModifiers());
}
}
输出:
a->1
b->9
c->25
d->2
通过返回的int类型的值,还是很难判断这些变量有哪些修饰符。
在这里,需要用到java.lang.reflect.Modifier这个类。Modifier提供了很多静态方法。如public static String toString(int mod)就可以输出该整数对应的所有的修饰符。public static boolean isPublic(int mod)就可以判断该整数对应的是不是包含public修饰符。
修改上面代码,把返回的整数转换通过Modifier转换一下:
public static void main(String[] args) {
Class<?> clazz = MyTest.class;
Field[] fields = clazz.getDeclaredFields();
for(Field field : fields)
{
System.out.print(field.getName() +"->");
System.out.println(Modifier.toString(field.getModifiers()));
}
}
输出:
a->public
b->public static
c->public static final
d->private
通过Modifier的isPublic、isPrivate、isStatic等方法,可以判断是否包含某些修饰符,现在如果有这么一个需求,我需要找到仅有publicstatic两个修饰符的变量。
现在看下Modifier的源码,可以看到以下代码:
public static final int PUBLIC = 0x00000001;
public static final int PRIVATE =0x00000002;
public static final int PROTECTED = 0x00000004;
public static final int STATIC =0x00000008;
public static final int FINAL =0x00000010;
public static final int SYNCHRONIZED = 0x00000020;
……还有很多 此处省略一万字
把它们转换成二进制,可以看出,Modifier使用一个二进制的位来表示是否包含某个修饰符。
……
native
transient
volatile
synchronized
final
static
protected
private
public
也就是,如果是public static,对应的整数就是二进制的:1001,也就是9。如下图:
……
native
transient
volatile
synchronized
final
static
protected
private
public
0
0
0
0
0
1
0
0
1
如果是public static final就是11001,也就是25。
现在如果想判断是否仅有public static两个修饰符,那么就可以判断是否field.getModifiers() == 25。
另外,我们也可以看到这里设计之精妙之处:用二进制位来标记是否包含某个修饰符。