[英]Idiomatic migration from Java to Kotlin of class that accesses inner class private members
[英]Private members variables in Data Class in Kotlin
我是 kotlin 的新手,当我阅读 kotlin 中的数据类时,我发现了这些代码。它基本上是 java 模型类和 kotlin 数据类之间的比较,并且写在那里,这两个代码都执行相同的任务。
代码 1
public class VideoGame {
private String name;
private String publisher;
private int reviewScore;
public VideoGame(String name, String publisher, int reviewScore) {
this.name = name;
this.publisher = publisher;
this.reviewScore = reviewScore;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public String getPublisher() {
return publisher;
}
public void setPublisher(String publisher) {
this.publisher = publisher;
}
public int getReviewScore() {
return reviewScore;
}
public void setReviewScore(int reviewScore) {
this.reviewScore = reviewScore;
}}
代码 2
data class VideoGame(val name: String, val publisher: String, var reviewScore: Int)
我的问题是,在数据类中,所有变量都是公共的而不是私有的,任何人都可以在此类对象的帮助下直接访问该变量,但在 java 代码中,所有变量都是私有的,这就是为什么我们必须编写这些成员变量的 getter 和 setter 如果是公开的,则不需要 getter 和 setter。那么这些代码是如何相同的呢?
data class
主要用途是简化编写POJO的工作。
如果您进入 IDE 并编写VideoGame
类:
data class VideoGame(val name: String, val publisher: String, var reviewScore: Int)
然后你反编译它:
您的 IDE 工具栏 > 工具 > Kotlin > 显示 Kotlin 字节码 > 反编译
你得到这个:
@Metadata(
mv = {1, 1, 16},
bv = {1, 0, 3},
k = 1,
d1 = {"\u0000\"\n\u0002\u0018\u0002\n\u0002\u0010\u0000\n\u0000\n\u0002\u0010\u000e\n\u0002\b\u0002\n\u0002\u0010\b\n\u0002\b\r\n\u0002\u0010\u000b\n\u0002\b\u0004\b\u0086\b\u0018\u00002\u00020\u0001B\u001d\u0012\u0006\u0010\u0002\u001a\u00020\u0003\u0012\u0006\u0010\u0004\u001a\u00020\u0003\u0012\u0006\u0010\u0005\u001a\u00020\u0006¢\u0006\u0002\u0010\u0007J\t\u0010\u000f\u001a\u00020\u0003HÆ\u0003J\t\u0010\u0010\u001a\u00020\u0003HÆ\u0003J\t\u0010\u0011\u001a\u00020\u0006HÆ\u0003J'\u0010\u0012\u001a\u00020\u00002\b\b\u0002\u0010\u0002\u001a\u00020\u00032\b\b\u0002\u0010\u0004\u001a\u00020\u00032\b\b\u0002\u0010\u0005\u001a\u00020\u0006HÆ\u0001J\u0013\u0010\u0013\u001a\u00020\u00142\b\u0010\u0015\u001a\u0004\u0018\u00010\u0001HÖ\u0003J\t\u0010\u0016\u001a\u00020\u0006HÖ\u0001J\t\u0010\u0017\u001a\u00020\u0003HÖ\u0001R\u0011\u0010\u0002\u001a\u00020\u0003¢\u0006\b\n\u0000\u001a\u0004\b\b\u0010\tR\u0011\u0010\u0004\u001a\u00020\u0003¢\u0006\b\n\u0000\u001a\u0004\b\n\u0010\tR\u001a\u0010\u0005\u001a\u00020\u0006X\u0086\u000e¢\u0006\u000e\n\u0000\u001a\u0004\b\u000b\u0010\f\"\u0004\b\r\u0010\u000e¨\u0006\u0018"},
d2 = {"Lcom/yourpackage/VideoGame;", "", "name", "", "publisher", "reviewScore", "", "(Ljava/lang/String;Ljava/lang/String;I)V", "getName", "()Ljava/lang/String;", "getPublisher", "getReviewScore", "()I", "setReviewScore", "(I)V", "component1", "component2", "component3", "copy", "equals", "", "other", "hashCode", "toString", "app"}
)
public final class VideoGame {
@NotNull
private final String name;
@NotNull
private final String publisher;
private int reviewScore;
@NotNull
public final String getName() {
return this.name;
}
@NotNull
public final String getPublisher() {
return this.publisher;
}
public final int getReviewScore() {
return this.reviewScore;
}
public final void setReviewScore(int var1) {
this.reviewScore = var1;
}
public VideoGame(@NotNull String name, @NotNull String publisher, int reviewScore) {
Intrinsics.checkParameterIsNotNull(name, "name");
Intrinsics.checkParameterIsNotNull(publisher, "publisher");
super();
this.name = name;
this.publisher = publisher;
this.reviewScore = reviewScore;
}
@NotNull
public final String component1() {
return this.name;
}
@NotNull
public final String component2() {
return this.publisher;
}
public final int component3() {
return this.reviewScore;
}
@NotNull
public final VideoGame copy(@NotNull String name, @NotNull String publisher, int reviewScore) {
Intrinsics.checkParameterIsNotNull(name, "name");
Intrinsics.checkParameterIsNotNull(publisher, "publisher");
return new VideoGame(name, publisher, reviewScore);
}
// $FF: synthetic method
public static VideoGame copy$default(VideoGame var0, String var1, String var2, int var3, int var4, Object var5) {
if ((var4 & 1) != 0) {
var1 = var0.name;
}
if ((var4 & 2) != 0) {
var2 = var0.publisher;
}
if ((var4 & 4) != 0) {
var3 = var0.reviewScore;
}
return var0.copy(var1, var2, var3);
}
@NotNull
public String toString() {
return "VideoGame(name=" + this.name + ", publisher=" + this.publisher + ", reviewScore=" + this.reviewScore + ")";
}
public int hashCode() {
String var10000 = this.name;
int var1 = (var10000 != null ? var10000.hashCode() : 0) * 31;
String var10001 = this.publisher;
return (var1 + (var10001 != null ? var10001.hashCode() : 0)) * 31 + this.reviewScore;
}
public boolean equals(@Nullable Object var1) {
if (this != var1) {
if (var1 instanceof VideoGame) {
VideoGame var2 = (VideoGame)var1;
if (Intrinsics.areEqual(this.name, var2.name) && Intrinsics.areEqual(this.publisher, var2.publisher) && this.reviewScore == var2.reviewScore) {
return true;
}
}
return false;
} else {
return true;
}
}
}
所以支持字段是私有的。 Kotlin 正在为您完成所有繁琐而繁重的工作。 那是:
toString
基本实现equals
实现hashCode
实现copy
方法你会得到 82 行臃肿的代码,你可能会用 Java 自己编写这些代码(可能会在其中添加一些错误?),以换取 1 个关键字。 是不是很棒?
data class
使编写 POJO 变得更容易val
告诉编译器为给定的属性实现一个 getter。 该属性在类本身内也是不可变的。 (您不能在属于该类的函数中更改它)var
告诉编译器提供 setter 和 getterdata class
设置属性的可见性,例如: data class MyClass(private val prop: Int)
进一步的例子:
您在 val 和 var 前面使用的关键字只会影响 getter 和 setter 的可见性。
例如:公共获取但私有设置:
class VideoGame {
var pegiRating: Int = 0
private set
}
那些不做同样的任务,因为 Java 版本有 getter 而 Kotlin 版本没有,因为属性是val
而不是var
。 如果它们是var
,那么功能就会全部存在(此外,Kotlin 数据类还具有copy
功能,并且为您实现了equals
、 hashcode
和toString
)。
默认情况下(没有自定义的 getter/setter),Kotlin 属性(公共或非公共)就像一个 Java 私有字段,带有(公共或非)getter 和/或 setter。
在 Java 中,通常不建议将字段设为公开,以便其他对象可以直接修改它们。 这不是面向未来的,因为如果您决定例如在更改值时产生副作用,则必须将该字段更改为私有并添加一个 setter。 这会破坏任何与该类一起工作的代码。 为了避免这种情况,字段应该是私有的,getter 和 setter 可以是公共的,不幸的是,这是很多样板代码。
在 Kotlin 中,您可以使用自定义 setter 将默认属性更改为一个,并且不会破坏使用它的代码。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.