首页
学习
活动
专区
圈层
工具
发布
社区首页 >专栏 >【Kotlin】常用的 Kotlin 类 ① ( 嵌套类 | 数据类 | 数据类 copy 函数 | 数据类解构声明 operator fun component1 | 数据类运算符重载 )

【Kotlin】常用的 Kotlin 类 ① ( 嵌套类 | 数据类 | 数据类 copy 函数 | 数据类解构声明 operator fun component1 | 数据类运算符重载 )

作者头像
韩曙亮
发布2023-03-30 18:54:19
发布2023-03-30 18:54:19
1.3K0
举报

文章目录

一、嵌套类


嵌套类 指的是 在 类 A 中 定义 类 B , 一般是 类 B 对 类 A 有一定的作用 , 将 类 B 嵌套进 类 A 中 ; 格式如下 :

代码语言:javascript
复制
class A {
	class B {
	}
}

在 类 A 中调用 B , 可以直接使用 B() 进行调用 ;

在 外部调用 类 B 时 , 可以使用 A.B() 的形式进行调用 ;

在 Java 中内部类有 成员内部类 , 静态嵌套类 , 方法内部类 , 匿名内部类 几种类型 ;

Kotlin 嵌套类 , 相当于 Java 中的 静态嵌套类 ;

代码示例 : 在下面的代码中 , 在 Person 类内部中定义 Student 嵌套类 , 在 外部使用 Person.Student 使用其内部的 Student 嵌套类 ;

代码语言:javascript
复制
open class Person(val name: String, val age: Int) {
    class Student(val school: String) {
        fun goSchool() {
            println("去 ${school} 上学")
        }
    }
}

fun main() {
    Person.Student("小学").goSchool()
}

执行结果 :

代码语言:javascript
复制
去 小学 上学

二、数据类


数据类型定义 : Kotlin 中的 数据类型 是 专门用于存储 数据的 类 , 一般该类中不定义 成员方法 ;

数据类信息 : 数据类 中 自动提供了 toString 实现 , 可以将数据转为 字符串 ;

数据类型对比 : == 运算符 对比两个数数据类对比的是引用地址 , equals 和 hashCode 函数 可以对比具体的数据值 ;

在 Kotlin 中 , 一般情况下 == 比较的是内容 , === 比较的是引用 ;

== 运算符相当于调用的是 equals 方法 , 只要重写了 equals 方法 , == 比较的就是内容 ;

在 Any 超类中 , 没有重写 equals 方法 , == 对比的是 引用地址 ;

数据类定义形式 :

代码语言:javascript
复制
data class 数据类类名(数据类型参数) {}

代码示例 :

代码语言:javascript
复制
data class Student(var name: String, var age: Int)

fun main() {
    println(Student("Tom", 18))

    // 在 Kotlin 中 , 一般情况下 == 比较的是内容 , === 比较的是引用
    println(Student("Tom", 18) == Student("Tom", 18))
}

执行结果 : 定义上述数据类 , 打印该数据类实例对象 , 得到的不是地址信息 , 而是数据类的实际值 ;

代码语言:javascript
复制
Student(name=Tom, age=18)
true

查看字节码信息 , 双击 Shift , 选择 Show Kotlin Bytecode 选项 ,

在 Kotlin Bytecode 界面 中 , 选择 Decompile 选项 , 将 字节码数据 反编译为 Java 代码数据 ,

发现 Kotlin 编译器 为 Student 数据类 自动生成了一个 toString 方法 , 将其数据打印出来 ;

同时还 重写 Student 数据类 中的 equals 和 hashCode 函数 ;

代码语言:javascript
复制
// Student.java
import kotlin.Metadata;
import kotlin.jvm.internal.Intrinsics;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;

@Metadata(
   mv = {1, 4, 2},
   bv = {1, 0, 3},
   k = 1,
   d1 = {"\u0000 \n\u0002\u0018\u0002\n\u0002\u0010\u0000\n\u0000\n\u0002\u0010\u000e\n\u0000\n\u0002\u0010\b\n\u0002\b\r\n\u0002\u0010\u000b\n\u0002\b\u0004\b\u0086\b\u0018\u00002\u00020\u0001B\u0015\u0012\u0006\u0010\u0002\u001a\u00020\u0003\u0012\u0006\u0010\u0004\u001a\u00020\u0005¢\u0006\u0002\u0010\u0006J\t\u0010\u000f\u001a\u00020\u0003HÆ\u0003J\t\u0010\u0010\u001a\u00020\u0005HÆ\u0003J\u001d\u0010\u0011\u001a\u00020\u00002\b\b\u0002\u0010\u0002\u001a\u00020\u00032\b\b\u0002\u0010\u0004\u001a\u00020\u0005HÆ\u0001J\u0013\u0010\u0012\u001a\u00020\u00132\b\u0010\u0014\u001a\u0004\u0018\u00010\u0001HÖ\u0003J\t\u0010\u0015\u001a\u00020\u0005HÖ\u0001J\t\u0010\u0016\u001a\u00020\u0003HÖ\u0001R\u001a\u0010\u0004\u001a\u00020\u0005X\u0086\u000e¢\u0006\u000e\n\u0000\u001a\u0004\b\u0007\u0010\b\"\u0004\b\t\u0010\nR\u001a\u0010\u0002\u001a\u00020\u0003X\u0086\u000e¢\u0006\u000e\n\u0000\u001a\u0004\b\u000b\u0010\f\"\u0004\b\r\u0010\u000e¨\u0006\u0017"},
   d2 = {"LStudent;", "", "name", "", "age", "", "(Ljava/lang/String;I)V", "getAge", "()I", "setAge", "(I)V", "getName", "()Ljava/lang/String;", "setName", "(Ljava/lang/String;)V", "component1", "component2", "copy", "equals", "", "other", "hashCode", "toString", "KotlinDemo"}
)
public final class Student {
   @NotNull
   private String name;
   private int age;

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

   public final void setName(@NotNull String var1) {
      Intrinsics.checkNotNullParameter(var1, "<set-?>");
      this.name = var1;
   }

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

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

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

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

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

   @NotNull
   public final Student copy(@NotNull String name, int age) {
      Intrinsics.checkNotNullParameter(name, "name");
      return new Student(name, age);
   }

   // $FF: synthetic method
   public static Student copy$default(Student var0, String var1, int var2, int var3, Object var4) {
      if ((var3 & 1) != 0) {
         var1 = var0.name;
      }

      if ((var3 & 2) != 0) {
         var2 = var0.age;
      }

      return var0.copy(var1, var2);
   }

   @NotNull
   public String toString() {
      return "Student(name=" + this.name + ", age=" + this.age + ")";
   }

   public int hashCode() {
      String var10000 = this.name;
      return (var10000 != null ? var10000.hashCode() : 0) * 31 + this.age;
   }

   public boolean equals(@Nullable Object var1) {
      if (this != var1) {
         if (var1 instanceof Student) {
            Student var2 = (Student)var1;
            if (Intrinsics.areEqual(this.name, var2.name) && this.age == var2.age) {
               return true;
            }
         }

         return false;
      } else {
         return true;
      }
   }
}
// HelloKt.java
import kotlin.Metadata;

@Metadata(
   mv = {1, 4, 2},
   bv = {1, 0, 3},
   k = 2,
   d1 = {"\u0000\b\n\u0000\n\u0002\u0010\u0002\n\u0000\u001a\u0006\u0010\u0000\u001a\u00020\u0001¨\u0006\u0002"},
   d2 = {"main", "", "KotlinDemo"}
)
public final class HelloKt {
   public static final void main() {
      Student var0 = new Student("Tom", 18);
      boolean var1 = false;
      System.out.println(var0);
   }

   // $FF: synthetic method
   public static void main(String[] var0) {
      main();
   }
}

三、数据类 copy 函数


Kotlin 编译器不仅为 数据类 重写了 toString , equals , hashCode 函数 , 还 提供了 copy 函数 , 借助该函数可以 快速创建一个相同内容的数据类 ;

特别注意 : 数据类 copy 函数调用的是主构造函数 , 如果数据类有属性是在次构造函数中赋值的 , 则该数据不会被 copy 函数复制 ;

在上个章节查看的 数据类 字节码 反编译后的 Java 代码 , 查看其 copy 函数 , 内容如下 :

代码语言:javascript
复制
   // $FF: synthetic method
   public static Student copy$default(Student var0, String var1, int var2, int var3, Object var4) {
      if ((var3 & 1) != 0) {
         var1 = var0.name;
      }

      if ((var3 & 2) != 0) {
         var2 = var0.age;
      }

      return var0.copy(var1, var2);
   }

代码示例 :

代码语言:javascript
复制
data class Student(var name: String, var age: Int)

fun main() {
    val student = Student("Tom", 18)
    println(student)

    // 拷贝数据类, name 属性设置为 Jerry
    val student2 = student.copy(name = "Jerry")
    println(student2)
}

执行结果 :

代码语言:javascript
复制
Student(name=Tom, age=18)
Student(name=Jerry, age=18)

四、数据类解构声明


在之前的博客 【Kotlin】集合操作 ③ ( List 集合遍历 | for in | forEach | forEachIndexed | List 通过解构一次性给多个元素赋值 ) 中介绍了 , 使用集合一次性给多个变量赋值 ;

Kotlin 普通类 和 数据类 都可以 支持 解构语法 , 为多个变量进行赋值 ;

数据类 自带 支持解构语法的特性 , 不需要使用 operator fun component1() 解构声明 ;

1、Kotlin 普通类解构声明 operator fun component1

在 普通 Kotlin 类中使用 如下声明 , 即可支持解构语法 ;

代码语言:javascript
复制
operator fun component1() = 成员属性名1
operator fun component2() = 成员属性名2
operator fun component3() = 成员属性名3
operator fun component4() = 成员属性名4
...
...

代码示例 :

代码语言:javascript
复制
class Student(var name: String, var age: Int) {
    operator fun component1() = name
    operator fun component2() = age
}

fun main() {
    val student = Student("Tom", 18)
    println(student)

    val (name, age) = student
    println("name = $name, age = $age")
}

执行结果 :

代码语言:javascript
复制
Student@61bbe9ba
name = Tom, age = 18

2、数据类解构声明

数据类 自带 支持解构语法的特性 , 不需要使用 operator fun component1() 解构声明 ;

代码示例 :

代码语言:javascript
复制
data class Student(var name: String, var age: Int)

fun main() {
    val student = Student("Tom", 18)
    println(student)

    val (name, age) = student
    println("name = $name, age = $age")
}

执行结果 :

代码语言:javascript
复制
Student(name=Tom, age=18)
name = Tom, age = 18

查看字节码信息 , 双击 Shift , 选择 Show Kotlin Bytecode 选项 ,

在 Kotlin Bytecode 界面 中 , 选择 Decompile 选项 , 将 字节码数据 反编译为 Java 代码数据 ,

反编译后的 Java 代码数据如下 :

代码语言:javascript
复制
// Student.java
import kotlin.Metadata;
import kotlin.jvm.internal.Intrinsics;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;

@Metadata(
   mv = {1, 4, 2},
   bv = {1, 0, 3},
   k = 1,
   d1 = {"\u0000 \n\u0002\u0018\u0002\n\u0002\u0010\u0000\n\u0000\n\u0002\u0010\u000e\n\u0000\n\u0002\u0010\b\n\u0002\b\r\n\u0002\u0010\u000b\n\u0002\b\u0004\b\u0086\b\u0018\u00002\u00020\u0001B\u0015\u0012\u0006\u0010\u0002\u001a\u00020\u0003\u0012\u0006\u0010\u0004\u001a\u00020\u0005¢\u0006\u0002\u0010\u0006J\t\u0010\u000f\u001a\u00020\u0003HÆ\u0003J\t\u0010\u0010\u001a\u00020\u0005HÆ\u0003J\u001d\u0010\u0011\u001a\u00020\u00002\b\b\u0002\u0010\u0002\u001a\u00020\u00032\b\b\u0002\u0010\u0004\u001a\u00020\u0005HÆ\u0001J\u0013\u0010\u0012\u001a\u00020\u00132\b\u0010\u0014\u001a\u0004\u0018\u00010\u0001HÖ\u0003J\t\u0010\u0015\u001a\u00020\u0005HÖ\u0001J\t\u0010\u0016\u001a\u00020\u0003HÖ\u0001R\u001a\u0010\u0004\u001a\u00020\u0005X\u0086\u000e¢\u0006\u000e\n\u0000\u001a\u0004\b\u0007\u0010\b\"\u0004\b\t\u0010\nR\u001a\u0010\u0002\u001a\u00020\u0003X\u0086\u000e¢\u0006\u000e\n\u0000\u001a\u0004\b\u000b\u0010\f\"\u0004\b\r\u0010\u000e¨\u0006\u0017"},
   d2 = {"LStudent;", "", "name", "", "age", "", "(Ljava/lang/String;I)V", "getAge", "()I", "setAge", "(I)V", "getName", "()Ljava/lang/String;", "setName", "(Ljava/lang/String;)V", "component1", "component2", "copy", "equals", "", "other", "hashCode", "toString", "KotlinDemo"}
)
public final class Student {
   @NotNull
   private String name;
   private int age;

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

   public final void setName(@NotNull String var1) {
      Intrinsics.checkNotNullParameter(var1, "<set-?>");
      this.name = var1;
   }

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

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

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

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

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

   @NotNull
   public final Student copy(@NotNull String name, int age) {
      Intrinsics.checkNotNullParameter(name, "name");
      return new Student(name, age);
   }

   // $FF: synthetic method
   public static Student copy$default(Student var0, String var1, int var2, int var3, Object var4) {
      if ((var3 & 1) != 0) {
         var1 = var0.name;
      }

      if ((var3 & 2) != 0) {
         var2 = var0.age;
      }

      return var0.copy(var1, var2);
   }

   @NotNull
   public String toString() {
      return "Student(name=" + this.name + ", age=" + this.age + ")";
   }

   public int hashCode() {
      String var10000 = this.name;
      return (var10000 != null ? var10000.hashCode() : 0) * 31 + this.age;
   }

   public boolean equals(@Nullable Object var1) {
      if (this != var1) {
         if (var1 instanceof Student) {
            Student var2 = (Student)var1;
            if (Intrinsics.areEqual(this.name, var2.name) && this.age == var2.age) {
               return true;
            }
         }

         return false;
      } else {
         return true;
      }
   }
}
// HelloKt.java
import kotlin.Metadata;

@Metadata(
   mv = {1, 4, 2},
   bv = {1, 0, 3},
   k = 2,
   d1 = {"\u0000\b\n\u0000\n\u0002\u0010\u0002\n\u0000\u001a\u0006\u0010\u0000\u001a\u00020\u0001¨\u0006\u0002"},
   d2 = {"main", "", "KotlinDemo"}
)
public final class HelloKt {
   public static final void main() {
      Student student = new Student("Tom", 18);
      boolean var1 = false;
      System.out.println(student);
      String name = student.component1();
      int age = student.component2();
      String var3 = "name = " + name + ", age = " + age;
      boolean var4 = false;
      System.out.println(var3);
   }

   // $FF: synthetic method
   public static void main(String[] var0) {
      main();
   }
}

关注 数据类 Student 的结构声明 , Kotlin 编译器在编译时自动给数据类添加上了结构声明 ;

代码语言:javascript
复制
   @NotNull
   public final String component1() {
      return this.name;
   }

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

五、运算符重载函数


下图是 Kotlin 运算符重载操作符对应的函数名 :

如果需要 为类添加 + 操作符 支持 , 则 需要在类中设置 plus 运算符重载函数 ;

运算符重载函数定义 :

代码语言:javascript
复制
operator fun plus(t: T): T

代码示例 :

代码语言:javascript
复制
data class Student(var name: String, var age: Int) {
    operator fun plus(other: Student): Student {
        return Student("$name, ${other.name}", age + other.age)
    }
}

fun main() {
    val student = Student("Tom", 18)
    val student2 = Student("Jerry", 12)
    println(student + student2)
}

执行结果 :

代码语言:javascript
复制
Student(name=Tom, Jerry, age=30)
本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2023-01-19,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 文章目录
  • 一、嵌套类
  • 二、数据类
  • 三、数据类 copy 函数
  • 四、数据类解构声明
    • 1、Kotlin 普通类解构声明 operator fun component1
    • 2、数据类解构声明
  • 五、运算符重载函数
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档