kotlin 基础概览

news2025/1/8 5:39:06

继承类/实现接口

继承类和实现接口都是用的 : ,如果类中没有构造器 ( constructor ),需要在父类类名后面加上 ()

class MainActivity : BaseActivity(), View.OnClickListener

空安全设计

Kotlin 中的类型分为「可空类型」和「不可空类型」:
  • 不可空类型  val editText : EditText
  • 可空类型  val editText : EditText?

lateinit 关键字

  • lateinit 只能修饰 var 可读可写变量 
  • lateinit  关键字声明的变量的类型必须是「不可空类型」
  • lateinit  声明的变量不能有「初始值」
  • lateinit  声明的变量不能是「基本数据类型」
  • 在构造器中初始化的属性不需要 lateinit 关键字

平台类型

在类型后面加上一个感叹号的类型是「平台类型」 Java 中可以通过注解减少这种平台类型的产生
  • @Nullable 表示可空类型
  • @NotNull @NonNull 表示不可空类型

类型判断

  • is 判断属于某类型
  • !is 判断不属于某类型
  • as 类型强转,失败时抛出类型强转失败异常
  • as? 类型强转,但失败时不会抛出异常而是返回 null

获取 Class 对象

  • 使用 类名 ::class 获取的是 Kotlin 的类型是 KClass
  • 使用 类名 ::class.java 获取的是 Java 的类型

setter/getter

  • Kotlin 声明属性的时候 ( 没有使用 private 修饰 ) ,会自动生成一个私有属性和 一对公开的 setter/getter 函数。
  • 在写 setter/getter 的时候使用 field 来代替内部的私有属性(防止递归栈溢 出)。

构造器

  • 使用 constructor 关键字声明构造器
class User {
    constructor()
}

如果我们在构造器主动调用了父类构造,那么在继承类的时候就不能在类的后面加上小括号

constructor(context: Context) : this(context, null)
// 主动调用了父类的构造器
constructor(context: Context, attr: AttributeSet?) : super(context, attr)
class CodeView : TextView {
    constructor(context: Context): super(context)
}

等价:

class CodeView(context: Context) : TextView(context) {
}

@JvmField 生成属性

  • 通过 @JvmField 注解可以让编译器只生成一个 public 的成员属性,不生成对 应的 setter/getter 函数

Any 和 Unit

  • Any : Kotlin 的顶层父类是 Any ,对应 Java 当中的 Object ,但是比 Object 少了 wait()/notify() 等函数
  • Unit : Kotlin 中的 Unit 对应 Java 中的 void

数组

  • 使用 arrayof() 来创建数组,基本数据类型使用对应的 intArrayOf()

静态函数和属性

  • 顶层函数
  • object
  • companion object

其中,「顶层函数」直接在文件中定义函数和属性,会直接生成静态的,在 Java 通过「文件名Kt」来 访问,同时可以通过 @file:JvmName 注解来修改这个「类名」。

需要注意,这种顶层函数不要声明在 module 内最顶层的包中,至少要在一个包中例如 com 。不然不能方便使用。

object companion object 都是生成单例对象,然后通过单例对象访问函数和属性的。

@JvmStatic

通过这个注解将 object companion object 的内部函数和属性,真正生成静态的。

单例模式/匿名内部类

  • 通过 object 关键字实现
// 单例
object Singleton {

}
// 匿名内部类
object : OnClickListener {

}

字符串模版

  • 通过 ${} 的形式来作为字符串模版
val number = 100
val text = "向你转账${number}元。"
// 如果只是单一的变量,可以省略掉 {}
val text2 = "向你转账$number元。"

多行字符串

val s = """
    我是第一行
    我是第⼆行
    我是第三行
    """.trimIndent()

区间

  • 200..299 表示 [200, 299]  的区间 ( 包括 299 )

when 关键字

  • Java 当中的 switch 的高级版,分支条件上可以支持表达式

声明接口/抽象类/枚举/注解

// 声明抽象类
abstract class
// 声明接口
interface
// 声明注解
annotation class
// 声明枚举
enum class

编译期常量

  • 在静态变量上加上 const 关键字变成编译期常量

标签

  • Java 中通过 「 类名 .this 例如 Outer.this 」 获取目标类引用
  • Kotlin 中通过「 this@ 类名 例如 this@Outer

遍历

记得让 IDE 来帮助生成 for 循环

for(item in items)

内部类

  • Kotlin 当中,内部类默认是静态内部类
  • 通过 inner 关键字声明为嵌套内部类

可见性修饰符

  • 默认的可见性修饰符是 public
  • 新增的可见性修饰符 internal 表示当前模块可见

注释

  • 注释中可以在任意地方使用 [] 来引用目标,代替 Java 中的 @param @link 等。

非空断言

  • 可空类型强制类型转换成不可空类型可以通过在变量后面加上 !! ,来达到类型转换。

open/final

  • Kotlin 中的类和函数,默认是被 final  修饰的 ( abstract override 例外 ) 除非加上  open  关键字才可以被子类覆写(默认方法都是 closed 关闭的,不能被重写的)

次级构造

class CodeView : TextView {
    constructor(context: Context): super(context)
}

主构造器

class CodeView constructor(context: Context) : TextView(context) 

// 如果没有被「可见性修饰符」「注解」标注,那么 `constructor` 可以省略
class CodeView(context: Context) : TextView(context)

成员变量初始化可以直接访问到主构造参数

class CodeView constructor(context: Context) : TextView(context) {
    val color = context.getColor(R.color.white)
}

init 代码块

主构造不能包含任何的代码,初始化代码可以放到 init 代码块中

class CodeView constructor(context: Context) : TextView(context) {
    init {
        //...
    }
}

在初始化的时候,初始化块会按照它们在「文件中出现的顺序」执行。

class CodeView constructor(context: Context) : TextView(context) {
    init {
        //...
    }
    val paint = Paint() // 会在 init{} 之后运行
}

构造属性

在主构造参数前面加上 var/val 使构造参数同时成为成员变量

class User constructor(var username: String?, var password: String?, var code: String?)

data class

数据类同时会生成
  • toString()
  • hashCode()
  • equals()
  • copy() (浅拷贝)
  • componentN() ...

相等比较

  • == 结构相等 ( 调用 equals() 比较 )
  • === 引用 相等 (比较 地址值 )

解构

可以把一个对象「解构」成很多变量

val (code, message, body) = response

对应的 Java 代码

val code = response.component1()
val message = response.component2()
val body = response.component3()

Elvis 操作符

  • 可以通过 ?: 的操作来简化 if null 的操作
// lesson.date 为空时使用默认值
val date = lesson.date ?:  "日期待定"

// lesson.state 为空时提前返回函数
val state = lesson.state ?:  return

// lesson.content 为空时抛出异常
val content = lesson.content ?: throw IllegalArgumentException("content expected")

when 操作符

  • when 表达式可以接受返回值,多个分支相同的处理方式可以放在一起,用逗号分
val colorRes = when (lesson.state) {
    Lesson.State.PLAYBACK, null -> R.color.playback
    Lesson.State.LIVE -> R.color.live
    Lesson.State.WAIT -> R.color.wait
}
  • when 表达式可以用来取代 if-else-if 链。如果不提供参数,所有的分支条件都是布尔表达式
val colorRes = when {
    (lesson.state == Lesson.State.PLAYBACK) -> R.color.playback
    (lesson.state == null) -> R.color.playback
    (lesson.state == Lesson.State.LIVE) -> R.color.live
    (lesson.state == Lesson.State.WAIT) -> R.color.wait
    else -> R.color.playback
}

operator

  • 通过 operator 修饰「特定函数名」的函数,例如 plus get 可以达到重载运算符的效果

lambda

如果函数的最后一个参数是 lambda ,那么 lambda 表达式可以放在圆括号之外:

lessons.forEach() { lesson : Lesson ->
    // ...
}

如果你的函数传入参数只有一个 lambda 的话,那么小括号也可以省略的:

lessons.forEach { lesson : Lesson ->
    // ...
}

如果 lambda 表达式只有一个参数,那么可以省略,通过隐式的 it 来访问:

lessons.forEach { // it
    // ...
}

循环

通过标准函数 repeat()

repeat(100) {
    //..
}

通过区间:

for (i in 0..99) {

}

// until 不包括右边界
for (i in 0 until 100) {

}

infix 函数

  • 必须是 成员函数 扩展函数
  • 必须 只能接受一个参数,并且不能有默认值
// until() 函数的源码
public infix fun Int.until(to: Int): IntRange {
    if (to <= Int.MIN_VALUE) return IntRange.EMPTY
    return this .. (to - 1).toInt()
}

潜逃函数

Kotlin 中可以在函数中继续声明函数

fun func(){
    fun innerFunc(){
    }
}
  • 内部函数可以访问外部函数的参数
  • 每次调用时,会产生一个函数对象

注解使用处目标

当某个元素可能会包含多种内容 ( 例如构造属性,成员属性 ) ,使用注解时可以通过 「注解使用处目标」,让注解对目标发生作用,例如 @file: @get: @set: 等。

函数简化

可以通过符号 = 简化原本直接 return 的函数

fun get(key :String) = SP.getString(key,null)

函数参数默认值

可以通过函数参数默认值来代替 Java 的函数重载

// 使用 @JvmOverloads 对 Java 暴露重载函数
@JvmOverloads
fun toast(text: CharSequence, duration: Int = Toast.LENGTH_SHORT) {
    Toast.makeText(this, text, duration).show()
}

扩展函数

  • 扩展函数可以为任何类添加上一个函数,从而代替工具类
  • 扩展函数和成员函数相同时,成员函数优先被调用
  • 扩展函数是静态解析的,在编译时就确定了调用函数 ( 没有多态 )

函数类型

  • 函数类型由「传入参数类型」和「返回值类型」组成,用「 -> 」连接,
  • 传入参数需要用「 () 」,如果返回值为 Unit 不能省略 
  • 函数类型实际是一个接口,如果需要将函数作为参数传递时可以通过 ::函数名 或者「匿名函数」或者使用 「 lambda

内联函数

  • 内联函数配合「函数类型」,可以减少「函数类型」生成的对象
  • 使用 inline 关键字声明的函数是「内联函数」,在「编译时」会将「内联函数」中的函数体直接插入到调用处。
  • 所以在写内联函数的时候需要注意,尽量将内联函数中的代码行数减少!

部分禁用内联

  • noinline 可以禁止部分参数参与内联编译
inline fun foo(inlined: () -> Unit, noinline notInlined:() -> Unit) {
    //......
}

具体化的类型参数

  • 因为内联函数的存在,我们可以通过配合 inline + reified 达到「真泛型」的效果
val RETROFIT = Retrofit.Builder()
    .baseUrl("https://api.github.com/")
    .build()

inline fun <reified T> create(): T {
    return RETROFIT.create(T::class.java)
}

val api = create<API>()

抽象属性

Kotlin 中,我们可以声明抽象属性,子类对抽象属性重写的时候需要重写对应的  setter/getter

属性委托

有些常见的属性操作,我们可以通过委托的方式,让它只实现一次,例如:
  • lazy 延迟属性:值只在第一次访问的时候计算
  • observable 可观察属性:属性发生改变时的通知
  • map 集合:将属性存在一个 map

对于一个只读属性( val 声明的),委托对象必须提供一个名为 getValue() 函数

对于一个可变属性( var 声明的),委托对象同时提供 setValue() getValue() 函数

类委托

可以通过类委托的模式来减少继承类委托的,编译器会优先使用自身重写的函数,而不是委托对象的函数
interface Base {
    fun print()
}
class BaseImpl(val x: Int) : Base {
    override fun print() {
        print(x)
    }
}

// Derived 的 print 实现会通过构造参数中的 b 对象来完成。
class Derived(b: Base) : Base by b

Kotlin 标准函数apply、also、run、let

使用时可以通过简单的规则作出一些判断:

需要返回自身 -> apply also 中选
  • 作用域中使用 this 作为参数 ----> 选择 apply
  • 作用域中使用 it 作为参数 ----> 选择 also
不需要返回自身 -> run let 中选择
  • 作用域中使用 this 作为参数 ----> 选择 run
  • 作用域中使用 it 作为参数 ----> 选择 let

apply 适合对一个对象做附加操作的时候

let 适合配合判空的时候 (最好是成员变量,而不是局部变量,局部变量更适合 if )

with 适合对同一个对象进行多次操作的时候

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

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

相关文章

嵌入式培训-数据结构-day23-线性表

线性表 线性表是包含若干数据元素的一个线性序列 记为&#xff1a; L(a0, ...... ai-1, ai, ai1 ...... an-1) L为表名&#xff0c;ai (0≤i≤n-1)为数据元素&#xff1b; n为表长,n>0 时&#xff0c;线性表L为非空表&#xff0c;否则为空表。 线性表L可用二元组形式描述…

冲压模具市场调研:2023年该行业发展现状及前景分析

汽车冲压件模具是汽车车身生产的重要工艺装备&#xff0c;是汽车换型的主要制约因素。汽车冲压件模具具有尺寸大、型面复杂、精度要求高等特点&#xff0c;属于技术密集型产品。 汽车冲压模具能快速精密地把材料直接加工成零件或半成品并通过焊接、铆接、拼装等工艺装配成零部件…

ChatGLM3-6B模型介绍及微调

文章目录 ChatGLM3-6B的强大特性更强大的基础模型更完整的功能支持更全面的开源序列 ChatGLM3-6B的部署basic版部署集成版部署 ChatGLM3-6B-base 微调ChatGLM3-6B-chat 微调多轮对话微调单轮对话微调 ChatGLM3-6B的强大特性 项目地址&#xff1a;https://github.com/THUDM/Cha…

基于node 安装express后端脚手架

1.首先创建文件件 2.在文件夹内打开终端 npm init 3.安装express: npm install -g express-generator注意的地方&#xff1a;这个时候安装特别慢,最后导致不成功 解决方法&#xff1a;npm config set registry http://registry.npm.taobao.org/ 4.依次执行 npm install -g ex…

shopify商城开发 引用谷歌字体库 fonts.google.com

引用谷歌字体库 https://fonts.google.com/ <link rel"preconnect" href"https://fonts.googleapis.com"> <link rel"preconnect" href"https://fonts.gstatic.com" crossorigin> <link href"https://fonts.goo…

详细了解云堡垒机的作用,提高企业数据信息安全

随着上云企业的不断增加&#xff0c;云上数据安全性成为企业面临的重要问题。为了保障企业的核心数据安全&#xff0c;越来越多的企业采购了云堡垒机来提升数据安全性。今天我们就来详细了解一下云堡垒机的作用&#xff0c;以及如何提高企业数据安全。 一、云堡垒机定义 云堡垒…

LabVIEW进行癌症预测模型研究

LabVIEW进行癌症预测模型研究 癌症是一种细胞异常增生的疾病。随着年龄的增长&#xff0c;细胞分裂速度放缓&#xff0c;但癌细胞会失去控制地不断分裂&#xff0c;形成可能良性或恶性的肿瘤。 2012年的国际癌症数据显示&#xff0c;新发癌症病例和癌症相关死亡人数有所增加。…

springboot使用EasyExcel导入数据

springboot使用EasyExcel导入数据 1. 引入依赖 <!-- Easy Excel --> <dependency><groupId>com.alibaba</groupId><artifactId>easyexcel</artifactId><version>3.2.1</version> </dependency>2. 建立对应实体类 假如…

多线程JUC 第2季 CAS的作用介绍

一 CAS作用介绍 1.1 CAS作用 CAS有3个操作数&#xff0c;位置内存值V&#xff0c;旧的预期值A&#xff0c;要修改的更新值B&#xff0c;如果内存值V和预期值相同则&#xff0c;内存值改为B&#xff0c;否则什么都不做。当它重来重试的这种行为称为-自旋。 CAS是一条cpu的原…

为什么越来越多的人从事软件测试行业?

1.市场需求增加&#xff1a;随着数字化转型和互联网的普及&#xff0c;各行各业都需要高质量、稳定可靠的软件来支持其业务运作。因此&#xff0c;对软件测试人员的需求也随之增加。同时&#xff0c;新兴技术的发展&#xff0c;如物联网、大数据、区块链、人工智能等&#xff0…

git学习笔记03(小滴课堂)

详解分支的基本操作 创建分支&#xff1a; 查看分支&#xff1a; 切换分支&#xff1a; git branch 中星号是当前分支。 idea中也更新了。 提交上去。 我们新建个分支&#xff1a; 我们新建分支是复制当前分支&#xff0c;而不是直接复制的主分支。 我们切换回主分支&#xf…

Bounding boxes augmentation for object detection

Different annotations formats Bounding boxes are rectangles that mark objects on an image. There are multiple formats of bounding boxes annotations. Each format uses its specific representation of bouning boxes coordinates 每种格式都使用其特定的边界框坐标…

51单片机控制1602LCD显示屏输出两行文字一

51单片机控制1602LCD显示屏输出两行文字一 1.概述 这篇文章介绍1602型号显示屏的基础知识&#xff0c;以及使用单片机控制它输出两行内容。 2.1602基础知识 1602 液晶显示模块是一种通用的工业液晶显示模块&#xff0c;专门用来显示字母、数字、符号等的点阵型液晶显示模块…

C++智能指针介绍

引言 为了充分利用RAII思想&#xff0c;C 11开始引入了智能指针&#xff0c;本文介绍RAII以及三种智能指针&#xff1a; std::unique_ptrstd::shared_ptrstd::weak_ptr 除此之外&#xff0c;本文还会介绍智能指针的常用创建方法&#xff1a; std::make_uniquestd::make_sha…

微信小程序:用map()将对象数组中的某一项组合成新数组

使用分析 使用map()方法来遍历 info 数组中的每个元素&#xff0c;并整合每一个对象中的某一项进行新数组的重组 效果展示 这里是查询对象数组中的全部name值 原始数据 提取出name的数组 核心代码 var infos items.map(item > item.name); 完整代码&#xff08;用微信小程…

iOS按钮控件UIButton使用

1.在故事板中添加按钮控件,步聚如下: 同时按钮Shift+Commad+L在出现在控件库中选择Button并拖入View Controller Scene中 将控件与变量btnSelect关联 关联后空心变实心 如何关联?直接到属性窗口拖按钮变量到控件上,出现一条线,然后松开,这样就关联成功了 关联成功后属性窗口…

Redis HyperLogLog 数据结构模型统计

HyperLogLog HyperLogLog 不是一种新的数据结构 &#xff0c; 本质上是字符串类型。 是一种基数算法。 通过 HyperLogLog 可以节省内存空间&#xff0c;并完成独立总数的统计。 HyperLogLog 数据结构可用于仅使用少量恒定内存来计算集合中的唯一元素&#xff0c;具体而言&…

Web开发:VS2022列表导出CSV中文乱码问题(已解决)

目录 一、问题重现 二、解决方案 1.新建一个EXCEL文档 2.点击数据-点击导入&#xff08;生成的文件&#xff09;-设置中文格式 一、问题重现 使用VS2022 DEBUG导出列表时&#xff0c;打开CSV文件发现中文乱码 二、解决方案 1.新建一个EXCEL文档 2.点击数据-点击导入&…

新手选电视盒子什么牌子好?内行分享最新电视盒子排名

新手们在面对众多品牌和机型时难免不知道如何挑选电视盒子&#xff0c;电视盒子的品质良莠不齐&#xff0c;究竟电视盒子什么牌子好&#xff1f;我身为从业人员&#xff0c;身边朋友在挑选电视盒子时都会咨询我的意见&#xff0c;我特意整理了业内最新发布的热门电视盒子排名TO…

选择销售技巧培训机构注意事项

选择销售技巧培训机构注意事项 随着市场竞争的日益激烈&#xff0c;销售技巧对于企业的成功至关重要。为了提升销售团队的技能&#xff0c;许多企业选择投资于销售技巧培训机构。然而&#xff0c;在选择培训机构时&#xff0c;有几个关键因素需要考虑。本文将介绍选择销售技巧…