首页 > 解决方案 > kotlin 和验证中的嵌套 let 块

问题描述

我想以功能方法处理验证。我有如下的用户验证逻辑

  1. 如果用户对象为空,则抛出异常
  2. 如果用户不活跃,则抛出异常
  3. 如果用户类型是 super 什么都不做。如果用户类型是 admin 做一些操作。

在我的代码片段中,如果生成的随机数是偶数,我将抛出异常,否则完成流程。

 * You can edit, run, and share this code. 
 * play.kotlinlang.org 
 */

import java.time.Instant
import java.util.UUID
import java.lang.IllegalStateException
import java.util.Random
fun main() {
    val user : User? = User(UUID.randomUUID(),Instant.now(),UserStatus.ACTIVE,UserType.SUPER) 
     //val user : User? =  User(UUID.randomUUID(),Instant.now(),UserStatus.ACTIVE,UserType.NORMAL) // works fine

    user?.let{ existing -> existing.takeIf{it.status == UserStatus.ACTIVE}?.let{ activeUser ->
            activeUser.takeUnless{ user -> user.userType == UserType.SUPER}?.let{
                    val number =  Random().nextInt(5);
                    println(number)
                    if(number %2 == 0) throw IllegalStateException("invalid random number")
            }
    }?: throw IllegalStateException("User is not active right now")

    } ?:throw IllegalStateException("user not created at all")
}

data class User(val uuid:UUID, val created:Instant, val status:UserStatus, val userType:UserType)



enum class UserStatus {
    ACTIVE,INACTIVE
}

enum class UserType{
    SUPER,NORMAL
}

user is not active right now当生成的随机数是奇数时,它会抛出不正确的错误消息。它应该默默地完成函数调用。任何人帮助我代码有什么问题?

标签: kotlinscopefunctional-programminglet

解决方案


在内部块

existing.takeIf { it.status == UserStatus.ACTIVE }?.let { activeUser ->
    activeUser.takeUnless { user -> user.userType == UserType.SUPER }?.let {
        val number = Random().nextInt(5);
        println(number)
        if (number % 2 == 0) throw IllegalStateException("invalid random number")
    }
} ?: throw IllegalStateException("User is not active right now")

你有一个非空值。

因为你没有链接?.电话。更好的主意是尽早返回/投掷。

val existing = user ?: throw IllegalStateException("user not created at all")

所以嵌套块可以展平。

当两项检查之一未得到满足时,会发生第二个异常“用户现在不处于活动状态”。描述这种业务逻辑的编程概念是不可为空的变量。定期的旧if支票更清洁。

if (existing.status != UserStatus.ACTIVE || existing.userType == UserType.SUPER) {
    throw IllegalStateException("User is not active right now")
}

顺便说一句,您的错误消息似乎是错误的。


通过“功能方法”,我认为您的意思是基于代码表达式。但是可空类型对于您的用例来说还不够强大。您应该寻找EitherTry。两者都可以在Arrow 库中找到。它们都是单子。它们的链接 ( flatMap) 和?.let调用一样,也需要回调地狱。

在某些语言中,有语法糖可以使flatMaps 的回调地狱变平。你可以看看Monad Comprehensions,看看你是否喜欢。

如果没有,没问题。在 Kotlin 中,提前返回/抛出没有任何羞耻感。

务实。


推荐阅读