参考博客 【Kotlin】Kotlin 中使用 Lambda 表达式替代对象表达式原理分析 ( 尾随 Lambda - Trailing Lambda 语法 | 接口对象表达式 = 接口#函数类型对象 )





一、错误记录



在 Android 中 , 使用 Kotlin 开发 , 为 BottomNavigationView 设置 OnNavigationItemSelectedListener 监听接口 ;

设置的接口是一个匿名内部类 BottomNavigationView.OnNavigationItemSelectedListener 对象 , 其中定义了一个

boolean onNavigationItemSelected(@NonNull MenuItem var1); 

函数 , 需要返回一个布尔值 ;

直接使用 return 返回布尔值 , 就报如下错误 ;

在这里插入图片描述





二、问题分析




1、匿名内部类


BottomNavigationView 调用 setOnNavigationItemSelectedListener 函数 , 设置的监听器是 BottomNavigationView.OnNavigationItemSelectedListener 类型的匿名内部类 ;

最原始的设置方式如下 , 首先创建 BottomNavigationView.OnNavigationItemSelectedListener 类型的 对象表达式 , 也就是匿名内部类 , 然后 调用 setOnNavigationItemSelectedListener 函数将其设置给 BottomNavigationView 作为 选择监听器 ;

        // 创建匿名内部类
        val listener = object : BottomNavigationView.OnNavigationItemSelectedListener {
            override fun onNavigationItemSelected(p0: MenuItem): Boolean {
                return false
            }
        }
        // 设置匿名内部类参数
        navView.setOnNavigationItemSelectedListener(listener)

在上一步的基础上 , 可以不进行声明 , 直接设置 匿名内部类 , 如下代码所示 :

        // 直接设置匿名内部类
        navView.setOnNavigationItemSelectedListener(object : BottomNavigationView.OnNavigationItemSelectedListener {
            override fun onNavigationItemSelected(p0: MenuItem): Boolean {
                return false
            }
        })

2、尾随 Lambda 规范 - Lambda 替换接口


参考博客 【Kotlin】Kotlin 中使用 Lambda 表达式替代对象表达式原理分析 ( 尾随 Lambda - Trailing Lambda 语法 | 接口对象表达式 = 接口#函数类型对象 ) , 符合 尾随 Lambda 表达式的要求 , 最后一个函数是匿名内部类 , 匿名内部类中只实现了一个函数 , 此时使用 Lambda 表达式替代该 匿名内部类 ;


Lambda 表达式 其本质 就是 函数类型 的 匿名对象 , 也是一个实例对象 , 在堆内存中分配相应的空间 ;

在下面的代码中 , 使用 对象表达式 创建了匿名对象 , 该匿名类实现了 BottomNavigationView.OnNavigationItemSelectedListener 接口 , 并实现了其中的 onNavigationItemSelected 函数 ;

object : BottomNavigationView.OnNavigationItemSelectedListener {
	override fun onNavigationItemSelected(p0: MenuItem): Boolean {
		return false
	}
}

符合以下两个条件 :

  • 函数 接收一个 接口类型 的匿名内部类 或 对象表达式 ;
  • 该 接口类型 中 只定义了一个函数 ;

可以 省略掉 匿名内部类 也就是 对象表达式的定义 , 直接使用 接口中的函数 类型对象 , 也就是 Lambda 表达式 / 匿名函数 / 闭包 来替代该 接口类型 变量 ;


省略后的简写方式如下 :

        // Lambda 替换对象表达式
        navView.setOnNavigationItemSelectedListener {
            return@setOnNavigationItemSelectedListener false
        }

3、Lambda 表达式中 return 需要 @ 标签


这里特别注意 :

在 Kotlin 中 , 在 lambda 表达式或匿名函数中使用 return 语句时 , 必须使用 return@label 语法来指定你要返回的标签 ;

在 Kotlin 中 , return 语句默认是从最近的封闭函数返回的 , 而在 lambda 表达式中使用 return 时 , 它会尝试从包含它的函数返回 ;





三、解决方案



在 Lambda 表达式的 return 返回时 , 添加 @ 标签 , 不能直接使用 return 进行返回 ;

        // Lambda 替换对象表达式
        navView.setOnNavigationItemSelectedListener {
            return@setOnNavigationItemSelectedListener false
        }
Logo

欢迎加入西安开发者社区!我们致力于为西安地区的开发者提供学习、合作和成长的机会。参与我们的活动,与专家分享最新技术趋势,解决挑战,探索创新。加入我们,共同打造技术社区!

更多推荐