java - 为什么 Kotlins 类型推断会失败,而 Javas 却不会?

标签 java generics kotlin type-inference

给出以下代码(使用虚拟返回但显示问题):

import com.github.roookeee.datus.api.Datus
import com.github.roookeee.datus.api.Mapper
import com.github.roookeee.datus.immutable.ConstructorParameter

data class EntryDto(val id: Long?, val title: String, val content: String)
data class EntryEntity(val id: Long? = null, val title: String, val content: String) {
    fun toDto(): EntryDto {
        val mapper: Mapper<EntryEntity, EntryDto> = Datus.forTypes(this.javaClass, EntryDto::class.java)
                .immutable(::EntryDto)
                .from(EntryEntity::id).to(ConstructorParameter::bind)
                .from(EntryEntity::title).to(ConstructorParameter::bind)
                .from(EntryEntity::content).to(ConstructorParameter::bind)
                .build()
        return EntryDto(null, "", "")
    }
}

Kotlin 无法推断出正确的泛型类型,而 Java >= 8 可以推断出正确的泛型类型(假设两个 Java 类与此处的数据类相同 - 两个不可变对象(immutable对象)类)。我尝试使用 Kotlin 1.3.0 和 -XXLanguage:+NewInference 的默认值但后者甚至无法推断出正确的重载来选择 .immutable .

这是使上述代码编译的数据依赖信息(如果没有库,我无法减少这个问题,它的通用用法太复杂了):

<dependency>
    <groupId>com.github.roookeee</groupId>
    <artifactId>datus</artifactId>
    <version>1.3.0</version>
</dependency>

我错过了什么吗?我想让我的库与 kotlin 更加兼容,但我不知道如何从这里开始或推理错误的确切名称是什么。

您可以找到数据来源here .

这是对应的java代码:

import com.github.roookeee.datus.api.Datus;
import com.github.roookeee.datus.api.Mapper;
import com.github.roookeee.datus.immutable.ConstructorParameter;

class Java8Code {
    static class EntryDto {
        private final Long id;
        private final String title;
        private final String content;

        EntryDto(Long id, String title, String content) {
            this.id = id;
            this.title = title;
            this.content = content;
        }

        public Long getId() {
            return id;
        }

        public String getTitle() {
            return title;
        }

        public String getContent() {
            return content;
        }
    }

    static class EntryEntity {
        private final Long id;
        private final String title;
        private final String content;

        EntryEntity(Long id, String title, String content) {
            this.id = id;
            this.title = title;
            this.content = content;
        }

        public Long getId() {
            return id;
        }

        public String getTitle() {
            return title;
        }

        public String getContent() {
            return content;
        }

        public EntryDto toDto() {
            Mapper<EntryEntity, EntryDto> mapper = Datus.forTypes(EntryEntity.class, EntryDto.class)
                    .immutable(EntryDto::new)
                    .from(EntryEntity::getId).to(ConstructorParameter::bind)
                    .from(EntryEntity::getTitle).to(ConstructorParameter::bind)
                    .from(EntryEntity::getContent).to(ConstructorParameter::bind)
                    .build();
            return mapper.convert(this);
        }
    }

}

编辑2:错误消息的图像+下面的一些注释 enter image description here

3 type arguments expected for interface ConstructorParameter<In : Any!, GetterReturnType : Any!, Result : Any!> - Kotlin 似乎期望接口(interface)方法引用使用泛型类型参数,但这在 Kotlin 中根本不可能,在 Java 中也不需要。

最佳答案

to 扩展函数不是问题。它只是弹出,因为编译器没有看到成员方法的正确参数。

Kotlin 不喜欢的是没有附加类型参数的泛型类型。在指定类型参数之前,ConstructorParameter 不是一个类型。因此,当它看到 ConstructorParameter::bind 时,它会提示,:: 的左侧应该是一个类型。

如果您编写 .to { x, y -> x.bind(y) },Kotlin 可以很好地推断出类型。 但是您不能指望您的用户多次编写这个“相同”的 lambda。

扩展函数来救援!

fun <In, CurrentType, Next> ConstructorParameterBinding<In, CurrentType, out ConstructorParameter<In, CurrentType, Next>>.bind(): Next =
    this.to { x, y -> x.bind(y) }

val mapper: Mapper<EntryEntity, EntryDto> = Datus.forTypes(this.javaClass, EntryDto::class.java)
    .immutable(::EntryDto)
    .from(EntryEntity::id).bind()
    .from(EntryEntity::title).bind()
    .from(EntryEntity::content).bind()
    .build()

关于java - 为什么 Kotlins 类型推断会失败,而 Javas 却不会?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/57316688/

相关文章:

java - 如何创建一个水平滚动条,它会在一段时间间隔后自动滚动

java - 扩展 Collection 时如何在 'this' 上使用 Iterable?

c# 另一个泛型的泛型类

android - 为什么我的数据库不起作用,我在官方文档中使用的方式?

android - 错误: Class is referenced as a converter but it does not have any converter methods

Spring(kotlin) Controller 接收数据类的不可为空的参数为空

java - Spring CORS。在允许的来源中添加模式

java - 如何对 NatTable 中的每一列使用不同的比较器对 TreeList 进行排序

java - 无法打开我在 eclipse 中保存的 java 程序

arrays - 强制 TypeScript 数组包含给定值的元素