Java 17 Reflection 反射 Field 篇
这篇来说说反射中剩下的两个知识点的其中之一,分别是 Field (属性,领域,字段)和 Method (方法或函数)。
先看 Field 知识点, Method 篇就要另外再开一篇了。
Field
对于 Field 提供有关类或接口的单个属性的信息以及对属性的动态访问。 对于反射的属性而言, 可以是类的属性,类的静态属性,以及类成员实例。
对于 Field 的使用,允许 get 和 set 访问操作期间发生扩大转换,但是如果发生了缩写转换,将会引发 IllegalArgumentException 异常信息。
Field 定义
public final class Field extends AccessibleObject implements Member
想要访问属性,就必须先有操作的对象, 这里可以参考反射知识点 Class 篇。不详细说这个, 至直接演示 Field 的使用。
基础的 equals、toString、toGenericString、hashCode 不再演示,使用方式很简单,之前也详细说过这些, 包括下面的 Method 也会包含该方法。
对于 Field 中的方法主要包含三种类型, 一种是注解相关的,另外是获得不同类型的基础类型封装类型以及设置对应的值。第三种就是基础的操作。获得属性的名称,以及获得属性对应的类对象,类型等操作信息。
对应的方法如下所示:
简单的使用方式,根据类获得对应的字段信息。这里需要注意 getField 和 getDeclaredField 区别。 getField 是获得 public 修饰符的 Field 属性值。
既然是字段属性,就一定有对应的值。毕竟变量只是一个标识符。 最终还是为了获得值。这里演示如果获取对象的值。
当然除了这种方法, 还可以直接获得数据基础类型的封装类型。格式为:
get[封装类型](Object obj)
演示案例如下:
定义基础的操作类:
class FieldTest {
FieldTest() {
}
public String string1;
public int int1;
public char char1;
public boolean boolean1;
public double double1;
public float float1;
public short short1;
public byte byte1;
}
完整代码如下:
根据 Field 的 getType 获得类型的字符串,然后根据类型获得对应的数据。这里只是为了演示这些获得封装类型数据的方法。
有获得就应该有对应的方法。对于设置的方法, 使用 set 开头。
定义的格式如下:
set[封装类型](Object obj, [封装类型 | 基础类型])
举例如下:
isEnumConstant(): 判断是否是枚举类型。
isSynthetic(): 如果此字段是合成字段,则返回 true;否则返回 false。 演示效果如下。
如果是私有对象又想访问该怎么办呢?可以参考 setAccessible 方法。
public void setAccessible(boolean flag)
设置反射对象的可访问权,当值为 true 时,反射对象在使用时会取消对 Java 语言访问控制的检查。值为 false 时,表示反射对象在使用时应检查 Java 语言访问控制,并且会在类中描述中会注明变化。
此方法虽然可以启用修改但是不是所有的操作都可以, 在以下的字段中不可修改。
- 在任何类或接口中声明的静态最终字段
- 在隐藏类中声明的最终字段
- 在记录中声明的最终字段
- 为 true 时的可访问标志禁止 Java 语言访问控制检查,以仅启用对这些不可修改的最终字段的读取访问。
错误的例子如下:
java.lang.NoSuchFieldException
字段没有找到异常。
如果想访问私有对象就需要使用 setAccessible(boolean) 方法。演示案例如下:
写一个小例子, 比方说有一个实体类, UserInfo.java 用来存放用户表中的字段。这个时候, 我们就可以使用反射以及接下来的字段的注解来演示这个案例。
首先 UserInfo.java 的代码:
@TableName(value="user_info")
class UserInfo {
private String uid;
private String uname;
private int age;
private String birthday;
private String auth;
private LocalDateTime created;
@FieldName(value="nick_name")
private String nickName;
}
定义了 2 个注解对象,分别是 TableName 和 FieldName
@Documented
@Target(ElementType.TYPE_USE)
@Retention(RetentionPolicy.RUNTIME)
@interface TableName{
String value();
}
@Documented
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
@interface FieldName{
String value();
}
封装两个方法, 一个是通过 Class 获得对应的表名,这里表名设计成类名或者注解的值。字段一样也是。
public static String getClassName(Class<?> clazz){
TableName tableName = clazz.getAnnotation(TableName.class);
if(tableName == null){
return clazz.getName();
}else{
return tableName.value();
}
}
以及根据字段类, 获得字段名或者注解名
public static String getFieldName(Field field){
FieldName FieldName = field.getAnnotation(FieldName.class);
if(FieldName == null){
return field.getName();
}else{
return FieldName.value();
}
}
测试方法:
public static void main(String[] args) throws Exception {
Class clazz = UserInfo.class;
Field[] fields = clazz.getDeclaredFields();
StringBuffer sb = new StringBuffer();
sb.append("select ");
int i = 0;
for (Field field : fields) {
i ++;
sb.append(getFieldName(field));
if(i < fields.length ){
sb.append(", ");
}
}
sb.append(" from ");
sb.append(getClassName(clazz));
System.out.println(sb);
}
完整代码如下:
测试运行查看效果:
从这里可以看出来, 既然可以获得对应的字段名称, 其实你可以在这个基础上做更多的操作, 比方说,根据实体类获得创建表的 SQL 语句等。
对于 Field 类基本上就这些内容了。 希望对您有所帮助。感谢阅读。
点赞, 关注, 收藏。
现在热点, Java 编程知识点是一个相对成熟的体系, 并且对应有自己的优缺点。随着 Golang 和 Rust 的出现, Java 后端服务被替代的概率越来越大,但是这个时候,出现了静态编译技术。该技术主要是基于 GraalVM 。各位看官有精力可以尝试学习一下。还是很有必要的。最近也在优化一些项目使用该技术。Java 的基础技术体系,写完了, 可以单独开一个体系说一说这个。立个 2022 的flag。