任何人都面临这个问题。
break or continue jump across class boundary kotlin
当我打算使用break或continue时会出现此问题。在lambda中使用接收器我创建'letIn'
lambda with receiver code
fun letIn(componentName: String?, values: List<LifeService.Value?>?,
body: (String, List<LifeService.Value?>) -> Unit) {
if (!TextUtils.isEmpty(componentName) && (values != null && values.isNotEmpty())) {
body(componentName!!, values)
}
}
此示例代码。
for (option in 0 until optionsSize) {
val component = optionsGroup?.options?.get(option)
component?.let {
with(component) {
letIn(presentation, values, { componentName, values ->
if (componentName == LifeComponentViewType.CHECKBOX) {
letIn(transformCheckBoxValues(optionsGroup), { data ->
dataSource?.push(componentName, ComponentDataCheckBoxCollection(name, data))
view.buildComponent(componentName)
// break or continue didnt work
})
} else {
dataSource?.push(componentName, ComponentDataCollection(name, values))
view.buildComponent(componentName)
}
})
}
}
}
因为上面的代码没有用,所以我使用命令式方法。
for (option in 0 until optionsSize) {
val component = optionsGroup?.options?.get(option)
if (component != null) {
val presentation: String? = component.presentation
val values = component.values
if (!TextUtils.isEmpty(presentation)) {
if (presentation == LifeComponentViewType.CHECKBOX) {
val data = transformCheckBoxValues(optionsGroup)
if (data.isNotEmpty()) {
dataSource?.push(presentation, ComponentDataCheckBoxCollection(optionsGroup.name, data))
view.buildComponent(presentation)
return
}
} else {
dataSource?.push(presentation!!, ComponentDataCollection(component.name, values))
view.buildComponent(presentation!!)
}
} else {
return
}
}
}
有没有人有建议?
更新 我已经通过内联高阶函数解决了这个问题。
答案 0 :(得分:6)
(除了其他编码错误)您看到错误,因为在lambda中,您不能使用break
或continue
跳出lambda到最近的循环。相反,您可以使用限定的return
跳出lambda到标签。
return-expression从最近的封闭函数返回,即foo。 (请注意,仅对传递给内联函数的lambda表达式支持此类非本地返回。)如果我们需要从lambda表达式返回,我们必须标记它并限定返回:
(强调我的)
您的第二个示例显示您希望lambda从封闭函数执行非本地返回。因此,您不需要对return
进行限定,但必须将您的函数letIn
声明为inline
(否则您只能进行本地的合格回复)。
inline fun letIn(componentName: String?, values: List<LifeService.Value?>?,
body: (String, List<LifeService.Value?>) -> Unit) {
if (!TextUtils.isEmpty(componentName) && (values != null && values.isNotEmpty())) {
body(componentName!!, values)
}
}
...或者如果你想让它有接收器......
inline fun String?.letIn(values: List<LifeService.Value?>?,
body: String.(List<LifeService.Value?>) -> Unit) {
if (!TextUtils.isEmpty(this) && (values != null && values.isNotEmpty())) {
this!!.body(values)
}
}
当您将letIn
声明为inline
时,您可以将return
放在lambda中,而不会让编译器抱怨。如果你的lambdas只做本地返回,你的函数不需要是inline
,但它需要有一个合格的返回(例如return@letIn
)。
你的第一个例子就像这样......
for (option in 0 until optionsSize) {
val component = optionsGroup?.options?.get(option)
component?.let {
with(component) {
presentation.letIn(values, { values ->
if (this == LifeComponentViewType.CHECKBOX) {
this.letIn(transformCheckBoxValues(optionsGroup), { data ->
dataSource?.push(this, ComponentDataCheckBoxCollection(this, data))
view.buildComponent(this)
return //returns from function
})
} else {
dataSource?.push(this, ComponentDataCollection(name, values))
view.buildComponent(this)
return //returns from function
}
})
}
}
}
最后,请注意,如果你想提前跳出lambda,但继续外循环,如:
fun test1() {
val list = listOf("a", "b", "c")
val optionsSize = 2
for(i in 0..optionsSize) loop@ {
println("calliing list.forEach")
list.forEach lit@ {
if(it == "a") return@lit
if(it == "c") return@loop
println(it)
}
}
}
它不会起作用。 Intelli-sense并没有抱怨它,但是编译器抛出了一个内部错误。但是你可以将外部循环转换为lambda,它确实可以工作......
fun test() {
val list = listOf("a", "b", "c")
val optionsSize = 2
(0..optionsSize).forEach() loop@ {
println("calliing list.forEach")
list.forEach lit@ {
if(it == "a") return@lit
if(it == "c") return@loop
println(it)
}
}
}
同样,这只有在传递lambda的函数被声明为inline
时才会起作用(如forEach
声明为inline
)。