
java 本身不支持像 javascript 那样用 `obj["propertyname"]` 的语法直接通过字符串访问字段,但可通过反射(reflection)api 在运行时动态获取字段值、调用方法,实现类似效果。
在 Java 中,对象属性的访问是静态编译型的——编译器必须在编译期确认字段或方法是否存在,因此 object["property"] 这类动态语法不被语言语法支持(JLS 第18章明确未定义此类操作符)。但若需在运行时根据字符串名称操作成员(例如实现通用序列化、配置驱动逻辑、ORM 映射或脚本桥接),Java 提供了功能完备的 java.lang.reflect 包来达成这一目标。
✅ 基础示例:通过字符串读取字段值
假设我们有如下类:
class Example {
private String string = "stack overflow";
public int number = 42;
}要通过字符串 "string" 获取其值,需使用反射:
Example example = new Example();
try {
Field field = example.getClass().getDeclaredField("string");
field.setAccessible(true); // 绕过 private 访问限制
String value = (String) field.get(example);
System.out.println(value); // 输出:stack overflow
} catch (NoSuchFieldException | IllegalAccessException e) {
throw new RuntimeException(e);
}⚠️ 注意:
立即学习“Java免费学习笔记(深入)”;
- getDeclaredField() 仅查找本类声明的字段(含 private),不包含继承字段;如需父类字段,应使用 getField()(仅限 public)或递归遍历 getSuperclass()。
- 私有字段默认不可访问,必须调用 setAccessible(true) 启用暴力反射(在 JDK 12+ 及安全策略严格环境下可能受限,需配置 --illegal-access=permit 或迁移到模块化开放方案)。
✅ 动态调用方法(带参数)
继续使用上述 Example 类,若定义了方法:
public int compute(int a, int b) {
return a * b;
}可通过字符串 "compute" 调用:
try {
Method method = example.getClass().getDeclaredMethod("compute", int.class, int.class);
int result = (int) method.invoke(example, 3, 5);
System.out.println(result); // 输出:15
} catch (Exception e) {
throw new RuntimeException(e);
}? 提示:getDeclaredMethod() 的第二个及后续参数为 形参类型的 Class 对象(如 String.class, List.class),顺序和类型必须严格匹配,否则抛出 NoSuchMethodException。
⚠️ 重要注意事项与最佳实践
-
性能开销:反射比直接调用慢数倍至数十倍,切勿在高频路径(如循环内部、实时渲染、高并发请求处理)中滥用。可考虑缓存 Field/Method 实例(如用 ConcurrentHashMap
)。 - 安全性与模块化:自 JDK 9 起,模块系统默认限制对非导出包的反射访问。若在模块化项目中使用,需在 module-info.java 中添加 opens package.name to java.base;。
- 健壮性:务必捕获并处理 NoSuchFieldException、IllegalAccessException、InvocationTargetException 等检查异常,避免运行时崩溃。
- 替代方案考量:对于通用数据绑定场景,推荐使用成熟库(如 Apache Commons BeanUtils、Spring Framework 的 BeanWrapper 或 Jackson 的 ObjectMapper),它们已封装反射细节并提供类型安全、空值处理与性能优化。
✅ 总结
虽然 Java 不具备 JavaScript 式的动态属性语法,但反射机制提供了强大而灵活的运行时元编程能力。掌握 Field.get()、Method.invoke() 及其配套 API,能有效支撑框架开发、工具构建与低代码平台等高级需求。关键在于——理解其适用边界,权衡灵活性与性能/安全性,并优先选用经过验证的抽象层而非裸反射。










