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

相关文章

Java中的String是不可变对象 在面向对象及函数编程语言中,不...
String, StringBuffer 和 StringBuilder 可变性 String不可变...
序列化:把对象转换为字节序列的过程称为对象的序列化. 反序...
先说结论,是对象!可以继续往下看 数组是不是对象 什么是对...
为什么浮点数 float 或 double 运算的时候会有精度丢失的风险...
面试题引入 这里引申出一个经典问题,看下面代码 Integer a ...