微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

java – 内存如何分配给lambda |它是如何通过非超类引用变量引用的

我正在创建功能界面的实现,下面是我的代码
Consumer<Integer> consumer=new Consumer<Integer>() {
    @Override
    public void accept(Integer t) {
        System.out.println(t);
    }
};

按照javadoc

A variable of a class type T can hold a null reference or a reference
to an instance of class T or of any class that is a subclass of T.

这里创建了匿名对象的Object,它是Consumer的子类,可以通过引用变量consumer引用,这很精细.

但我看到Consumer是FunctionalInterface,所以我也可以在java8中做这样的事情 –

使用Lambda

Consumer<Integer> consumer=t->System.out.println(t);

或使用方法参考

Consumer<Integer> consumer=System.out::println;

我知道在上述两种情况下都没有创建子类或匿名类.所以这导致我两次混淆 –

1:如果这里的消费者不是指子类或消费者的匿名类,那么这不违反上面提到的概念变量只能引用child / self或null吗?

2:内存如何分配给lamdas以及JVM在运行时如何处理?

解决方法

首先,Jean-Baptiste向您展示了为什么作业首先起作用.

现在,我认为您缺少的部分是在Consumer< Integer>的情况下生成的Consumer类.消费者= t – >的System.out.println(T);仅在运行时由于invokedynamic而可见.

使用标志运行您的类:

java -Djdk.internal.lambda.dumpProxyClasses=/Your/Path

并且您会注意到有一个生成的类(在类的包名称中的文件夹路径中)包含.class文件,类似于此SOQuestion $$Lambda $1.class.

如果你反编译,你会发现它实际上是一个实现Consumer的类:

final class org.eugene.so.soQuestion$$Lambda$1 
            implements java.util.function.Consumer {
     public void accept(java.lang.Object);
}

如果你看一下定义lambda的生成字节代码,你会看到lambda表达式实际上是对静态方法去糖(对于非捕获lambda,如果它是一个捕获lambda,它也可以是非静态的) ).它的代码

private static void lambda$main$0(java.lang.Integer);
Code:
   0: getstatic     #3  // Field java/lang/System.out:Ljava/io/PrintStream;
   3: aload_0
   4: invokevirtual #4  // Method java/io/PrintStream.println:(Ljava/lang/Object;)V
   7: return

至于VM如何为它管理内存,对于非捕获lambda,你将得到一个单例,对于capture-lambda,你将获得每个调用的新实例.绝对必读为here

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐