我最近读了关于const关键字的文章,我很困惑!我找不到const和val关键字之间的任何区别,我的意思是我们可以用它们来创建一个不可变变量,还有什么我遗漏的吗?


当前回答

因为我读了很多,“val”意味着不可变:这绝对不是这样的,看看这个例子:

class Test {
    var x: Int = 2
    val y
        get() = x
}

fun main(args: Array<String>) {
    val test = Test()
    println("test.y = ${test.y}") // prints 2
    test.x = 4
    println("test.y = ${test.y}") // prints 4
}

遗憾的是,真正的不可变性目前只能通过const -实现,但这只能在编译时实现。在运行时,你不能创建真正的不变性。

val只是表示“只读”,你不能直接改变这个变量,只能像我在上面的例子中显示的那样间接改变。

其他回答

再补充一下卢卡的回答:

编译时常量 在编译时已知值的属性可以使用const修饰符标记为编译时常量。这些属性需要满足以下要求: 对象声明或伴生对象的顶级或成员。 初始化为String类型或基本类型的值 没有自定义getter 这些属性可以在注释中使用。

来源:官方文件

const是编译时常量。这意味着它们的值必须在编译时赋值,而不像val那样可以在运行时赋值。

这意味着,const永远不能被赋值给函数或任何类构造函数,而只能赋值给String或原语。

例如:

const val foo = complexFunctionCall()   //Not okay
val fooVal = complexFunctionCall()  //Okay

const val bar = "Hello world"           //Also okay

您可以将Kotlin转换为Java。 然后你可以看到const比val多一个静态修饰符。 像这样的简单代码。

科特林:

const val str = "hello"
class SimplePerson(val name: String, var age: Int)

Java(部分):

@NotNull
public static final String str = "hello";

public final class SimplePerson {
   @NotNull
   private final String name;
   private int age;

   @NotNull
   public final String getName() {
      return this.name;
   }

   public final int getAge() {
      return this.age;
   }

   public final void setAge(int var1) {
      this.age = var1;
   }

   public SimplePerson(@NotNull String name, int age) {
      Intrinsics.checkParameterIsNotNull(name, "name");
      super();
      this.name = name;
      this.age = age;
   }
}

让我们通过一个例子来学习。

object Constants {

    val NAME = "Amit"

}

注意:我们没有使用const。

并且,我们正在访问这个NAME,如下所示:

fun testValWithoutConst() {
    val name = Constants.NAME
}

现在,我们需要反编译这段代码。为此,我们必须将这个Kotlin源文件转换为Java源文件。

我们将得到以下输出:

public final void testValWithoutConst() {
  String name = Constants.INSTANCE.getNAME();
}

输出与预期一致。

上面的例子没有使用const关键字。现在,我们使用const关键字。

为此,我们将修改Kotlin中的对象类Constants:

object Constants {

    const val NAME = "Amit"

}

注意:我们使用的是const。

并且,我们正在访问这个NAME,如下所示:

fun testValWithConst() {
    val name = Constants.NAME
}

现在,当我们反编译这段代码时,我们将得到以下输出:

public final void testValWithConst() {
  String name = "Amit";
}

在这里,我们可以看到变量NAME已经被它的值Amit所取代。

由于值已内联,因此在运行时访问该变量不会产生任何开销。因此,它将导致应用程序的更好性能。

这就是在Kotlin中使用const的优势。

引用自我的博客:在Kotlin中使用const的优势

const kotlin到Java

const val Car_1 = "BUGATTI" // final static String Car_1 = "BUGATTI";

val kotlin to Java

val Car_1 = "BUGATTI"   // final String Car_1 = "BUGATTI";

简单地说

const变量的值在编译时已知。 val的值用于在运行时定义常量。

示例1 -

const val Car_1 = "BUGATTI" ✔  
val Car_2 = getCar() ✔    
const val Car_3 = getCar() ❌ 

//Because the function will not get executed at the compile time so it will through error

fun getCar(): String {
    return "BUGATTI"
}

这是因为getCar()在运行时被求值并将值赋给Car。

此外,

Val是只读的意味着在运行时是不可变的 Var是在运行时已知的可变变量 Const是不可变的,并且是编译时已知的变量