kotlin中var、val、const val的区别

  • var 变量 可以被读写
  • val 变量 只能读,不能写
  • const val 编译时常量
class Person {
    var name = "张三"
    val idCard = "123456789"
}

var与val 变量
上面的代码中name用var来定义的,idCard使用val关键字来定义的。现在我们从字节码的角度来看看他们之间的区别

public final class cclz.Person {
  public final java.lang.String getName();
    Code:
       0: aload_0
       1: getfield      #11                 // Field name:Ljava/lang/String;
       4: areturn

  public final void setName(java.lang.String);
    Code:
       0: aload_1
       1: ldc           #17                 // String <set-?>
       3: invokestatic  #23                 // Method kotlin/jvm/internal/Intrinsics.checkNotNullParameter:(Ljava/lang/Object;Ljava/lang/String;)V
       6: aload_0
       7: aload_1
       8: putfield      #11                 // Field name:Ljava/lang/String;
      11: return

  public final java.lang.String getIdCard();
    Code:
       0: aload_0
       1: getfield      #27                 // Field idCard:Ljava/lang/String;
       4: areturn

  public cclz.Person();
    Code:
       0: aload_0
       1: invokespecial #31                 // Method java/lang/Object."<init>":()V
       4: aload_0
       5: ldc           #33                 // String 张三
       7: putfield      #11                 // Field name:Ljava/lang/String;
      10: aload_0
      11: ldc           #35                 // String 123456789
      13: putfield      #27                 // Field idCard:Ljava/lang/String;
      16: return
}

观察上面的字节吗,你会发现name和idCard其实没有区别,只不过val修饰的idCard只有get方法没有set方法,而kotlin中对对象成员的操作实际上都是通过get set来进行的,即使我们没写对应的get set 编译器也会帮我们处理。而如果我们想给val修饰的变量添加set方法,编译器会直接不给通过,只是在编译器层面做了限制,这就是我们只能读取不能进行赋值的秘密了。但是对于jvm来说他们的类型还是一样的,。而对于编写在方法中的局部变量虽然没有get set方法,可以直接操作对应的变量。但是本质是一样的

    fun eat(){
        var str1 = "苹果"
        val str2 = "橘子"
    }

字节码如下:

  public final void eat();
    Code:
       0: ldc           #31                 // String 苹果
       2: astore_1
       3: ldc           #33                 // String 橘子
       5: astore_2
       6: return

可以看到这两个变量对于jvm来说没有区别,但是由于在编译器层面的限制,如果你尝试修改val声明的变量,编译器就会给你报错。

const val

  • 1、const val 变量申明的值必须在编译时期确定下来,因此它的类型只能是String或基本类型
  • 2 、const 只能修饰object(object单例或伴生对象)的属性,或者作为顶层的变量
const val num1  = 100
class Person {
    var name = "张三"
    fun test(){
        println(num1)
        println(name)
    }
}

对于上面的代码其直接码如下

public final class cclz.Person
Constant pool:
   #1 = Utf8               cclz/Person
   #2 = Class              #1             // cclz/Person
   #5 = Utf8               name
   #10 = NameAndType        #5:#6          // name:Ljava/lang/String;
   #11 = Fieldref           #2.#10         // cclz/Person.name:Ljava/lang/String;
   //...
  public final void test();
    descriptor: ()V
    flags: ACC_PUBLIC, ACC_FINAL
    Code:
         0: bipush        100
		//...
        13: getfield      #11                 // Field name:Ljava/lang/String;
        //...


可以看到在编译后所有对num1的引用都被替换成了字面量100,而且编译后的常量池中也找不到num1

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值