Kotlin重点理解安全性

目录

  • 一 Kotlin安全性
    • 1.1 可空类型
    • 1.2 安全调用运算符
    • 1.3 Elvis 运算符
    • 1.4 非空断言运算符
    • 1.5 安全类型转换
    • 1.6 延迟初始化

一 Kotlin安全性

Kotlin 在设计时采用了一系列策略,旨在尽可能地减少空指针异常(NullPointerException)的出现。空指针异常是许多编程语言中常见的错误之一,Kotlin 通过以下几种方式来避免它:

  1. 可空类型(Nullable Types):Kotlin引入了可空类型的概念,允许变量具有可以存储空值的能力。在 Kotlin 中,如果一个变量可能为 null,必须显式地声明为可空类型。例如,使用 String? 表示一个可能为null的字符串类型。这样做可以在编译时就捕捉到潜在的空指针异常,因为 Kotlin 不允许直接对可空类型进行操作,除非先进行非空判断或者安全调用。
  2. 安全调用运算符(Safe Call Operator):在 Kotlin 中,可以使用 ?. 运算符来调用可空对象的方法或访问其属性。如果对象为 null,则整个表达式将返回 null 而不会导致空指针异常。例如,obj?.someMethod() 将返回 null,如果 obj 是 null,否则调用 someMethod() 方法。
  3. Elvis 运算符(Elvis Operator):Elvis 运算符 ?: 可以用于提供默认值,如果表达式的值为 null,则可以返回提供的默认值。例如,val length = str?.length ?: 0 表示如果 str 是 null,length 将赋值为 0。
  4. 非空断言运算符(Non-null Assertion Operator):有时候我们可以确定某个可空类型的变量在某个地方不会为 null,这时可以使用非空断言运算符 !! 来显式地告诉编译器该变量不会为 null。但是需要小心使用它,如果变量为 null,将会触发空指针异常。
  5. 安全类型转换(Safe Cast):使用 as? 运算符进行安全类型转换,如果转换失败,则返回 null。这样可以避免在类型转换时出现空指针异常。
  6. 延迟初始化:在 Kotlin 中,类的属性可以使用 lateinit 延迟初始化,这使得在声明属性时不需要立即初始化,而可以在稍后进行初始化。但是要小心使用延迟初始化,必须确保在使用该属性之前已经初始化,否则会抛出异常。

1.1 可空类型

在 Kotlin 中,可空类型(Nullable Types)允许变量具有可以存储 null 值的能力。通常,变量的类型不能为 null,如果试图将 null 赋值给非空类型的变量,编译器会报错。但是,有时候我们需要表示一个变量可能为 null 的情况,这就是可空类型的用途。
在 Kotlin 中,声明可空类型的变量时,在类型名称后面添加一个问号 ? 来表示该变量可以为 null。例如,String? 表示一个可能为 null 的字符串类型,Int? 表示一个可能为 null 的整数类型。
以下是一些可空类型的示例:

fun main() {var name: String? = "John" // 可以为 null 的字符串var age: Int? = null // 可以为 null 的整数val nullableList: List<Int>? = listOf(1, 2, 3) // 可以为 null 的整数列表// 当然,我们也可以使用非空类型的列表val nonNullableList: List<Int> = listOf(1, 2, 3)
}

在使用可空类型时,需要特别注意处理可能为 null 的情况,以避免空指针异常。Kotlin 提供了多种方式来处理可空类型的变量:

  1. 安全调用操作符 ?.:允许在不确定变量是否为 null 的情况下,调用其方法或访问其属性。如果变量为 null,则整个表达式会返回 null。
val length = name?.length // 如果 name 不为 null,则返回 name 的长度,否则返回 null
  1. Elvis 操作符 ?::用于在表达式为 null 时提供默认值。
val nonNullName = name ?: "Unknown" // 如果 name 不为 null,则使用 name 的值,否则使用 "Unknown"
  1. 安全类型转换 as?:用于安全地进行类型转换,如果转换失败,则返回 null。
val intValue: Int? = stringValue as? Int // 如果 stringValue 可以转换为 Int,则返回 Int 值,否则返回 null

通过使用可空类型和相应的操作符,我们可以在编译时处理潜在的 null 值,避免空指针异常,并使得代码更加健壮和安全。

val str: String? = null
val length = str.length // 编译错误:str 可能为 null

let 函数可以用于在非空的情况下执行某些操作,例如:

val str: String? = null
str?.let {// str 不为 null 时执行println(it.length)
}

optional 函数可以用于在可空类型的变量为 null 时执行某些操作,例如:

val str: String? = null
str.optional {// str 为 null 时执行println("str is null")
}

1.2 安全调用运算符

在 Kotlin 中,安全调用运算符(Safe Call Operator)是一种用于处理可空类型的特殊运算符。它允许你在不确定一个可空类型变量是否为 null 的情况下,安全地调用它的方法或访问它的属性。如果变量为 null,那么整个表达式会返回 null,而不会导致空指针异常。
安全调用运算符的语法是 ?.,它用于在可空类型变量后面调用其方法或访问其属性。以下是使用安全调用运算符的示例:

fun printLength(name: String?) {val length = name?.lengthprintln("Length of name: $length")
}fun main() {printLength("John") // 输出: Length of name: 4printLength(null) // 输出: Length of name: null
}

在上面的示例中,我们定义了一个名为 printLength 的函数,它接受一个可空类型的字符串 name 作为参数。在函数中,我们使用安全调用运算符 name?.length 来获取 name 字符串的长度,如果 name 是 null,则整个表达式会返回 null,否则返回字符串的长度。这样,在处理可空类型时,就不需要手动进行 null 检查,可以避免空指针异常。
安全调用运算符在处理链式调用时尤为有用。例如,如果我们有一个可空类型的对象 person,它有一个可空类型的属性 address,并且 address 有一个可空类型的属性 city,我们可以通过链式使用安全调用运算符来避免多层嵌套的 null 检查:

val city = person?.address?.city

上面的代码会在 person、address 或 city 中任何一个为 null 时返回 null,而不会抛出空指针异常。
通过使用安全调用运算符,我们可以更加简洁和安全地处理可空类型的变量,减少了大量的 null 检查代码,提高了代码的可读性和健壮性。

1.3 Elvis 运算符

在 Kotlin 中,Elvis 运算符 ?: 是一种用于处理可空类型的特殊运算符。它提供了一种简洁的方式来为可能为 null 的表达式提供一个默认值。如果表达式的值为 null,那么 Elvis 运算符会返回指定的默认值,否则返回表达式的值。
Elvis 运算符的语法是 expression ?: defaultValue,其中 expression 是一个可能为 null 的表达式,defaultValue 是一个默认值,用于在 expression 为 null 时返回。
以下是使用 Elvis 运算符的示例:

fun printLength(name: String?) {val length = name?.length ?: -1println("Length of name: $length")
}fun main() {printLength("John") // 输出: Length of name: 4printLength(null) // 输出: Length of name: -1
}

在上面的示例中,我们定义了一个名为 printLength 的函数,它接受一个可空类型的字符串 name 作为参数。在函数中,我们使用 Elvis 运算符 name?.length ?: -1 来获取 name 字符串的长度,如果 name 是 null,则整个表达式会返回 -1,否则返回字符串的长度。这样,我们可以指定一个默认值 -1,以避免 name 为 null 时导致的空指针异常。
Elvis 运算符还可以用于链式调用中的多个可空类型变量:

val city = person?.address?.city ?: "Unknown"

在上面的代码中,如果 person、address 或 city 中任何一个为 null,都会返回 “Unknown” 作为默认值。
通过使用 Elvis 运算符,我们可以更加简洁地处理可空类型的变量,并在需要时为其提供默认值,使得代码更加健壮和安全。

1.4 非空断言运算符

在 Kotlin 中,非空断言运算符 !! 是一种用于处理可空类型的特殊运算符。它用于显式地告诉编译器一个可空类型的变量不会为 null,从而允许在变量为 null 的情况下进行非空操作。
非空断言运算符的语法是 expression!!,其中 expression 是一个可空类型的变量。如果 expression 不为 null,那么 expression!! 会返回它的值;但如果 expression 为 null,那么在运行时会抛出 NullPointerException。
需要注意的是,非空断言运算符 !! 是一种危险的操作。如果使用不当,会导致空指针异常,因此应该谨慎使用。通常情况下,最好避免使用 !!,而是使用安全调用运算符 ?. 或 Elvis 运算符 ?: 来处理可空类型的变量。
以下是使用非空断言运算符的示例:

fun printLength(name: String?) {val length = name!!.lengthprintln("Length of name: $length")
}fun main() {printLength("John") // 输出: Length of name: 4printLength(null) // 抛出 NullPointerException
}

在上面的示例中,我们定义了一个名为 printLength 的函数,它接受一个可空类型的字符串 name 作为参数。在函数中,我们使用非空断言运算符 name!!.length 来获取 name 字符串的长度。如果 name 是 null,则在运行时会抛出 NullPointerException。
尽管非空断言运算符 !! 可以在某些情况下方便地处理可空类型的变量,但它应该谨慎使用,并且只在你能确保变量不会为 null 的情况下使用。通常情况下,推荐使用更安全的操作符来处理可空类型,以避免潜在的空指针异常。

1.5 安全类型转换

在 Kotlin 中,安全类型转换(Safe Cast)是一种用于处理类型转换的特殊操作符。它允许你安全地将一个对象转换为指定类型,如果转换失败,则返回 null,而不会抛出 ClassCastException 异常。
安全类型转换的语法是 as?,它用于将一个对象转换为指定类型,并返回转换后的对象。如果转换失败,则整个表达式会返回 null。
以下是使用安全类型转换的示例:

fun printLength(name: Any) {val length = (name as? String)?.lengthprintln("Length of name: $length")
}fun main() {printLength("John") // 输出: Length of name: 4printLength(42) // 输出: Length of name: null
}

在上面的示例中,我们定义了一个名为 printLength 的函数,它接受一个 Any 类型的参数 name。在函数中,我们首先使用 as? 运算符将 name 对象安全地转换为 String 类型。如果 name 是 String 类型,则转换成功,我们可以获取它的长度并打印;如果 name 不是 String 类型,则转换失败,整个表达式会返回 null。
在 main 函数中,我们分别调用了 printLength(“John”) 和 printLength(42)。第一个调用中,name 是一个字符串,转换成功,打印字符串的长度。第二个调用中,name 是一个整数,转换失败,返回 null。
安全类型转换可以帮助我们处理不确定对象类型的情况,而不会出现 ClassCastException 异常。它是一种安全而方便的方式来进行类型转换,特别在处理多态类型的对象时很有用。

1.6 延迟初始化

在 Kotlin 中,延迟初始化(Late Initialization)是一种允许在声明属性时不立即初始化的特性。在某些情况下,我们希望在稍后的时间点才对属性进行初始化,例如在构造函数执行之后或在对象创建之后。
为了实现延迟初始化,我们需要在属性声明中使用 lateinit 关键字,并将属性的类型设置为非空类型(不可为 null)。这样,编译器会知道该属性会在稍后被初始化,而不会在初始化阶段要求立即赋值。
以下是一个简单的示例,演示了如何在 Kotlin 中使用延迟初始化:

class Person {lateinit var name: Stringlateinit var age: Intfun init(name: String, age: Int) {this.name = namethis.age = age}
}fun main() {val person = Person()person.init("Alice", 30)println("Name: ${person.name}")println("Age: ${person.age}")
}

在上面的示例中,我们定义了一个 Person 类,其中的 name 和 age 属性使用了延迟初始化。我们在类的 init 方法中进行属性的初始化。因为属性使用了 lateinit 关键字,所以在创建 Person 对象时,并不需要立即为 name 和 age 赋值。相反,我们可以在稍后的时间点通过调用 init 方法来为它们赋值。
需要注意的是,使用 lateinit 的属性必须是非空类型,并且必须在对象的生命周期内进行初始化,否则会抛出 UninitializedPropertyAccessException 异常。因此,使用延迟初始化时要确保在使用属性之前已经进行了初始化。
延迟初始化非常有用,可以在某些情况下优化对象的创建和初始化过程,避免不必要的资源消耗,并提高代码的性能。但是要小心使用它,确保在适当的时间点对属性进行初始化。

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

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

相关文章

通用视觉大模型

ViT: Google Swin-Transformer: Microsoft V-MOE: Google SAM&#xff1a; Meta 盘古CV&#xff1a; 华为 文心UFO&#xff1a; 百度 … 预训练大模型 # 导入必要的库和模块 import argparse import json import pathlib# 定义 Stanford Alpaca 使用的 prompt 格式 PROMPT_D…

纯函数 和 函数柯里化 05 (待补充)

加油&#xff0c;今天周二啦&#xff01;&#x1f60d; 文章目录 前言一、js 的纯函数二、副作用三、纯函数的优势四、JavaScript 柯里化五、柯里化作用 - 让函数的职责单一六、柯里化作用 - 逻辑的复用七、自动柯里化函数&#xff08;我没理解&#xff0c;等我理解了再更&…

如何在洛谷自己出的题中出数据

首先&#xff0c;假如你要加1个数据&#xff1a; 打开Dev-c&#xff08;其他也行&#xff09; 填入输入数据&#xff1a; &#xff08;这个数据只是我编的&#xff09; 将这个东东保存为in文件&#xff08;第一个数据就名为001&#xff0c;第二个002&#xff09;&#xff1a…

2022 robocom 世界机器人开发者大赛-本科组(国赛)

RC-u1 智能红绿灯 题目描述&#xff1a; RC-u1 智能红绿灯 为了最大化通行效率同时照顾老年人穿行马路&#xff0c;在某养老社区前&#xff0c;某科技公司设置了一个智能红绿灯。 这个红绿灯是这样设计的&#xff1a; 路的两旁设置了一个按钮&#xff0c;老年人希望通行马路时会…

没有进度管理的项目,都是在做无用功

在项目管理过程中&#xff0c;最大的挑战之一是确保项目实施与计划保持一致。 项目实施过程是一个相对漫长的过程&#xff0c;其中受到许多因素的影响。如果项目实施没有按照原始项目计划进行&#xff0c;很容易导致项目偏离计划&#xff0c;最终可能导致项目停滞或失败。 当…

Maven出现报错 ; Unable to import maven project: See logs for details错误的多种解决方法

问题现象; IDEA版本&#xff1a; Maven 版本 &#xff1a; 3.3.9 0.检查 maven 的设置 &#xff1a;F:\softeware\maven\apache-maven-3.9.3\conf 检查setting.xml 配置 本地仓库<localRepository>F:\softeware\maven\local\repository</localRepository>镜像…

使用 Python 和 Flask 构建简单的 Restful API 第 1 部分

一、说明 我将把这个系列分成 3 或 4 篇文章。在本系列的最后&#xff0c;您将了解使用flask构建 restful API 是多么容易。在本文中&#xff0c;我们将设置环境并创建将显示“Hello World”的终结点。 我假设你的电脑上安装了python 2.7和pip。我已经在python 2.7上测试了本文…

Netty的只读ByteBuf

说明 通过io.netty.buffer.ByteBuf的asReadOnly()函数&#xff0c;可以返回一个只读的ByteBuf&#xff1b;通过isReadOnly()函数&#xff0c;可以判断该ByteBuf是否是只读的。 示例 package com.thb;import io.netty.buffer.ByteBuf; import io.netty.buffer.Unpooled;publi…

基于java+Apriori算法的网络书城设计与实现

摘要 网络技术的发展市场的销售模式发生了巨大的改变&#xff0c;现在销售网站引用电子商务网站网站设计的基本流程&#xff0c;引用网站网站的开发生命周期法和结构化方法&#xff0c;使用JSP语言来设计并实现了书籍网上销售网站。本书籍销售网站分为前台、后台两个模块。前台…

汽车基础软件新「战争」:群雄混战,谁在抢跑?

中国车用基础软件迎来了发展的黄金窗口期。 当前&#xff0c;汽车电子电气架构正在加速向“跨域融合”的时代演进&#xff0c;汽车软件架构也在加速向SOA架构升级&#xff0c;软硬件分层解耦成为了实现“软件定义汽车”新开发模式的前提。 过去&#xff0c;汽车的所有功能开发…

Activiti7

文章目录 概述入门案例1.创建springboot项目pom.xml 2.获取ProcessEngine2.1 默认的方式2.2 编程方式获取2.3 表结构介绍 3.在线流程设计器 概述 官网地址&#xff1a;https://www.activiti.org/ Activiti由Alfresco软件开发&#xff0c;目前最高版本Activiti 7。是BPMN的一个…

Flutter iOS 集成使用 flutter boost

在 Flutter项目中集成完 flutter boost&#xff0c;并且已经使用了 flutter boost进行了路由管理&#xff0c;这时如果需要和iOS混合开发&#xff0c;这时就要到 原生端进行集成。 注意&#xff1a;之前建的项目必须是 Flutter module项目&#xff0c;并且原生项目和flutter m…

【Spring】Spring之事务底层源码解析

目的 能使用spring事务解决开发需求了解spring事务是如何被spring管理的了解spring事务底层原理实现&#xff0c;比如代理、事务传播机制等 Spring事务简单使用 配置数据源及事务管理器&#xff1a; Component ComponentScan("com.firechou.aop") EnableTransact…

bigemap如何添加高清在线地图?

说明&#xff1a;批量添加可以同时添加多个在线地图&#xff0c;一次性添加完成&#xff08;批量添加无法验证地址是否可以访问&#xff09; 添加后如下图&#xff1a; 第一步 &#xff1a; 制作地图配置文件&#xff1a;选择添加在线地图&#xff08;查看帮助&#xff09;。 …

Chatgpt AI newbing作画,文字生成图 BingImageCreator 二次开发,对接wxbot

开源项目 https://github.com/acheong08/BingImageCreator 获取cookie信息 cookieStore.get("_U").then(result > console.log(result.value)) pip3 install --upgrade BingImageCreator import os import BingImageCreatoros.environ["http_proxy"]…

基于 yolov8 的人体姿态评估

写在前面 工作中遇到&#xff0c;简单整理博文内容为使用预训练模型的一个预测 Demo测试图片来源与网络,如有侵权请告知理解不足小伙伴帮忙指正 对每个人而言&#xff0c;真正的职责只有一个&#xff1a;找到自我。然后在心中坚守其一生&#xff0c;全心全意&#xff0c;永不停…

华为认证 | 云计算HCIE3.0改版后有什么变化?

随着技术的不断进步和行业的发展&#xff0c;云计算的HCIE作为华为公司的顶级认证&#xff0c;也进行了版本的更新。 那改版后有哪些变化呢&#xff0c;今天给大家讲讲。 01 HCIE认证简介 HCIE认证是华为公司旗下的顶级专业认证&#xff0c;面向IT领域的高级专业人士。 它涵…

83 | Python可视化篇 —— Bokeh数据可视化

Bokeh 是一种交互式数据可视化库,它可以在 Python 中使用。它的设计目标是提供一个简单、灵活和强大的方式来创建现代数据可视化,同时保持良好的性能。Bokeh 支持多种图表类型,包括线图、散点图、柱状图、饼图、区域图、热力图等。此外,它还支持将这些图表组合在一起以创建…

收集到大量的名片怎么转为excel?

来百度APP畅享高清图片 参加完展会或集体会议&#xff0c;是不是收了一大堆名片&#xff0c;保管起来超级麻烦&#xff0c;还容易丢三落四&#xff1f;别急&#xff0c;我们有办法&#xff01;把名片转成电子版保存到电脑上就完美啦&#xff01;但要是名片数量有点多&#xff0…

ChatGPT应用在AIGC内容生产【赠书活动|第一期《硅基物语》】

文章目录 爆火的AI工具ChatGPT走入大众视野的AIGCAIGC领域的发展AIGC价值引领『赠书活动 &#xff5c; 第一期』 爆火的AI工具ChatGPT 2023年伊始&#xff0c;ChatGPT就火遍全网&#xff0c;成为了全球最快拥有1亿月活用户的产品。在地铁上、电梯中、咖啡厅到处都充满着讨论AI…