java - Lombok 和斯波克 : @RequiredArgsConstructor doesn't hide default no-args constructor for a field with a type of interface

标签 java spock lombok intellij-lombok-plugin

@RequiredArgsConstructor 似乎在下面的代码中不起作用 - 但仅在使用 Spock 框架的测试中,并且仅适用于接口(interface) Dao 类型的字段。
严格来说 - 代码正在工作,但在我看来它不应该工作,考虑到 JUnit5 下的类似测试根本无法编译。

有人可以解释一下这是一个错误还是一个功能吗?

package brumba;
public interface Dao {
    Integer getValueFor(Integer value);
}
<小时/>
package brumba;

import com.sun.istack.internal.NotNull;
import lombok.RequiredArgsConstructor;

@RequiredArgsConstructor
public class Brumba {

    @NotNull
    final private Dao dao;

//  If you uncomment the below 2 lines, then the test fails
//    @NotNull
//    final private String name;

    public Integer twice(Integer x){
        return x * 2;
    }

    public Integer twiceDao(Integer x){
        return dao.getValueFor(x);
    }
}
<小时/>

下面的代码工作正常 - 但仅在 Spock 中(JUnit5 下的类似测试无法编译)。
似乎 Spock 测试以某种方式看到默认的无参数构造函数(而 JUnit 测试看不到)这个构造函数)
但是当上面的 2 行注释被取消注释时,测试失败并出现以下错误:

groovy.lang.GroovyRuntimeException: Could not find matching constructor for: brumba.Brumba()
<小时/>
package brumba

import spock.lang.Specification

class BrumbaTest extends Specification {

    def "twice should multiply argument by 2"() {
        given:
            def testedObject = new Brumba();

        expect:
            y == testedObject.twice( x )

        where:
            x | y
            0 | 0
            1 | 2
            2 | 4
            3 | 6
    }
}
<小时/>

这个 JUnit 测试根本无法编译:

package brumba;

class BrumbaJUnit5Test {

    @org.junit.jupiter.api.Test
    void shouldTwice() {
        Brumba br = new Brumba();
    }
} 

错误是:

Error:(7, 21) java: constructor Brumba in class brumba.Brumba cannot be applied to given types;
  required: brumba.Dao,java.lang.String
  found: no arguments

以下是我用于此项目的依赖项:

<dependency>
    <groupId>org.spockframework</groupId>
    <artifactId>spock-core</artifactId>
    <version>1.2-groovy-2.5</version>
    <scope>test</scope>
</dependency>

<dependency>
    <groupId>org.projectlombok</groupId>
    <artifactId>lombok</artifactId>
    <version>1.18.4</version>
    <scope>provided</scope>
</dependency>
<dependency>
    <groupId>org.junit.jupiter</groupId>
    <artifactId>junit-jupiter-api</artifactId>
    <version>5.3.0-M1</version>
    <scope>test</scope>
</dependency>

最佳答案

首先,我可以确认这种情况也发生在我身上。以前没注意过。

我必须通过源代码进行调试并查看反编译的文件,以便至少了解这里发生的情况。我可以告诉你一些事情:

  • 与 Lombok 无关。它也会发生在任何具有采用对象类型(即不是像 int 这样的原语)的单参数构造函数的 Java 类上,例如String 或您的 Dao
  • 与 Spock 无关,因为它也发生在 Spock 之外。
  • 它似乎与动态 Groovy 运行时功能有关。
  • 我宁愿称其为微妙的错误,而不是功能,但我不确定。

Java 类:

package de.scrum_master.stackoverflow;

public class Brumba {
  public Brumba(String name) {}
}

Groovy 类:

package de.scrum_master.stackoverflow

class BrumbaApp {
  static void main(String[] args) {
    new Brumba()
  }
}

反编译的 Groovy 类:

package de.scrum_master.stackoverflow;

import groovy.lang.GroovyObject;
import groovy.lang.MetaClass;
import org.codehaus.groovy.runtime.callsite.CallSite;

public class BrumbaApp implements GroovyObject {
  public BrumbaApp() {
    CallSite[] var1 = $getCallSiteArray();
    MetaClass var2 = this.$getStaticMetaClass();
    this.metaClass = var2;
  }

  public static void main(String... args) {
    CallSite[] var1 = $getCallSiteArray();
    var1[0].callConstructor(Brumba.class);
  }
}

Groovy 运行时类 CallSite 实际上是一个接口(interface),但有 AbstractCallSite实现它。如果我们看一下这个方法

public Object callConstructor(Object receiver) throws Throwable {
    return callConstructor(receiver, CallSiteArray.NOPARAM);
}

以及这个定义

public final class CallSiteArray {
    // ...
    public static final Object [] NOPARAM = new Object[0];
    // ...

我们知道实际上会调用这个方法

public Object callConstructor(Object receiver, Object[] args) throws Throwable {
    return CallSiteArray.defaultCallConstructor(this, receiver, args);
}

等等。我认为发生的情况是,大小为 0 的 Object[] 将以某种方式作为构造函数参数传递,并且缺少被解释为 null 参数的元素。这也是您在对象实例化后在调试器中看到的内容,如果像在代码中一样将参数分配给成员:该成员的值为 null

关于java - Lombok 和斯波克 : @RequiredArgsConstructor doesn't hide default no-args constructor for a field with a type of interface,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/53905166/

相关文章:

java - 关于使用 hashset 定义新集合的问题

Groovy - 如何将 json 转换为精确类型列表

java - 将 Lombok 插件应用到 Gradle 导致 "Could not find any public constructor"错误

java - 在 Node 中加载 Java JNA 原生库

java - Jetty 应用程序抛出 SaxParseException

java - Groovy 和 Spock 之间的冲突

java - 是否可以将 spring rest 文档与 spock 一起使用

java - 如何在使用 Lombok 生成的 IntelliJ IDEA 中的 setter 方法中设置断点?

java - Findbugs 使用 Lombok 构建器创建问题

java - 在 Java 中,如果子类用实例子变量隐藏静态父变量,那么继承方法将使用哪个变量?