INDY工具的使用(读深入理解java虚拟机验证)

什么是INDY

由于invokedynamic指令所面对的使用者并非java语言,而是其他java虚拟机支持的动态语言,因此仅依靠java语言的编译器javac没有办法生成带有invokedynamic指令的字节码,所以要使用java语言来演示invokedynamic指令只能用一些变通的办法。John Rose编写了一个把程序的字节码转换未使用invokedynamic的简单工具INDY来完成这件事情,我们要用这个工具来产生最终要的字节码。

代码验证

转换类 InvokeDynamicTest

import java.lang.invoke.CallSite;
import java.lang.invoke.ConstantCallSite;
import java.lang.invoke.MethodHandle;
import java.lang.invoke.MethodHandles;
import java.lang.invoke.MethodType;

/**
 * @author hyl
 * @version v1.0: InvokeDynamicTest.java, v 0.1 2020/8/31 15:23 $
 */
public class InvokeDynamicTest {

    public static void testMethod(String s) {
        System.out.println("hello string:" + s);
    }

    public static CallSite BootstrapMethod(MethodHandles.Lookup lookup, String name, MethodType mt) throws Throwable {
        return new ConstantCallSite(lookup.findStatic(InvokeDynamicTest.class, name, mt));
    }

    private static MethodType MT_BootstrapMethod() {
        return MethodType.fromMethodDescriptorString(
            "(Ljava/lang/invoke/MethodHandles$Lookup;Ljava/lang/String;Ljava/lang/invoke/MethodType;)Ljava/lang/invoke/CallSite;",
            null);
    }

    private static MethodHandle MH_BootstrapMethod() throws Throwable {
        return MethodHandles.lookup()
            .findStatic(InvokeDynamicTest.class, "BootstrapMethod", MT_BootstrapMethod());
    }

    private static MethodHandle INDY_BootstrapMethod() throws Throwable{
        CallSite cs = (CallSite)MH_BootstrapMethod().invokeWithArguments(MethodHandles.lookup(),"testMethod",MethodType.fromMethodDescriptorString("(Ljava/lang/String;)V",null));
        return cs.dynamicInvoker();
    }

    public static void main(String[] args) throws Throwable {
        INDY_BootstrapMethod().invokeExact("hyl");
    }
}

测试类 ExampleTest


/**
 * @author hyl
 * @version v1.0: ExampleTest.java, v 0.1 2020/8/31 16:50 $
 */
public class ExampleTest {
    public static void main(String[] args) throws Throwable {
        System.out.println("main");
        Indify.main("--verify-specifier-count=1",
            /*INDY工具将Example.class转换成等效的class文件存放路径*/
            "--dest=INDY/",
            "--verbose",
            "--expand-properties", "--classpath", "${java.class.path}",
            /*输入class文件Example.class的全路径,然后使用javap打开该文件*/
            "D:\\git\\mybase\\projects\\github\\learner-JVM\\target\\classes\\com\\hyl\\learnerJVM\\jvmstack\\InvokeDynamicTest.class");
        //Example.main();
    }
}

工具类 Indify

链接:https://pan.baidu.com/s/1vQnzHEl1inVeIwA3T2NFtw
提取码:zfqh

后记

开始一直没有成功,工具格式不对,字符串有遗漏,还有转换类的关键方法 private 自己写成了 public。浪费了很多时间,特意记录一下。

参考

https://blog.csdn.net/QWE123ZXCZ/article/details/82764748

《深入理解Java虚拟机》第三版,周志明著。

使用TIdAntiFreeze对抗“冻结”   Indy使用一个特殊的组件TIdAntiFreeze来透明地解决客户程序用户界面“冻结”的问题。TIdAntiFreeze在Indy内部定时中断对栈的调用,并在中断期间调用Application.ProcessMessages方法处理消息,而外部的Indy调用继续保存阻塞状态,就好像TIdAntiFreeze对象不存在一样。你只要在程序中的任意地方添加一个TIdAntiFreeze对象,就能在客户程序中利用到阻塞式Socket的所有优点而避开它的一些显著缺点。   Indy使用了线程技术   阻塞式Socekt通常都采用线程技术,Indy也是如此。从最底层开始,Indy的设计都是线程化的。因此用Indy创建服务器和客户程序跟在Unix下十分相似,并且Delphi的快速开发环境和Indy对WinSock的良好封装使得应用程序创建更加容易。   Indy服务器模型   一个典型的Unix服务器有一个或多个监听进程,它们不停地监听进入的客户连接请求。对于每一个需要服务的客户,都fork一个新进程来处理该客户的所有事务。这样一个进程只处理一个客户连接,编程就变得十分容易。   Indy服务器工作原理同Unix服务器十分类似,只是Windows不像Unix那样支持fork,而是支持线程,因此Indy服务器为每一个客户连接分配一个线程。   图1显示了Indy服务器的工作原理。Indy服务器组件创建一个同应用程序主线程分离的监听线程来监听客户连接请求,对于接受的每一个客户,都创建一个新的线程来为该客户提供服务,所有与这一客户相关的事务都由该线程来处理。   使用组件TIdThreadMgrPool,Indy还支持线程池。   线程与Indy客户程序   Indy客户端组件并未使用线程。但是在一些高级的客户程序中,程序员可以在自定义的线程中使用Indy客户端组件,以使用户界面更加友好。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值