Kotlin中 !!. 跟 ?. 的区别跟用法

本文深入解析了Kotlin中?.和!!操作符的使用方法,对比了与Java中空值处理的区别,通过实例展示了如何优雅地处理可能为null的对象。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

在Kotlin中!!跟?都是用于判断空参数异常的

?.意思是这个参数可以为空,并且程序继续运行下去

!!.的意思是这个参数如果为空,就抛出异常

下面给大家举个简单的例子,在JAVA中判断一个参数非空是这样写的:

NullClass nullClass = null;
        
if (nullClass!=null) {
    ullClass.nullFun();
 }

将这段代码转换成Kotlin之后呢

val nullClass: NullClass? = null

nullClass?.nullFun()

短短两行就表述完了

在一开始的时候我们声明了一个类,并且在类名后面加了一个? 意思就是这个类可以为空,然后在下面用到这个类里面的一个方法时又加了一个问号,意思就是,当程序运行到这一行时,如果这个参数为空,就跳过这一行,程序继续执行下去

所以?.的用法就是相当于Java里的if()判断null

if (nullClass!=null) {//如果判断为空
//跳过这一行,程序继续执行
    nullClass.nullFun();
}

一般?.的用法是:

在新建一个参数的类名后面加一个? 表示这个参数可以为空.

还有就是在用到这个参数的时候后面加? 表示空参数就跳过并且程序继续执行

而!!只用于用到这个参数的时候在后面加!!,表示空参数就抛出异常

还是相同的例子:

val nullClass: NullClass?=null

nullClass!!.nullFun()

换成Java代码就是

NullClass nullClass = null;
        
if (nullClass!=null) {
    nullClass.nullFun();
}else {
    throw new NullPointerException();
}

这里大家应该都能看明白了,在第二行参数后面加个!!,意思就是当程序执行到这行,判断这个参数如果是空参数,就抛出异常

所以!!.的用法就是相当于Java里的if()else()判断null

if (nullClass!=null) {//如果判断为空
    nullClass.nullFun();
}else {//抛出空参数异常
    throw new NullPointerException();
}

以上就是本人对kotlin中!!.跟?.的看法,如有不对之处还请各位指出

<think>好的,我需要回答用户关于Kotlin中非断言运算符(!!)的用法和最佳实践。首先,我得回忆一下Kotlin安全机制。Kotlin通过可类型和不可类型来避免指针异常,变量默认是不可的,如果要允许值,必须显式声明为可类型,比如String?。 非断言运算符!!的作用是将可类型强制转换为非类型。当开发者确定某个可变量在此时不为null时,可以使用!!,但如果变量实际上为null,就会抛出NPE。这个运算符虽然方便,但滥用会增加风险,所以最佳实践通常是建议尽量避免使用!!,优先使用安全调用操作符(?.)、Elvis运算符(?:)或者智能转换。 接下来,我需要整理!!的具体使用场景。例如,当与某些返回可类型的Java代码交互时,或者在单元测试中明确知道某个值不为null的情况下,可能不得不使用!!。但大部分情况下,应该考虑是否有更安全的方式替代。 然后,要列举使用!!的风险,比如潜在的运行时崩溃,代码可维护性降低等。同时,给出替代方案,比如lateinit修饰符用于延迟初始化,或者使用let函数结合安全调用。 还需要考虑引用资料中的内容,用户提供的引用[1]提到了Kotlin的基础知识,包括安全部分,可以结合这些内容来增强回答的权威性。例如,引用中提到的Nullability部分可能涉及如何处理可类型,这和非断言运算符直接相关。 最后,生成相关问题,可能需要覆盖替代方案、与Java互操作时的处理、以及如何处理可能潜在null的情况等。确保问题对用户深入学习有帮助。 现在,我需要将以上思考整理成结构化的回答,分点说明用法、风险、最佳实践,并给出代码示例。同时,确保使用正确的中文术语,避免技术术语错误。另外,注意引用标识的添加,比如在提到安全机制时,引用提供的资料[^1]。</think>### Kotlin断言运算符(!!)用法与最佳实践 Kotlin通过**安全机制**从语言层面减少指针异常(NPE)的发生。非断言运算符`!!`是处理可类型的特殊工具,需谨慎使用。 #### 一、基本用法 ```kotlin val name: String? = getNameFromNetwork() val length = name!!.length // 显式断言非 ``` - 作用:将可类型强制转换为非类型 - 效果:若变量为`null`立即抛出`NullPointerException` - 类型转换:`String?` ➔ `String` #### 二、适用场景 1. **与Java代码交互时** ```kotlin // Java方法:public @Nullable String getValue() val result = JavaClass().value!! // 明确知晓返回值非时 ``` 2. **单元测试环境** ```kotlin @Test fun `测试非数据`() { val testData = mockService.getData()!! assertTrue(testData.isValid()) } ``` 3. **契约式编程场景** ```kotlin fun processData(data: Data?) { require(data != null) // 前置条件检查 val size = data!!.payload.size // 此时已通过null检查 } ``` #### 三、风险警示 1. **代码异味指标**:项目中出现`!!`次数越多,安全防护越薄弱 2. **崩溃风险**:错误使用时100%触发NPE 3. **维护成本**:增加后续代码修改时遗漏null检查的风险 #### 四、最佳实践 1. **优先安全调用** ```kotlin name?.let { println(it.length) } // 安全调用+作用域函数 ``` 2. **使用Elvis运算符** ```kotlin val length = name?.length ?: 0 // 提供默认值 ``` 3. **lateinit延迟初始化** ```kotlin lateinit var config: Configuration fun init() { config = loadConfig() // 确保初始化后才使用 } ``` 4. **类型智能转换** ```kotlin if (name != null) { println(name.length) // 自动智能转换 } ``` 5. **require/check函数** ```kotlin fun save(data: Data?) { checkNotNull(data) database.save(data) // 此时data已确认为非 } ``` #### 五、代码审查建议 1. 将`!!`使用纳入代码评审重点 2. 配置Lint规则检测`!!`使用 3. 统计项目中的`!!`密度指标(建议低于0.1%) [^1]: Kotlin安全机制通过类型系统在编译期检测潜在指针风险,这是其相较于Java的重要改进特性。
评论 10
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值