Kotlin面向对象基础使用方法(继承、接口、Lambda、空指针检查机制等)

news2025/1/13 10:54:48

三、面向对象

1、继承

1.1 open改变类的继承属性

在kotlin设计时默认所有的非抽象类是无法被继承的,如果想要使得一个非抽象类可以被继承,我们需要使用open关键字。

open class Person {
    var name ="";
    var age = 0;

    fun eat() {
        println(name + " is eating. He is " + age +" years old.")
    }
}

同理如果类中的方法需要继承也需要使用open关键字

1.2 主构造函数与次构造函数

主构造函数:

  • 主构造函数的特点是没有函数体,直接定义在类名后面即可。
  • 主构造函数只能有一个,而且通常用于初始化类的属性,以及执行与类的整体初始化相关的操作。
  • 主构造函数的参数可以带有默认值,从而允许创建对象时省略某些参数。

我们使用init关键字编写主构造函数的逻辑

class Person(val name: String, val age: Int) {
    init {
        println("Person对象已创建,姓名:$name,年龄:$age")
    }
}

注意:

在主构造函数中声明val、var的字段会自动成为该类的字段,这样就会导致和父类中同名的字段冲突。如果不使用val、var关键字数据的作用域仅限定在主构造函数中。

image-20230907182600740

副构造函数:

  • 副构造函数是类中的额外构造函数,可以有多个,它们用constructor关键字声明。
  • 副构造函数通常用于提供不同的构造方式,允许使用不同的参数组合来创建对象。
class Person(val name: String, val age: Int) {
    // 主构造函数
    init {
        println("Person对象已创建,姓名:$name,年龄:$age")
    }

    // 副构造函数1,接受姓名参数
    constructor(name: String) : this(name, 0) {
        println("副构造函数1被调用")
    }

    // 副构造函数2,接受年龄参数
    constructor(age: Int) : this("未知姓名", age) {
        println("副构造函数2被调用")
    }
}

1.3 继承的格式

格式:

class 新的类名(无/参数) : 被继承类(无/参数){}

重写类的继承方法需要关键字override,并且被重写的方法需要有open关键字修饰。

open class Person(val name:String,val age:Int) {

    open fun eat() {
        println(name + " is eating. He is " + age +" years old.")
    }

}

class Student(val sno:String,val age1:Int) : Person(sno,age1){
    override fun eat(){
        println("$name Dont $age")
    }
}

fun main(){
    var i = Student("JAck",19)
    i.eat()
}

kotlin规定,当一个类既有主构造函数又有次构造函数,所有的次构造函数都必须调用主构造函数。

open class Person(val name:String,val age:Int) {

    fun eat() {
        println(name + " is eating. He is " + age +" years old.")
    }

}

class Student(val sno:String,val grade:Int,name: String,age: Int) : Person(name,age){
    //通过次构造函数调用主构造函数
    constructor(name: String,age: Int) : this("",0,name,age){

    }
    constructor() : this("",0)
}

fun main(){
    var i = Student("JAck",19)
    var p = Student("abs",1,"jaks",1)
    i.eat()
}

在这里this的用法是什么??

在这个类中,this关键字的使用是为了调用不同构造函数的目的。

具体来说,这个Student类具有以下构造函数:

  1. 主构造函数:Student类的主构造函数接受四个参数,分别是snogradenameage。这些参数用于初始化类的属性,其中snograde是继承自Person类的属性,而nameage是本类的属性。
  2. 第一个副构造函数:这个副构造函数接受两个参数,即nameage,它通过使用this关键字来**调用了主构造函数,**传递了空字符串 ""0 作为参数,这意味着它将调用主构造函数以初始化snograde属性,并且将传递的nameage参数用于初始化本类的属性。
  3. 第二个副构造函数:这个副构造函数不接受任何参数,它同样通过使用this关键字来**调用了第一个副构造函数,**传递了空字符串 "" 和零 0 作为参数,从而也会触发调用主构造函数,并且初始化本类的属性。

2、接口

接口的基本用法和java一致,但是在kotlin中允许代码对接口中的方法进行默认实现。

interface Study{
    fun readBooks()
    fun doHomework(){
        println("ANT")
    }
}

3、数据类——Java中toString、equals、hashCode的省略

在kotlin中我们不必写这些繁琐的方法,直接使用关键字data即可,代码如下:

//使用data修饰
data class Cellphone(val brand: String,val price: Double) {
}

fun main(args: Array<String>) {

    val cellphone1 = Cellphone("Samsung",1299.99)
    val cellphone2 = Cellphone("Samsung",1299.99)
    println(cellphone1)
    println( "cellphone1 == cellphone2 ??   "+(cellphone1 == cellphone2))
}

image-20230907204200998

Kotlin会根据主构造函数中的参数帮你将equals(),hash Code(),toString()等固定且无实际逻辑意义的方法自动生成。

4、单例类——object关键字

在kotlin中创建一个单例类的方式极其简单,只需要将class关键字改为object关键字即可。

object Singleton {
    fun singletonTest(){
        println("This is a OneInstance")
    }
}

此时Singleton就是一个单例类了,我们可以在这个类中直接编写需要的函数。

可以看出在Kotlin中我们不需要私有化构造函数。

5、Lambda编程

5.1 集合的创建和遍历

5.11 List集合

5.111、listOf

一般情况下我们需要使用这种方式初始化:

    val list = ArrayList<String>()
    list.add("Apple")
    list.add("Banana")
    list.add("Orange")

在kotlin中提供了一个内置的listOf函数来简化初始化集合的写法。

val list = listOf<String>("Apple","Banana","Orange")

当然我们也可以使用for-in遍历这个集合,并且Kotlin会自动补全fruit的类型。

    for(fruit in list){
        println(fruit)
    }

image-20230910204303363

不过需要注意的是使用listOf创建的集合是一个不可变集合,就是该集合只能用于读取操作。

为什么会不可修改呢?

image-20230910205556031

通过源码我们发现底层是数组的存储方式这样就能解释listOf`创建的集合只能用于读写操作了,同样他的查询效率也是比较高的。

5.112、mutableListOf

使用mutableListOf创建的集合是一个可变集合,可以对集合进行增删改查的操作。

image-20230910205431981

查看源码可以发现创建的是一个Arraylist类型的集合。使用链表的形式存储。

5.12 Set类型

Set集合的用法基本上和List集合的一样,只不过创建集合的方式变成了setOf()mutableSetof()

5.13 Map类型

Map类的的基本创建方法和Java中是一致的。

    val map = HashMap<String, Int>()
    map.put("Apple",1)
    map.put("Banana",2)
    map.put("Orange",3)

当然kotlint在Map中同样提供了mapOfmutableMapOf方法用于集合的创建。Kotlin 的 mapOfmutableMapOf 在底层都使用哈希表(Hash Table)作为存储结构,mapOf是数组存储,mutableMapOfLinkedHash

val map = mapOf("Apple" to 1,"Banana" to 2,"Orange" to 3)

for((fruit,number) in map){
    println(fruit + number)
}

需要注意的时这里的to不是关键字而是一个infix函数。

Kotlin 中,infix函数允许你以更自然的方式使用中缀表示法调用函数。

image-20230911124706544

5.2 集合函数式API

5.21 Lambda表达式的优化过程

{参数名:参数类型 -> 函数体}

val maxLengthFruit = list.maxBy({fruit:String -> fruit.length})
  • Kotlin规定当Lambda参数是函数的最后一个参数时,可以将Lambda表达式移到括号外面
val maxLengthFruit = list.maxBy(){fruit:String -> fruit.length}
  • 如果Lambda表达式是函数的唯一个参数,还可以将括号省略。而且kotlin具有出色的推到机制,我们还可以不声明参数类型。
val maxLengthFruit = list.maxBy{fruit -> fruit.length}
  • 并且当Lambda表达式参数列表只有一个参数时,也不必要声明参数名。可以使用it代替
val maxLengthFruit = list.maxBy{it.length}

5.22 常用API

  1. 计算最大长度:maxBy{it.length}

  2. 小写转大写:map { it.uppercase() }

  3. 过滤集合中的数据:filter函数

    val newList = list.filter { it.length > 5 }.map { it.uppercase() }
    
  4. 判断集合是否存在至少一个元素满足要求:any.{}

  5. 判断集合是否所有元素都满足要求:all.{}

    val anyResult = list.any{it.length >= 5}
    val allResult = list.all { it.length >= 10 }
    

    image-20230911173431696

6、空指针检查机制

Kotlin将空指针检查机制提到了编译时期,它默认所有的参数和变量都是非空的。

那么我们应该如何表示一个可空的类型呢?

6.1 ?

很简单在类名后加一个?代表可以是非空整形,例如:
I n t ? Int? Int
但是此时就可能出现潜在的空指针异常风险,那么我们应该如何解决呢?Kotlin提供了一个辅助判空工具——?.。

6.2 辅助判空工具

6.2.1 ?.

例如这段代码:

if(A != null){
	A.do()
}

可以简化为这样的形式:

A?.do()

6.2.2 ?:

例如这段代码:

if(A != null){
	A
}else{
	B
}

可以简化为以下形式:

Val c = A ?: C

左边表达式不为空就返回左边的值,否则返回右边的值。

6.2.3 非空断言工具——!!

Kotlin的这种检查机制在某些情况下也会出现问题,例如:

image-20230911194503533

此时是无法通过编译的,因为uppercase无法知道content不是空值。

此时就需要一种断言工具——!!

val upString = content!!.uppercase()

这是一种有风险的写法,意在告诉Kotlin,这里对象肯定不是空你不需要帮我检查空指针。

6.2.4 let辅助工具

这个函数提供了函数式API的编程接口,并将原始对象作为参数传递到Lambda表达式中。

obj.let{ obj1 ->
        //业务逻辑
       }

在这里,obj和obj1实际上是同一个对象。

如使用let修改这一段代码:

fun doStuday(study: Study?) {
    study?.doHomework()
    study?.readBooks()
}

修改结果如下:

fun doStuday(study: Study?) {
    study?.let {
        it.readBooks()
        it.doHomework()
    }
}

let是可以处理全局变量的判空问题但是if不行,因为if可以导致别的线程修改了这个全局变量。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1000837.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

入门人工智能 ——自然语言处理介绍,并使用 Python 进行文本情感分析(5)

入门人工智能 ——自然语言处理介绍&#xff0c;并使用 Python 进行文本情感分析&#xff08;5&#xff09;&#xff09; 入门人工智能 ——自然语言处理介绍&#xff0c;并使用 Python 进行文本情感分析介绍自然语言处理的挑战NLP的基本任务NLP的基本技术NLP的应用领域 使用 P…

RHCSA Linux环境搭建

目录 一、安装Linux操作系统 二、创建虚拟机 1、成功激活后&#xff0c;开始“创建新的虚拟机” 新建虚拟机 2、自定义--根据我们的需求来创建 3、默认即可 4、选择稍后安装操作系统&#xff08;可自定义设置某些选项&#xff09; 5、选择Linux操作系统&#xff0c;版本…

Linux dup dup2函数

/*#include <unistd.h>int dup2(int oldfd, int newfd);作用&#xff1a;重定向文件描述符oldfd 指向 a.txt, newfd 指向b.txt,调用函数之后&#xff0c;newfd和b.txt close&#xff0c;newfd指向a.txtoldfd必须是一个有效的文件描述符 */ #include <unistd.h> #i…

Fourier傅里叶变换的线性性质和位移性质

Fourier傅里叶变换的线性性质和位移性质 为了阐述方便, 假定在这些性质中, 凡是需要求Fourier变换的函数都满足Fourier积分定理中的条件。在证明这些性质时, 不再重述这些条件。 一、线性性质 设 F 1 ( ω ) F [ f 1 ( t ) ] {F_1}(\omega ) {\mathscr F}[{f_1}(t)] F1​(…

2023/9/11 -- C++/QT

作业 仿照string类&#xff0c;完成myString 类 02mystring.h: #ifndef __02MYSTRING_H__ #define __02MYSTRING_H__#include <iostream> #include <cstring>using namespace std;class myString{ private:char *str;int size; public://无参构造myString();//有…

C++算法 —— 动态规划(5) 子序列

文章目录 1、动规思路简介2、最长递增子序列3、摆动序列4、最长递增子序列的个数5、最长数对链6、最长定差子序列7、最长斐波那契子序列的长度8、最长等差数列9、等差数列划分 II 每一种算法都最好看完第一篇再去找要看的博客&#xff0c;因为这样会帮你梳理好思路&#xff0c;…

Python 图形化界面基础篇:添加标签( Label )到 Tkinter 窗口

Python 图形化界面基础篇&#xff1a;添加标签&#xff08; Label &#xff09;到 Tkinter 窗口 引言什么是 Tkinter 标签&#xff08; Label &#xff09;&#xff1f;步骤1&#xff1a;导入 Tkinter 模块步骤2&#xff1a;创建 Tkinter 窗口步骤3&#xff1a;创建标签&#x…

kafka管理工具之kafka-ui的环境搭建笔记

由于项目需要kafka支持认证功能&#xff0c;就把kafka升级到3.2.0了。之前一直使用的kafka tools(现在叫Offset Explorer&#xff0c;个人使用免费&#xff0c;商用付费)&#xff0c;开了认证之后就不好用了&#xff0c;卡的很&#xff0c;一点也不丝滑了&#xff0c;于是只好重…

pytorch代码实现之Partial Convolution (PConv卷积)

Partial Convolution (PConv卷积) Partial Convolution (PConv卷积)&#xff0c;有助于提升模型对小目标检测的性能。目前许多研究都集中在减少浮点运算&#xff08;FLOPs&#xff09;的数量上。然而FLOPs的这种减少不一定会带来延迟的类似程度的减少。这主要源于每秒低浮点运…

浅析建筑电气火灾问题和预防方案

安科瑞 华楠 摘要&#xff1a;近几年来随着技术化和信息化的不断发展&#xff0c;电器在建筑中的应用也是越来越广泛&#xff0c;电气也成为人们生活当中的一部分。现如今建筑物设计中都要增加电气线路的设计&#xff0c;几年电气引起的火灾也不在少数。建筑电气在运行的过程中…

OpenCV(三十七):拟合直线、三角形和圆形

1.点集拟合的含义 点集拟合是一种通过拟合函数或曲线来近似描述给定离散数据点的技术,在点集拟合中&#xff0c;可以使用不同的函数或曲线拟合方法来拟合直线、三角形和圆形。 直线拟合&#xff1a;对于给定的二维数据点集合&#xff0c;可以使用最小二乘法来拟合一条直线。 …

springboot之三:原理分析之自动配置condition

导入&#xff1a; SpringBoot是如何知道要创建哪个Bean的&#xff1f;比如Spring Boot是如何知道要创建RedisTemplate的&#xff1f; Condition&#xff1a; Condition是在Spring4.0增加的条件判断功能&#xff0c;通过这个可以实现选择性的创建Bean操作。 自定义条件&…

C++之string::npos应用实例(一百九十三)

简介&#xff1a; CSDN博客专家&#xff0c;专注Android/Linux系统&#xff0c;分享多mic语音方案、音视频、编解码等技术&#xff0c;与大家一起成长&#xff01; 优质专栏&#xff1a;Audio工程师进阶系列【原创干货持续更新中……】&#x1f680; 人生格言&#xff1a; 人生…

使用IDEA创建Vue3通过Vite实现工程化

1、创建Vite项目的分步说明 IntelliJ IDEA与Vite构建工具集成&#xff0c;改善了前端开发体验。Vite 由一个开发服务器和一个构建命令组成。构建服务器通过本机 ES 模块提供源文件。生成命令将代码与汇总捆绑在一起&#xff0c;汇总预配置为输出高度优化的静态资产以供生产。In…

AI绘画:Midjourney超详细教程Al表情包超简单制作,内附关键词和变现方式

大家好&#xff0c;本篇文章主要介绍AI绘画完成表情包的制作和变现方式分享。 你还不会AI表情包制作吗&#xff1f;下面我们详细的拆解制作过程。跟着这个教程做出一套属于自己的表情包。 核心工具Midjourney PS&#xff0c;你就可以得到一套自己的专属表情包啦~ 整体制作…

Go 基础15-在init()函数中检查包级变量的初始状态

从程序逻辑结构角度来看&#xff0c;包&#xff08;package&#xff09;是Go程序逻辑封装的基本单元&#xff0c;每个包都可以理解为一个“自治”的、封装良好的、对外部暴露有限接口的基本单元。一个Go程序就是由一组包组成的。 在Go包这一基本单元中分布着常量、包级变量、函…

胡焕庸线,我国东西地级市分布密度分界线

背景 黑河—腾冲线&#xff0c;又名胡焕庸线&#xff0c;是我国人口密度分布的东西近似分界线。今天把地级市坐标分布密度做成热力图&#xff0c;并与胡焕庸线一起展示时&#xff0c;惊奇的发现&#xff0c;胡焕庸线貌似也是我车东西地级市分布密度的分界线。 生成folium地图…

【17 > 分布式接口幂等性】2. Update的幂等性原理解析

一、 根据 唯一业务号去更新 数据的情况 1.1 原理 1.2 操作 1.3 实战 Stage 1&#xff1a;表添加 version 字段 Stage 2&#xff1a;前端 > 版本号放入隐藏域 Stage 3&#xff1a;后台 > 使用版本号作为更新条件 二、更新操作没有唯一业务号&#xff0c;可使用Tok…

FL Studio Producer Edition 21.0.3.3713中文完整破解版功能特点及安装激活教程

FL Studio Producer Edition 21.0.3.3713中文完整破解版是一款由Image Line公司研发几近完美的虚拟音乐工作站,同时也是知名的音乐制作软件。它让你的计算机就像是全功能的录音室&#xff0c;漂亮的大混音盘&#xff0c;先进的创作工具&#xff0c;让你的音乐突破想象力的限制。…

4.2 Ioc容器加载过程-Bean的生命周期深度剖析

Bean生命周期详解 第一步拿到父类BeanFactory子类 第二步&#xff0c;读取配置类 AnnotatedBeanDefinitionReader 用来读取配置类之外和还做了 第一个是解析类的处理器&#xff0c;没有的话我们的配置类就无法解析 总结this()无参构造函数里面实现了【三大步】 实例…