简单例子

import java.lang.invoke.MethodHandle;
import java.lang.invoke.MethodHandles;
import java.lang.invoke.MethodType; public class MHT { private String name; // 构造方法
public MHT(String name) {
this.name = name;
} // 公共方法
public String publicTest() {
return name + "'publicTest";
} // 静态方法
public static String publicStaticTest() {
return "'publicStaticTest";
} // 私有方法
private String test(int param) {
switch (param) {
case 1:
return "suych1";
case 2:
return "suych2";
case 3:
return "suych3";
default:
return "suych4";
}
} // Get方法
public String getName() {
return name;
} // Set方法
public void setName(String name) {
this.name = name;
} public static void main(String[] args) throws Throwable {
// 构造方法
MethodType mtConstructor = MethodType.methodType(void.class, String.class); //返回值类型,参数类型
MethodHandle mhConstructor = MethodHandles.lookup().findConstructor(MHT.class, mtConstructor);
MHT businessHandle = ( MHT ) mhConstructor.invokeExact("suych");
System.out.println(businessHandle.getName()); // 公共方法
MethodType mtPublic = MethodType.methodType(String.class); //返回值类型
MethodHandle mhPublic = MethodHandles.lookup().findVirtual(MHT.class, "publicTest", mtPublic);
String resultPublic = ( String ) mhPublic.invokeExact(businessHandle);
System.out.println(resultPublic); // 静态方法
MethodType mtPublicStatic = MethodType.methodType(String.class);
MethodHandle mhPublicStatic = MethodHandles.lookup().findStatic(MHT.class, "publicStaticTest",
mtPublicStatic);
String resultPublicStatic = ( String ) mhPublicStatic.invokeExact();
System.out.println(resultPublicStatic); // 私有方法
MethodType mtPrivate = MethodType.methodType(String.class, int.class); //返回值类型,参数类型
MethodHandle mhPrivate = MethodHandles.lookup().findSpecial(MHT.class, "test", mtPrivate,
MHT.class);
String resultPrivate = ( String ) mhPrivate.invokeExact(businessHandle, 1);
System.out.println(resultPrivate); // Set方法
MethodHandle mhSet = MethodHandles.lookup().findSetter(MHT.class, "name", String.class);
mhSet.invokeExact(businessHandle, "A");
System.out.println(businessHandle.name); // Get方法
MethodHandle mhGet = MethodHandles.lookup().findGetter(MHT.class, "name", String.class);
String resultGet = ( String ) mhGet.invokeExact(businessHandle);
System.out.println(resultGet); } }

输出:

suych   构造方法输出
suych'publicTest 公共方法输出
'publicStaticTest 静态方法输出
suych1 私有方法输出
A Set方法输出
A Get方法输出

基于MethodHandle实现的调用Runtime执行系统命令

import java.io.InputStream;
import java.lang.invoke.MethodHandle;
import java.lang.invoke.MethodHandles;
import java.lang.invoke.MethodType;
import java.util.Scanner; /**
* @author yz
*/
public class MethodHandlesTest { public static void main(String[] args) {
try {
String str = "arp -a";
Class runtimeClass = Runtime.class;
MethodHandles.Lookup lookup = MethodHandles.lookup(); // Runtime rt = Runtime.getRuntime()
MethodHandle methodHandle = lookup.findStatic(
runtimeClass, "getRuntime", MethodType.methodType(runtimeClass)
); // 获取Runtime的exec方法
MethodHandle execMethod = lookup.findVirtual(
runtimeClass, "exec", MethodType.methodType(Process.class, new Class[]{
String.class
})
); // 获取Process的getInputStream方法
MethodHandle inputStreamMethod = lookup.findVirtual(
Process.class, "getInputStream", MethodType.methodType(InputStream.class)
); // 调用Runtime.getRuntime().exec(xxx).getInputStream()
InputStream in = (InputStream) inputStreamMethod.invoke(
execMethod.invoke(methodHandle.invoke(), str)
); // 输出InputStream内容到
Scanner scanner = new Scanner(in).useDelimiter("\\A");
System.out.println(scanner.hasNext() ? scanner.next() : "");
} catch (Throwable t) {
t.printStackTrace();
}
} }

JAVA反射修改private,final值

FinalName.java

class FinalName {
public final String name="init";
}

PrivateName.java

class PrivateName {
private String name = "init";
public String getName() {
return name;
}
}

Test.java

import java.lang.reflect.Field;

public class Test {
public static void main(String[] args) throws Exception {
PrivateName privateName = new PrivateName();
FinalName finalName = new FinalName();
// System.out.println(finalName.name);
modify(privateName, "name", "private change");
modify(finalName, "name", "final change");
System.out.println(privateName.getName());
System.out.println(finalName.name);
}
public static void modify(Object object, String fieldName, Object newFieldValue) throws Exception {
Field field = object.getClass().getDeclaredField(fieldName);
field.setAccessible(true);
field.set(object, newFieldValue);
System.out.println(field.get(object));
}
}

输出结果,private修饰的变量结果已经被修改,但是final修改正常但是获取的依然是原先的值,这是因为内联优化。

private change
final change
private change
init

如果将FinalName.java修改为通过构造方法给final修饰的属性赋值。

class FinalName {
public final String name; FinalName(String name) {
this.name = name;
}
}
FinalName Finalname = FinalName("aaaaaa");

再次运行Test.java,发现final修饰的变量已经被修改。

private change
final change
private change
final change

总结:private修饰的变量可以通过反射的方法将值修改,需要设置访问权限为true。field.setAccessible(true);

final修饰的变量如果是直接赋值,则对属性值进行修改无效。如果是通过构造方法修改属性的值,则可以通过反射的方法修改final修饰的变量。

通过反射方式执行命令。

import java.io.InputStream;
import java.lang.reflect.Method;
import java.util.Scanner;
public class ReflectionTest { public static void exec() {
try {
System.out.println(Runtime.class.getMethod("exec", String.class).invoke(Runtime.class.getMethod("getRuntime").invoke(null), "curl -i localhost:8000"));
} catch (Exception e) {
e.printStackTrace();
}
} public static void main(String[] args) {
try {
String str = "arp -a"; // java.lang.Runtime
String runtime = new String(new byte[]{106, 97, 118, 97, 46, 108, 97, 110, 103, 46, 82, 117, 110, 116, 105, 109, 101}); // Runtime.class
Class<?> c = Class.forName(runtime); // 获取getRuntime方法,Runtime.getRuntime()
Method m1 = c.getMethod(new String(new byte[]{103, 101, 116, 82, 117, 110, 116, 105, 109, 101})); // 获取Runtime的exec方法,rt.exec(xxx)
Method m2 = c.getMethod(new String(new byte[]{101, 120, 101, 99}), String.class); // Runtime.getRuntime().exec(str)
Object obj2 = m2.invoke(m1.invoke(null), str); // 获取命令执行结果Process类的getInputStream()方法
Method m = obj2.getClass().getMethod(new String(new byte[]{103, 101, 116, 73, 110, 112, 117, 116, 83, 116, 114, 101, 97, 109}));
m.setAccessible(true); // process.getInputStream()
InputStream in = (InputStream) m.invoke(obj2, new Object[]{}); // 输出InputStream内容到
Scanner scanner = new Scanner(in).useDelimiter("\\A");
System.out.println(scanner.hasNext() ? scanner.next() : "");
} catch (Throwable t) {
t.printStackTrace();
}
} }

看下为什么获取命令执行结果Process类的getInputStream()方法访问权限需要设置true。

跟入java.lang.ProcessImpl类



getInputStream方法返回stdout_stream是私有变量





注释掉,则会报下面这样的错误

参考链接:

https://docs.oracle.com/javase/7/docs/api/java/lang/invoke/MethodType.html

https://mp.weixin.qq.com/s/mlqjOlhefcsO9z51cw4S7w

https://blog.csdn.net/yhd723948277/article/details/82661870

# JDK7+ MethodHandle的更多相关文章

  1. 【JVM】2、关于jdk7的MethodHandle类

    关于MethodHandle类,这个类是在jdk1.7之后加入的,这个类的作用类似函数指针的意思 这个类中有一个方法 这里我的jdk有一个问题,就是我在进行MethodHandle操作的时候,我们会发 ...

  2. java7 java MethodHandle解析

    简介 JDK6之前我们会使用java反射来实现动态方法调用,多数框架用反射的比较多,例如mybatis.spring等.在JDK7中,新增了java.lang.invoke.MethodHandle( ...

  3. MethodHandle

    JDK7为间接调用方法引入新的API,在java.lang.invoke包下,可以看作为反射的升级版,但它不像反射API那样显得冗长.繁重 主要的类 MethodHandle 方法句柄.对可直接执行的 ...

  4. 阿里云CentOS7系列一 -- 安装JDK7的方法.

    最近因为数据采集以及生产环境冲突.导入windows Server 2008系统经常死机.经讨论决定把采集服务程序和生产服务进行分开.采集程序通过windows Server2008运行.而生产程序通 ...

  5. JDK7和JDK8一些重要新特性

    jdk7新特性(部分) switch支持字符串 List AutoCloseable接口实现自动关闭,在try()中 新增获取环境信息的工具方法,getJavaHomeDir,getUserHomeD ...

  6. 《java JDK7 学习笔记》之异常处理

    1.java中所有的错误都会被打包为对象,JVM会尝试执行try区块中的程序代码,如果发生错误,执行流程会跳离错误发生点,然后比较catch括号中声明的异常类型,是否符合被抛出的错误对象类型,如果是的 ...

  7. MyEclipse10的一些问题(git插件,jdk7)

    egit: MyEclipse10 要装 egit2.3,版本错了安装不成功; jdk7: 10.5好像是不支持JDK1.7的,换成10.7; JDK1.7中的switch支持String类型的,1. ...

  8. win7 安装JDK7和JDK8后,卸载JDK8后出错

    这是本人学习Java过程中遇到的一些问题和解决方法,在此记录,方便本人查看,解决他人疑惑. 本人win7 x64旗舰版,同时安装了JDK7和JDK8,卸载了JDK8之后,cmd命令行输入:java - ...

  9. java 28 - 6 JDK7的新特性

    JDK7的新特性: 二进制字面量 数字字面量可以出现下划线 switch 语句可以用字符串 泛型简化 异常的多个catch合并 try-with-resources 语句 二进制字面量 JDK7开始, ...

随机推荐

  1. web手工项目04-测试报告的核心内容-HTTP请求和响应-fiddler工具进行抓包

    回顾 登录功能测试(需求分析(输入分析,处理分析,输出分析),数据构造,编写用例) 测试点设计思路--扩展(相关业务,特殊角度-分支,异常,逆向) 流程图元素与画法(开始/结束,路径,数据,判定,步骤 ...

  2. 修改root密码

    Linux 密码的修改,使用passwd 命令修改 命令如下图:sudo passwd root 即可修改成功

  3. ABAP程序并行处理

    CASE1. 程序中 start new task ,并在后面获取处理结果 *"------------------------------------------------------- ...

  4. 【Leetcode_easy】599. Minimum Index Sum of Two Lists

    problem 599. Minimum Index Sum of Two Lists 题意:给出两个字符串数组,找到坐标位置之和最小的相同的字符串. 计算两个的坐标之和,如果与最小坐标和sum相同, ...

  5. EXCEL中,在其中列 前面or后面加一个“元”字的技巧

    EXCEL小技巧,我们平常需要用到一些,记录下,供有需要的人参考! 案例: EXCEL其中的一列,每个后面加一个“元”字,如果要1个1个去加,相当麻烦,其实很简单,只需要一个公式即可! 解决方法: ( ...

  6. 这一次,真正掌握composer

      思维导图 composer是现代PHP的基石 现代高级编程语言,依赖管理工具是必不可少的.Java有Maven,Python有pip,Nodejs有npm, 而在composer出现之前,PHP只 ...

  7. Edit Delete Mysql的主从复制

    参考博客 https://www.cnblogs.com/zhoujie/p/mysql1.html Mysql的主从复制至少是需要两个Mysql的服务,当然Mysql的服务是可以分布在不同的服务器上 ...

  8. 乐字节Java构造器(构造方法|构造函数)、方法与内存分析

    一. 构造器(构造方法|构造函数) 在创建对象时(new),必会调用一个特殊的方法,这个方法是初始化对象信息的为new服务的.这个方法称为“构造器” 使用 new + 构造方法创建一个新的对象. 构造 ...

  9. [CF369E]Valera and Queries_离线_树状数组

    Valera and Queries 题目链接:codeforces.com/problemset/problem/369/E 数据范围:略. 题解: 这种题,就单独考虑一次询问即可. 我们发现,包括 ...

  10. php之简单算法

    选择排序 方式:先让第一位与其他位比较大小找到最小的数字,然后是第二位与除第一位的其他位比较大小找出第二位,依此类推 $arr = [2,45,12,67,33,5,23,132,46]; for ( ...