在前面几个章节章节中,详细的讲解了Koltin中的接口类(Interface)枚举类(Enmu),还不甚了解的可以查看我的上一篇文章Kotlin——中级篇(五):枚举类(Enum)、接口类(Interface)详解。当然,在Koltin中,除了接口类、枚举类之外,还有抽象类、内部类、数据类以及密封类。在今天的章节中,为大家详细讲解数据类密封类。在下一章节中,再为大家奉上Kotlin中的抽象类以及内部类的知识。如果还对Kotlin的分类还不清楚的可以查看我的另一篇博文Kotlin——中级篇(一):类(class)详解

目录

一、数据类

  • Java中,或者在我们平时的Android开发中,为了解析后台人员给我们提供的接口返回的Json字符串,我们会根据这个字符串去创建一个或者实例对象,在这个类中,只包含了一些我们需要的数据,以及为了处理这些数据而所编写的方法。这样的类,在Kotlin中就被称为数据类

1、关键字

声明数据类的关键字为:data

1.1、声明格式

data class 类名(var param1 :数据类型,...){}

或者

data class 类名 可见性修饰符 constructor(var param1 : 数据类型 = 默认值,...)

说明:

  • data为声明数据类的关键字,必须书写在class关键字之前。
  • 在没有结构体的时候,大括号{}可省略。
  • 构造函数中必须存在至少一个参数,并且必须使用valvar修饰。这一点在下面数据类特性中会详细讲解。
  • 参数的默认值可有可无。(若要实例一个无参数的数据类,则就要用到默认值)

例:

// 定义一个名为Person的数据类
data class Preson(var name : String,val sex : Int, var age : Int)

1.2、约定俗成的规定

  • 数据类也有其约定俗成的一些规定,这只是为增加代码的阅读性。

即,当构造函数中的参过多时,为了代码的阅读性,一个参数的定义占据一行。

例:

data class Person(var param1: String = "param1",
var param2: String = "param2",
var param3 : String,
var param4 : Long,
var param5 : Int = 2,
var param6 : String,
var param7 : Float = 3.14f,
var param8 : Int,
var param9 : String){
// exp
.
.
.
}

1.3、编辑器为我们做的事情

当我们声明一个数据类时,编辑器自动为这个类做了一些事情,不然它怎么又比Java简洁呢。它会根据主构造函数中所定义的所有属性自动生成下列方法:

  • 生成equals()函数与hasCode()函数
  • 生成toString()函数,由类名(参数1 = 值1,参数2 = 值2,....)构成
  • 由所定义的属性自动生成component1()、component2()、...、componentN()函数,其对应于属性的声明顺序。
  • copy()函数。在下面会实例讲解它的作用。

其中,当这些函数中的任何一个在类体中显式定义或继承自其基类型,则不会生成该函数

2、数据类的特性

数据类有着和Kotlin其他类不一样的特性。除了含有其他类的一些特性外,还有着其独特的特点。并且也是数据类必须满足的条件:

  • 主构造函数需要至少有一个参数
  • 主构造函数的所有参数需要标记为 val 或 var;
  • 数据类不能是抽象、开放、密封或者内部的;
  • 数据类是可以实现接口的,如(序列化接口),同时也是可以继承其他类的,如继承自一个密封类。

3、用实例说明其比Java的简洁性

3.1、数据类的对比

Kotlin版:

data class User(val name : String, val pwd : String)

Java版:

public class User {
private String name;
private String pwd; public User(){} public User(String name, String pwd) {
this.name = name;
this.pwd = pwd;
} public String getName() {
return name;
} public void setName(String name) {
this.name = name;
} public String getPwd() {
return pwd;
} public void setPwd(String pwd) {
this.pwd = pwd;
} @Override
public String toString() {
return "User{" +
"name='" + name + '\'' +
", pwd='" + pwd + '\'' +
'}';
}
}

分析:实现同一个功能,从代码量来说,KoltinJava少了很多行代码,比起更简洁。

3.2、修改数据类属性

例:修改User类的name属性

Kotlin版:

  • Koltin要修改数据类的属性,则使用其独有的copy()函数。其作用就是:修改部分属性,但是保持其他不变
val mUser = User("kotlin","123456")
println(mUser)
val mNewUser = mUser.copy(name = "new Kotlin")
println(mNewUser)

输出结果为:

User(name=kotlin, pwd=123456)
User(name=new Kotlin, pwd=123456)

Java版:

User mUser = new User("Java","123456");
System.out.println(mUser);
mUser.setName("new Java");
System.out.println(mUser);

输出结果为:

User{name='Java', pwd='123456'}
User{name='new Java', pwd='123456'}

分析:从上面对两种方式的实现中可以看出,Kotlin是使用其独有的copy()函数去修改属性值,而Java是使用setXXX()去修改

4、解构声明

  • 在前面讲到,Kotlin中定义一个数据类,则系统会默认自动根据参数的个数生成component1() ... componentN()函数。其...,componentN()函数就是用于解构声明的
val mUser = User("kotlin","123456")
val (name,pwd) = mUser
println("name = $name\tpwd = $pwd")

输出结果为:

name = kotlin	pwd = 123456

5、系统标准库中的标准数据类

  • 标准库提供了 Pair 和 Triple。尽管在很多情况下命名数据类是更好的设计选择, 因为它们通过为属性提供有意义的名称使代码更具可读性。
  • 其实这两个类的源码部分不多,故而贴出这个类的源代码来分析分析

5.1、源码分析

@file:kotlin.jvm.JvmName("TuplesKt")
package kotlin // 这里去掉了源码中的注释
public data class Pair<out A, out B>(
public val first: A,
public val second: B) : Serializable { // toString()方法
public override fun toString(): String = "($first, $second)"
} // 转换
public infix fun <A, B> A.to(that: B): Pair<A, B> = Pair(this, that) // 转换成List集合
public fun <T> Pair<T, T>.toList(): List<T> = listOf(first, second) // 这里去掉了源码中的注释
public data class Triple<out A, out B, out C>(
public val first: A,
public val second: B,
public val third: C ) : Serializable { // toString()方法
public override fun toString(): String = "($first, $second, $third)"
} // 转换成List集合
public fun <T> Triple<T, T, T>.toList(): List<T> = listOf(first, second, third)

分析:从上面的源码可以看出,标准库中提供了两个标准的数据类,Pair类以及Triple类.其中:

  • 两个类中都实现了toList()方法以及toString()方法。
  • to()方法乃Pair类特有,起作用是参数转换
  • Pair类需要传递两个参数,Triple类需要传递三个参数。

5.2、用法

val pair = Pair(1,2)        // 实例
val triple = Triple(1,2,3) // 实例
println("$pair \t $triple") // 打印:即调用了各自的toString()方法
println(pair.toList()) // 转换成List集合
println(triple.toList()) // 转换成List集合
println(pair.to(3)) // Pair类特有: 其作用是把参数Pair类中的第二个参数替换

输出结果为:

(1, 2) 	 (1, 2, 3)
[1, 2]
[1, 2, 3]
((1, 2), 3)

二、密封类

密封类是用来表示受限的类继承结构。若还不甚清楚Kotlin的类继承,请参见我的上一篇文章Kotlin——中级篇(四):继承类详解

1、什么是受限的类继承结构

  • 所谓受限的类继承结构,即当类中的一个值只能是有限的几种类型,而不能是其他的任何类型。
  • 这种受限的类继承结构从某种意义上讲,它相当于是枚举类的扩展。但是,我们知道Kotlin的枚举类中的枚举常量是受限的,因为每一个枚举常量只能存在一个实例。若对Kotlin中的枚举类不甚了解的,请参见我的另一篇文章Kotlin——中级篇(五):枚举类(Enum)、接口类(Interface)详解
  • 但是其和枚举类不同的地方在于,密封类的一个子类可以有可包含状态的多个实例。
  • 也可以说成,密封类是包含了一组受限的类集合,因为里面的类都是继承自这个密封类的。但是其和其他继承类(open)的区别在,密封类可以不被此文件外被继承,有效保护代码。但是,其密封类的子类的扩展是是可以在程序中任何位置的,即可以不再统一文件下。

上面的几点内容是密封类的特点,请详细的看下去,小生会对这几点内容进行详细的分析。

2、关键字

定义密封类的关键字:sealed

2.1、声明格式

sealed class SealedExpr()

注意:密封类是不能被实例化的

val mSealedExpr = SealedExpr()  // 这段代码是错误的,编译器直接会报错不能编译通过。

既然密封类是不能实例化,那么我们要怎么使用,或者说它的作用是什么呢?请继续往下看

3、密封类的作用及其详细用法。

3.1、作用

用来表示受限的类继承结构。

例:

sealed class SealedExpr{
data class Person(val num1 : Int, val num2 : Int) : SealedExpr() object Add : SealedExpr() // 单例模式
object Minus : SealedExpr() // 单例模式
} // 其子类可以定在密封类外部,但是必须在同一文件中 v1.1之前只能定义在密封类内部
object NotANumber : SealedExpr()

分析:即所定义的子类都必须继承于密封类,表示一组受限的类

3.2、和普通继承类的区别

  • 我们知道普通的继承类使用open关键字定义,在项目中的类都可集成至该类。如果你对Koltin的继承类还不甚了解。请参见我的另一篇文章Kotlin——中级篇(四):继承类详解
  • 而密封类的子类必须是在密封类的内部或必须存在于密封类的同一文件。这一点就是上面提到的有效的代码保护。

3.3、和枚举类的区别

  • 枚举类的中的每一个枚举常量都只能存在一个实例。而密封类的子类可以存在多个实例。

例:

val mPerson1 = SealedExpr.Person("name1",22)
println(mPerson1) val mPerson2 = SealedExpr.Person("name2",23)
println(mPerson2) println(mPerson1.hashCode())
println(mPerson2.hashCode())

输出结果为:

Person(name=name1, age=22)
Person(name=name2, age=23)
-1052833328
-1052833296

3.4、其子类的类扩展实例

  • Kotlin支持扩展功能,其和C#Go语言类似。这一点是Java没有的。如果你还对Koltin中的扩展功能还不甚清楚的。请参见我的另一篇博文Kotlin——扩展功能详解

为了演示密封类的子类的扩展是可以在项目中的任何位置这个功能,大家可以下载源码。源码链接在文章末尾会为大家奉上。

例:

// 其存在于SealedClassDemo.kt文件中

sealed class SealedExpr{
data class Person(val name : String, val age : Int) : SealedExpr()
object Add : SealedExpr()
companion object Minus : SealedExpr()
} object NotANumber : SealedExpr() 其存在TestSealedDemo.kt文件中 fun <T>SealedExpr.Add.add(num1 : T, num2 : T) : Int{
return 100
} fun main(args: Array<String>) {
println(SealedExpr.Add.add(1,2))
}

输出结果为:

100

说明:上面的扩展功能没有任何的意义,只是为了给大家展示密封类子类的扩展不局限与密封类同文件这一个功能而已。如果你还对Koltin中的扩展功能还不甚清楚的。请参见我的另一篇博文Kotlin——扩展功能详解

3.5、使用密封类的好处

  • 有效的保护代码(上面已说明原因)
  • 在使用when表达式 的时候,如果能够验证语句覆盖了所有情况,就不需要为该语句再添加一个else子句了。

例:

sealed class SealedExpr{
data class Person(val name : String, val age : Int) : SealedExpr()
object Add : SealedExpr()
companion object Minus : SealedExpr()
} object NotANumber : SealedExpr() fun eval(expr: SealedExpr) = when(expr){
is SealedExpr.Add -> println("is Add")
is SealedExpr.Minus -> println("is Minus")
is SealedExpr.Person -> println(SealedExpr.Person("Koltin",22))
NotANumber -> Double.NaN
}

输出结果为:

is Minus

三、总结

在实际的项目开发当中,数据类(data)类的用处是很多的,因为在开发APP时,往往会根据后台开发者所提供的接口返回的json而生成一个实体类,现在我们学习了数据类后,就不用再像Java一样写那么多代码了,即使是用编辑器提供的方法去自动生成。但是代码量上就能节省我们很多时间,并且也更加简洁。何乐而不为呢!密封类的情况在实际开发中不是很常见的。只有当时特殊的需求会用到的时候,才会使用密封类。当然我们还是要学习的。

源代码

如果各位大佬看了之后感觉还阔以,就请各位大佬随便star一下,您的关注是我最大的动力。

我的个人博客Jetictors

我的githubJetictors

我的掘金Jetictors

欢迎各位大佬进群共同研究、探索

QQ群号:497071402

Kotlin——最详细的数据类、密封类详解的更多相关文章

  1. 【转】 个人认为,这是最详细的 android------HttpURLConnection 类用法详解。一些教材没讲到的,它讲到了

    站在巨人的肩膀上,渐渐进步. 原文链接:http://www.blogjava.net/supercrsky/articles/247449.html 针对JDK中的URLConnection连接Se ...

  2. Kotlin——最详细的抽象类(abstract)、内部类(嵌套类)详解

    如果您对Kotlin很有兴趣,或者很想学好这门语言,可以关注我的掘金,或者进入我的QQ群大家一起学习.进步. 欢迎各位大佬进群共同研究.探索QQ群号:497071402 进入正题 在前面几个章节中,详 ...

  3. Android中Application类的详解:

    Android中Application类的详解: 我们在平时的开发中,有时候可能会须要一些全局数据.来让应用中的全部Activity和View都能訪问到.大家在遇到这样的情况时,可能首先会想到自定义一 ...

  4. Delphi中的线程类 - TThread详解

    Delphi中的线程类 - TThread详解 2011年06月27日 星期一 20:28 Delphi中有一个线程类TThread是用来实现多线程编程的,这个绝大多数Delphi书藉都有说到,但基本 ...

  5. 【详细解析】MySQL索引详解( 索引概念、6大索引类型、key 和 index 的区别、其他索引方式)

    [详细解析]MySQL索引详解( 索引概念.6大索引类型.key 和 index 的区别.其他索引方式) MySQL索引的概念: 索引是一种特殊的文件(InnoDB数据表上的索引是表空间的一个组成部分 ...

  6. 在java poi导入Excel通用工具类示例详解

    转: 在java poi导入Excel通用工具类示例详解 更新时间:2017年09月10日 14:21:36   作者:daochuwenziyao   我要评论   这篇文章主要给大家介绍了关于在j ...

  7. UML类图详解_关联关系_多对一

    首先先来明确一个概念,即多重性.什么是多重性呢?多重性是指两个对象之间的链接数目,表示法是“下限...上限”,最小数据为零(0),最大数目为没有设限(*),如果仅标示一个数目级上下限相同. 实际在UM ...

  8. qml学习笔记(二):可视化元素基类Item详解(上半场anchors等等)

    原博主博客地址:http://blog.csdn.net/qq21497936本文章博客地址:http://blog.csdn.net/qq21497936/article/details/78516 ...

  9. 10.Spark Streaming源码分析:Receiver数据接收全过程详解

    原创文章,转载请注明:转载自 听风居士博客(http://www.cnblogs.com/zhouyf/)   在上一篇中介绍了Receiver的整体架构和设计原理,本篇内容主要介绍Receiver在 ...

随机推荐

  1. ASP.NET MVC5+EF6+EasyUI 后台管理系统(88)-Excel导入和导出-自定义表模导出

    前言 之前说了导入和导出,也提供了自定义的表模的导入,可见LinqToExcel可以做的事情不仅仅如此 这次我们来演示比较复杂的导出Excel,导出复杂的Excel与导入复杂的Excel原理基本是一样 ...

  2. java虚拟机概述

    java 虚拟机是什么?       java虚拟机是一个将字节码指令映射为对应物理操作系统指令的程序.       java程序的运行需要事先安装 jdk,而在jdk内部的jre中其核心就是 jvm ...

  3. Web攻防之暴力破解(何足道版)

    原创文章 原文首发我实验室公众号 猎户安全实验室 然后发在先知平台备份了一份 1 @序 攻防之初,大多为绕过既有逻辑和认证,以Getshell为节点,不管是SQL注入获得管理员数据还是XSS 获得后台 ...

  4. is 和==的区别

    Python中的对象包含三要素:id.type.value.其中id用来唯一标识一个对象,type标识对象的类型,value是对象的值.is判断的是a对象是否就是b对象,是通过id来判断的.==判断的 ...

  5. Win10 之最新最简单有效安装配置adb

    今天在新买的笔记本上安装配置adb,开始觉得挺简单的事,公司win7电脑上有现成的,但实际过程中--没想的那么简单了!好了,废话少说,直接正题. 研究了好一会下,总算搞定,总结如下: 1.下载Andr ...

  6. PHP 算法

    1.首先来画个菱形玩玩,很多人学C时在书上都画过,咱们用PHP画下,画了一半. 思路:多少行for一次,然后在里面空格和星号for一次. ? 1 2 3 4 5 6 <?php for($i=0 ...

  7. SQL Server学习之路(一):建立数据库、建立表

    0.目录 1.前言 2.建立数据库 2.1 通过SSMS建立数据库 2.2 通过SQL语句建立数据库 3.建立表 3.1 通过SSMS建立表 3.2 通过SQL语句建立表 1.前言 配置是win10+ ...

  8. PF_RING install in centos7

    很多centos7是最小化安装. 这样很多kernel就没有安装全,而且很多开发库也没有. 在安装PF_RING过程中,会缺少很多依赖. 首先安装依赖包: yum -y install numactl ...

  9. (四):C++分布式实时应用框架——状态中心模块

    C++分布式实时应用框架--状态中心模块 上篇:(三):C++分布式实时应用框架--系统管理模块 技术交流合作QQ群:436466587 欢迎讨论交流 版权声明:本文版权及所用技术归属smartguy ...

  10. navigator.userAgent浏览器检测(前端基础系列)

    对于前端来说,浏览器检测已经不陌生了,在做一些页面是,需要针对不同的浏览器进行处理不同的逻辑,最简单的就是区分pc和移动端的浏览器,或是android 和ios下的浏览器. 一.浏览器检测的由来?  ...