获取public静态属性
ognl -c 7cd84586 '@com.system.framework.ArtahsDemoClassLoader@pubTestPrex' ognl -c 7cd84586 "@com.system.framework.ArtahsDemoClassLoader@pubfinalTestPrex" 输出各式 @Type[属性值],内容如下
@String[static public] ... @String[final static public] 获取private静态属性
ognl -c 7cd84586 '@com.system.framework.ArtahsDemoClassLoader@privTestPrex' ognl -c 7cd84586 '@com.system.framework.ArtahsDemoClassLoader@privFnalTestPrex' @String[static private] ... @String[final static private] 修改public静态属性,被final修饰,不能被修改
// author:herbert qq:464884492 date:20220331 测试代码想修改FINAL修饰符,结果没有成功 @Test public void testModifyFinal() throws Exception { Field finalField = this.getClass().getDeclaredField("privFnalTestPrex"); finalField.setAccessible(true); System.out.println("==========初始值=========="); System.out.println(finalField.get(null)); Field modiField = Field.class.getDeclaredField("modifiers"); modiField.setAccessible(true); modiField.setInt(finalField, finalField.getModifiers() & ~Modifier.FINAL); finalField.set(null, "修改后FInal"); System.out.println("==========修改值=========="); System.out.println(privFnalTestPrex); } 静态变量赋值,不能通过=直接赋值,需要采用反射的方式设置值
ognl '#c=@com.system.framework.ArtahsDemoClassLoader@class,#f=#c.getDeclaredField("pubTestPrex"),#f.set(#c,"modify static public ")' 修改private静态属性,需要在反射时调用方法setAccessible,使private特殊转化为public
ognl '#c=@com.system.framework.ArtahsDemoClassLoader@class,#f=#c.getDeclaredField("privTestPrex"),#f.setAccessible(true),#f.set(#c,"modify static private ")' 静态方法调用和静态属性一样,格式为@class@method(args)
无参数调用
ognl -c 7cd84586 '@com.system.framework.ArtahsDemoClassLoader@setPublicStaticMethod()' ognl -c 7cd84586 '@com.system.framework.ArtahsDemoClassLoader@modfiyPrivateStaticFiled()' ... ======第5次输出====== 源文件初始输出==>static public/static private/testRefect--1/final static public/final static private 源文件初始输出==>static public/static private/testRefect--2/final static public/final static private ======第6次输出====== 源文件初始输出==>modify by method static public/static private/testRefect--1/final static public/final static private 源文件初始输出==>modify by method static public/static private/testRefect--2/final static public/final static private ... ======第11次输出====== 源文件初始输出==>modify by method static public/static private/testRefect--1/final static public/final static private 源文件初始输出==>modify by method static public/static private/testRefect--2/final static public/final static private ======第12次输出====== 源文件初始输出==>modify by method static public/modify by method static private/testRefect--1/final static public/final static private 源文件初始输出==>modify by method static public/modify by method static private/testRefect--2/final static public/final static private ... // author:herbert qq:464884492 date:20220331 有参数调用
ognl -c 7cd84586 '@com.system.framework.ArtahsDemoClassLoader@getPublicStaticMethod("input params")' ... ognl -c 7cd84586 '@com.system.framework.ArtahsDemoClassLoader@getStaticPrivateMethod("input params")' @String[input params <==> public static method return string] ... @String[input params <==> private static method return string] 从以上的测试结果来说,静态方法不管是public还是private都可以直接调用。
查看某个类实例,无 --limit 参数默认10个
vmtool -c 3e2e18f2 -a getInstances --className *EncryptClass vmtool -c 3e2e18f2 -a getInstances --className *EncryptClass --express 'instances.length' vmtool -c 3e2e18f2 -a getInstances --className *EncryptClass --express 'instances[0]' @EncryptClass[][ @EncryptClass[com.system.framework.EncryptClass@3c573d32], @EncryptClass[com.system.framework.EncryptClass@68390fae], ] ... @Integer[2] ... @EncryptClass[ note=@String[testRefect--1], ] 经过上边测试发现,一个类存在多个classloader加载时,需要指定classloader。但从返回结果看,返回了所有classloader加载的实例
调用实例getNote 和setNote 方法
vmtool -c 3e2e18f2 -a getInstances --className *EncryptClass --express '#val=instances[0],#val.getNote()' vmtool -c 3e2e18f2 -a getInstances --className *EncryptClass --express '#val=instances[0],#val.setNote("modify by instance"+#val.getNote())' @String[testRefect--1] ======第7次输出====== 源文件初始输出==>static public/static private/testRefect--1/final static public/final static private 源文件初始输出==>static public/static private/testRefect--2/final static public/final static private ======第8次输出====== 源文件初始输出==>static public/static private/modify by instancetestRefect--1/final static public/final static private 源文件初始输出==>static public/static private/testRefect--2/final static public/final static private 从控制台输出结果,对比第7次和8次输出,我们可以发现第一个loader加载的class实例已经成功修改了
获取或者修改第一个实例 note 属性
vmtool -c 3e2e18f2 -a getInstances --className *EncryptClass --express '#val=instances[1].note' vmtool -c 3e2e18f2 -a getInstances --className *EncryptClass --express '#val=instances[1],#val.note="modify by instance"+#val.note' @String[testRefect--2] ... ======第121次输出====== 源文件初始输出==>static public/static private/modify by instancetestRefect--1/final static public/final static private 源文件初始输出==>static public/static private/testRefect--2/final static public/final static private ======第122次输出====== 源文件初始输出==>static public/static private/modify by instancetestRefect--1/final static public/final static private 源文件初始输出==>static public/static private/modify by instancetestRefect--2/final static public/final static private 从控制台输出结果,对比第121次和122次输出,我们可以发现第二个loader加载的class实例已经成功修改了
返回的对象集合,可以做二次筛选投影操作,也可以带条件查询符合的数据
vmtool -c 3e2e18f2 -a getInstances --className *EncryptClass --express '#val=instances' vmtool -c 3e2e18f2 -a getInstances --className *EncryptClass --express '#val=instances.{note}' vmtool -c 3e2e18f2 -a getInstances --className *EncryptClass --express '#val=instances.{#this.note}' vmtool -c 3e2e18f2 -a getInstances --className *EncryptClass --express '#val=instances.{? #this.note.indexOf("1")>0}.{note}' @EncryptClass[][ @EncryptClass[com.system.framework.EncryptClass@52790e67], @EncryptClass[com.system.framework.EncryptClass@822cf83], ] ... @ArrayList[ @String[modify by instancetestRefect--1], @String[modify by instancetestRefect--2], ] ... @ArrayList[ @String[modify by instancetestRefect--1], @String[modify by instancetestRefect--2], ] ... @ArrayList[ @String[modify by instancetestRefect--1], ]