- 浏览: 2181434 次
- 性别:
- 来自: 北京
文章分类
- 全部博客 (1240)
- mac/IOS (287)
- flutter (1)
- J2EE (115)
- android基础知识 (582)
- android中级知识 (55)
- android组件(Widget)开发 (18)
- android 错误 (21)
- javascript (18)
- linux (70)
- 树莓派 (18)
- gwt/gxt (1)
- 工具(IDE)/包(jar) (18)
- web前端 (17)
- java 算法 (8)
- 其它 (5)
- chrome (7)
- 数据库 (8)
- 经济/金融 (0)
- english (2)
- HTML5 (7)
- 网络安全 (14)
- 设计欣赏/设计窗 (8)
- 汇编/C (8)
- 工具类 (4)
- 游戏 (5)
- 开发频道 (5)
- Android OpenGL (1)
- 科学 (4)
- 运维 (0)
- 好东西 (6)
- 美食 (1)
最新评论
-
liangzai_cool:
请教一下,文中,shell、C、Python三种方式控制led ...
树莓派 - MAX7219 -
jiazimo:
...
Kafka源码分析-序列5 -Producer -RecordAccumulator队列分析 -
hp321:
Windows该命令是不是需要安装什么软件才可以?我试过不行( ...
ImageIO读jpg的时候出现javax.imageio.IIOException: Unsupported Image Type -
hp321:
Chenzh_758 写道其实直接用一下代码就可以解决了:JP ...
ImageIO读jpg的时候出现javax.imageio.IIOException: Unsupported Image Type -
huanghonhpeng:
大哥你真强什么都会,研究研究。。。。小弟在这里学到了很多知识。 ...
android 浏览器
使用过 Java 反射的大多都知道, 想要修改某个类或对象的私有变量的值的话, 在调用 set 设置新值之前执行一下 setAccessible(true) 即可。这样利用的 Java 的反射就能绕过 private 的限制 ,不再有 IllegalAccessException 异常了。这是一个 trick, 调用 Java 的私有方法也能这么做,有些人或许或这样来测试 Java 私有方法。
提前说一句:在修改 final 型值时,要特别留意它的常量值本身是否被编译器优化内联到某处,否则你会看到虽然没什么异常,但取出的还是原来的值。后面会稍为深入的讲到。
例如下面是一段完整的代码, 由于调用了 setAccessiable(true), 所以能成功把 OneCity 的私有属性 name 的值改为 "Shenzhen":
那么如果是一个 final 类型的属性呢,像 private static final String name = "Beijing"; 该如何用反射来修改它的值呢。要是仍然用上面的方法即使设置了 setAccessible(true) 也会报 IllegalAccessException。
这时候我们要做一个更彻底的反射 -- 对 Java 反射包中的类进行自我反射。Field 对象有个一个属性叫做 modifiers, 它表示的是属性是否是 public, private, static, final 等修饰的组合。这里把这个 modifiers 也反射出来,进而把 nameField 的 final 约束也去掉了,回到了上面的状况了。完整代码是这样的
在 ① 处把 Field 的 modifiers 找到,它也是个私有变量,所以也要 setAccessible(ture)。接着在 ② 处把 nameField 的 modifiers 值改掉,是用的按位取反 ~ 再按位与 ~ 操作把 final 从修饰集中剔除掉,其他特性如 private, static 保持不变。再想一下 modifierField.setInt() 可以把 private 改为 public, 如此则修改 name 时无需 setAccessible(true) 了。
通过把把属性的 final 去掉, 就成功把 name 改成了 Shenzhen。
注意上面为何把 OneCity 的 name 赋值为 new String("Beijing"), 这是为了不让 Java 编译器内联 name 到 getName() 方法中,而使 getName() 的方法体为 return "Beijing",造成 getName() 永远输出 ”Beijing" 。
本文原始链接 http://unmi.cc/java-reflection-modify-final-field-value/, 来自 隔叶黄莺 Unmi Blog
提到 Java 编译器对 final 属性的 Inline 优化,还有种情况会造成你能修改 final 型属性,但试图打印出的还是原来的值。如下面的代码片段:
上面对 OneCity 的属性修改看起来成功了,又好像不成功,其实是修改成功了的。原因是 Java 在对代码行
System.out.println(OneCity.name);
内联了 OneCity 的 final 属性 name 的常量值,编译为
System.out.println("Beijing");
所以执行该行总是输出 "Beijing"。让他能输出修改后的值的办法就是阻止 Java 作这个内联优化,让 final 属值的获得需要作个计算才成,如上面的 new String("Beijing") 就是这个目的。
反射已是一种 Hacker 行为了,感觉已无所不能,但 CGLIB, JavaAssist 那些东西与之之比起来更是流氓了。
提前说一句:在修改 final 型值时,要特别留意它的常量值本身是否被编译器优化内联到某处,否则你会看到虽然没什么异常,但取出的还是原来的值。后面会稍为深入的讲到。
例如下面是一段完整的代码, 由于调用了 setAccessiable(true), 所以能成功把 OneCity 的私有属性 name 的值改为 "Shenzhen":
package cc.unmi; import java.lang.reflect.Field; public class TestReflection { public static void main(String[] args) throws Exception { Field nameField = OneCity.class.getDeclaredField("name"); nameField.setAccessible(true); //这个起决定作用 nameField.set(null, "Shenzhen"); System.out.println(OneCity.getName()); //输出修改后的 Shenzhen } } class OneCity { private static String name = "Beijing"; public static String getName() { return name; } }
那么如果是一个 final 类型的属性呢,像 private static final String name = "Beijing"; 该如何用反射来修改它的值呢。要是仍然用上面的方法即使设置了 setAccessible(true) 也会报 IllegalAccessException。
这时候我们要做一个更彻底的反射 -- 对 Java 反射包中的类进行自我反射。Field 对象有个一个属性叫做 modifiers, 它表示的是属性是否是 public, private, static, final 等修饰的组合。这里把这个 modifiers 也反射出来,进而把 nameField 的 final 约束也去掉了,回到了上面的状况了。完整代码是这样的
package cc.unmi; import java.lang.reflect.Field; import java.lang.reflect.Modifier; public class TestReflection { public static void main(String[] args) throws Exception { Field nameField = OneCity.class.getDeclaredField("name"); Field modifiersField = Field.class.getDeclaredField("modifiers"); //① modifiersField.setAccessible(true); modifiersField.setInt(nameField, nameField.getModifiers() & ~Modifier.FINAL); //② nameField.setAccessible(true); //这个同样不能少,除非上面把 private 也拿掉了,可能还得 public nameField.set(null, "Shenzhen"); System.out.println(OneCity.getName()); //输出 Shenzhen } } class OneCity { private static final String name = new String("Beijing"); public static String getName() { return name; } }
在 ① 处把 Field 的 modifiers 找到,它也是个私有变量,所以也要 setAccessible(ture)。接着在 ② 处把 nameField 的 modifiers 值改掉,是用的按位取反 ~ 再按位与 ~ 操作把 final 从修饰集中剔除掉,其他特性如 private, static 保持不变。再想一下 modifierField.setInt() 可以把 private 改为 public, 如此则修改 name 时无需 setAccessible(true) 了。
通过把把属性的 final 去掉, 就成功把 name 改成了 Shenzhen。
注意上面为何把 OneCity 的 name 赋值为 new String("Beijing"), 这是为了不让 Java 编译器内联 name 到 getName() 方法中,而使 getName() 的方法体为 return "Beijing",造成 getName() 永远输出 ”Beijing" 。
本文原始链接 http://unmi.cc/java-reflection-modify-final-field-value/, 来自 隔叶黄莺 Unmi Blog
提到 Java 编译器对 final 属性的 Inline 优化,还有种情况会造成你能修改 final 型属性,但试图打印出的还是原来的值。如下面的代码片段:
//... 与上相同 nameField.set(null, "Shenzhen"); System.out.println(nameField.get(null)); //输出 Shenzhen System.out.println(OneCity.name); //仍然打印出 Beijing class OneCity { public static final String name = "Beijing"; }
上面对 OneCity 的属性修改看起来成功了,又好像不成功,其实是修改成功了的。原因是 Java 在对代码行
System.out.println(OneCity.name);
内联了 OneCity 的 final 属性 name 的常量值,编译为
System.out.println("Beijing");
所以执行该行总是输出 "Beijing"。让他能输出修改后的值的办法就是阻止 Java 作这个内联优化,让 final 属值的获得需要作个计算才成,如上面的 new String("Beijing") 就是这个目的。
反射已是一种 Hacker 行为了,感觉已无所不能,但 CGLIB, JavaAssist 那些东西与之之比起来更是流氓了。
发表评论
-
小程序textarea完美填坑
2020-07-07 16:09 503相信做微信小程序的码友们都被textarea这个原生组件坑过 ... -
Nginx+Https自己敲命令生成证书
2020-05-18 09:35 940一、准备 环境:centos6.8 ... -
https证书生成环境搭建配置(基于Tomcat和Nginx)
2020-04-24 11:06 816一、基于Tomcat、JDK内置密钥工具: 1、生成服务端证 ... -
史上最强Tomcat8性能优化
2019-11-01 21:41 847授人以鱼不如授人以渔 ... -
SpringBoot配置HTTPS,并实现HTTP访问自动转HTTPS访问
2019-10-07 09:13 5551.使用jdk自带的 keytools 创建证书 打开cmd ... -
Spring Boot工程集成全局唯一ID生成器 UidGenerator
2019-09-16 09:04 848概述 流水号生成器(全局唯一 ID生成器)是服务化系统的基础 ... -
CentOS7下Redis的安装与使用
2019-08-17 11:45 593一、手动安装过程 1、准备工作(安装gcc依赖) yum ... -
Nginx与tomcat组合的简单使用
2019-08-17 10:05 399配置tomcat跳转 请求http出现400的时候在这里配置 ... -
linux下lvs+keepalived安装配置
2019-07-10 14:20 458keepalived主机:192.168.174. ... -
使用Docker搭建Tomcat运行环境
2019-02-08 21:32 4831 准备宿主系统 准备一 ... -
Netty笔记-GlobalEventExecutor
2019-02-06 23:00 6251.概念 /** * Single-thread si ... -
Netty4转发服务的实现方案
2019-02-06 15:03 1114如果用Netty做转发服务(不需要同步应答),Netty中有一 ... -
java手机号归属地查询
2018-12-25 17:16 735所需的包:carrier-1.75.jar 、geocoder ... -
基于Netty4的HttpServer和HttpClient的简单实现
2018-10-17 20:02 670Http 消息格式: Http request: Met ... -
javafx : 支持使用微调(spinner)控制的数字的文本框(NemberTextField)
2018-10-16 00:00 1074最近花了一些时间学习javaFX, 要更深入地理解新GUI包, ... -
我的Java(定制你的Java/JavaFX Runtime)
2018-10-12 23:29 669最新的JDK 11发布了,撒花 新版本的JDK终于有了ope ... -
javaFX的几个新特性,让swing彻底过时
2018-10-12 22:42 658首先声明,Java的GUI曾经 ... -
mac os系统用install4j把jar包生成app
2018-10-05 23:02 1415install4j有windows版也有mac版 mac电脑 ... -
JavaFX Alert对话框
2018-10-05 22:01 23341. 标准对话框 消息对话框 Alert alert = ... -
IDEA Properties中文unicode转码问题
2017-02-17 19:54 980摘要: 如何让IDEA的properties中的中文进行uni ...
相关推荐
本文将探讨如何通过Java反射有效地修改`final`属性值,并理解其中的限制。 首先,我们来看一个简单的例子,定义一个`Person`类,包含一个`final`属性`name`: ```java class Person { public final String name =...
在Java编程语言中,反射(Reflection)是一种强大的工具,它允许程序在运行时检查和操作类、接口、字段和方法等对象。枚举(Enumeration)是Java中的一个特殊类类型,用于定义一组常量。本项目"test-enum-demo-...
Java反射机制是Java语言提供的一种强大功能,它允许运行中的Java程序对自身进行检查并且可以直接操作程序的内部属性。在Java中,反射机制的核心类是java.lang.Class,它代表了运行时的类信息。通过Class对象,我们...
### Java反射机制详解 #### 一、Java反射机制概述 Java反射机制是Java语言中一个非常重要的特性,它使得程序可以在运行时动态地获取类的信息并进行操作。反射能力允许程序在运行期间检查自身,包括类、接口、字段...
Java反射是Java编程语言的一个强大特性,它允许在运行时检查和操作类、接口、字段和方法的信息。通过反射,我们可以动态地创建对象、调用方法、访问私有成员等,这在处理不确定类型的对象或者需要进行元编程的场景中...
Java反射大全是对Java编程语言中一项重要特性——反射的深入探讨。反射机制使得Java程序在运行时能够获取关于自身的信息,如类、接口、字段和方法等,并能在运行时动态地创建对象、调用方法、修改属性。这使得Java...
Java反射是Java编程语言中的一个重要特性,允许程序在运行时检查和操作类、接口、对象的内部信息。在“java概述课件24反射.ppt”中,主要讲解了Java中关于类型信息和反射的核心概念。 类型信息在Java中是指类、接口...
- **反射机制**:该工具类大量使用了Java反射API(如`Introspector.getBeanInfo()`、`PropertyDescriptor`等)来获取和设置JavaBean的属性。 - **泛型**:使用了泛型`<T>`,使得该工具类可以应用于任意类型的...
9. **反射机制**:学习Java反射机制,如何在运行时动态获取类信息、创建对象、调用方法以及修改属性值。 10. **注解(Annotation)**:了解注解的定义、使用和处理,以及元注解的机制,探讨其在代码生成、AOP(面向...
#### 一、JAVA反射机制概述 JAVA的反射机制是一种强大的功能,允许程序在运行时获取自身的结构信息并能够直接操作这些内部结构。这种能力通常被称为运行时类型信息(RunTime Type Information, RTTI)。通过反射...
(Java 反射机制,可以在运行时访问和修改类的信息) 15. 什么是 Java 序列化?什么情况下需要序列化?(序列化是将对象转换为字节流,用于网络传输或持久化) Java 异常 16. `throw` 和 `throws` 的区别?(`...
此外,我们还可以看到,通过反射获取到的属性值可以通过`Properties`对象进行修改。 ##### 2. 修改配置信息 为了实现对数据库连接信息的修改,我们需要创建一个自定义的`ConnectionProvider`类,继承自`...
Java 中有基本数据类型(整数、浮点数、字符、布尔值等)和引用数据类型(数组、类、接口等)。 * Java 中什么是继承?继承是指一个类从另一个类继承属性和行为的机制。 * 如何实现继承?使用 extends 关键字实现...
- **变量**:存储值的容器。 - **作用域**:变量的有效范围。 - **局部变量**:在方法、构造函数或块内声明的变量。 - **成员变量**:在类体中声明的变量。 ##### 4.2 操作符 Java支持多种操作符,包括算术操作符...
如果既不指定name也不指定type属性,这时将通过反射机制使用byName自动注入策略。 二、final关键字 final关键字是Java语言中的一种修饰符,用于修饰变量、方法和类。final修饰变量时,表示该变量一旦获得了初始值...
属性和方法 构造方法 方法重载 方法的重写 初始化 类的初始化 成员初始化 构造器初始化初始化顺序 数组初始化 对象的销毁 对象作用域 this 和 super 访问控制权限继承 多态组合代理 向上转型static final 接口和...
- 参数传递:参数都是按值传递的,但对于引用型变量,传递的值是引用值,所以方法中对数据的操作可以改变对象的属性。 - 动态绑定:Java 中所有的非 final 的方法都会自动地进行动态绑定。 - 上溯造型:把派生...
除了这些基础知识,学习Java还需要掌握异常处理、文件操作、集合框架(List, Set, Map)、输入/输出流、线程并发、反射、注解等高级主题。通过不断的实践和项目经验积累,你将能够更好地运用Java解决实际问题,成为...