为什么我在 Kotlin 中使用原始的 parseList 函数时得不到正确的结果?
Why don't I get correct result when I use original parseList function in Kotlin?
我正在为 Android 开发人员学习关于 Kotlin Anko 的示例代码(书籍)https://github.com/antoniolg/Kotlin-for-Android-Developers
方法一来自样例代码,重写了parseList,但比较难理解
所以我尝试使用方法2而不是方法1,方法2使用原始的parseList函数,但是我使用方法2时得到空白记录,我在方法2中犯了什么错误
class DayForecast(var map: MutableMap<String, Any?>) {
var _id: Long by map
var date: Long by map
var description: String by map
var high: Int by map
var low: Int by map
var iconUrl: String by map
var cityId: Long by map
constructor(date: Long, description: String, high: Int, low: Int,
iconUrl: String, cityId: Long) : this(HashMap()) {
this.date = date
this.description = description
this.high = high
this.low = low
this.iconUrl = iconUrl
this.cityId = cityId
}
}
方法一
override fun requestForecastByZipCode(zipCode: Long, date: Long) =
forecastDbHelper.use {
val dailyRequest = "${DayForecastTable.CITY_ID} = ? AND ${DayForecastTable.DATE} >= ?"
val dailyForecast = select(DayForecastTable.NAME)
.whereSimple(dailyRequest, zipCode.toString(), date.toString())
.parseList { DayForecast(HashMap(it)) }
/* common code block */
}
fun <T : Any> SelectQueryBuilder.parseList(parser: (Map<String, Any?>) -> T):
List<T> = parseList(object : MapRowParser<T> {
override fun parseRow(columns: Map<String, Any?>): T = parser(columns)
})
方法二
override fun requestForecastByZipCode(zipCode: Long, date: Long) =
forecastDbHelper.use {
val dailyRequest = "${DayForecastTable.CITY_ID} = ? AND ${DayForecastTable.DATE} >= ?"
val dailyForecast = select(DayForecastTable.NAME)
.whereSimple(dailyRequest, zipCode.toString(), date.toString())
.exec { parseList(classParser<DayForecast>()) }
/* common code block */
}
我真的认为你应该坚持使用 'method 1' 方法,一旦你意识到 Kotlin 允许你做什么,它就会容易得多。由于我不知道您对 Kotlin 了解多少,所以我会尽量全面介绍这一点。
现有的 class SelectQueryBuilder
有(我推测)一个名为 parseList
的函数,现有函数采用 MapRowParser<T>
。 MapRowParser<T>
有一个函数 parseRow
接受一个 Map<String, Any?>
和 returns 一个 T
.
在旧的 Java 工作方式中,您将从 MapRowParser<T>
派生并覆盖 parseRow
以便它进行您想要的转换;将 Map<String, Any?>
转换为 DayForecast
(泛型 T
现在有一个类型)。此派生的 class 的一个实例被传递到现有的 parseList
函数中。您派生的 class 看起来像
class MapToDayForecastRowParser extends MapRowParser<DayForecast> {
@Override public DayForecast parseRow(Map<String, Object> map) {
// Note that Java's "Object" is more or less Kotlin's "Any?"
return new DayForecast(map); // Might need to convert the map type btw
}
}
扩展方法使得 wrap/hide/abstract 派生 class 的创建变得非常容易。扩展方法采用 lambda,也就是说,您必须将一段代码解析为新的 parseList
方法,该代码块采用 Map<String, Any?>
和 returns T
(这就是DayForecast(HashMap(it))
是干什么的,it
是一个自动命名的变量,就是Map
。扩展方法然后调用已有的parseList
方法,解析成一个匿名的class 它会创建自己。这意味着使用此扩展方法会创建一个新的匿名 class,但 Kotlin 编译器处理得很好。
一开始让我感到困惑的一个部分是 Kotlin 处理匿名的方式 class。
// Java
new MapRowParser<T>() {
@Override public T parseRow(Map<String, Object>) {
/* Map to T logic */
}
}
// Kotlin
object : MapRowParser<T> {
override fun parseRow(columns: Map<String, Any?>): T = parser(columns)
}
Kotlin 也使得处理 'lambda' 变得非常容易。它被解析为 parser
的扩展方法,然后设置为我们的匿名 class parseRow
函数的实现。如果你愿意,你也可以重用它们,所以如果你需要在很多地方做同样的解析,你可以使用命名函数。
这种新的 Kotlin 方式的最大优势在于它可以很容易地专注于您想要做的事情。使用该扩展方法,可以非常快速地重新使用它,以便在另一个查询中您可以执行 parseList{ it.getOrDefault("name", "unkown_user") }
。您现在可以轻松地思考 "If each row is a map, how do I convert that down to a value I want?".
我正在为 Android 开发人员学习关于 Kotlin Anko 的示例代码(书籍)https://github.com/antoniolg/Kotlin-for-Android-Developers
方法一来自样例代码,重写了parseList,但比较难理解
所以我尝试使用方法2而不是方法1,方法2使用原始的parseList函数,但是我使用方法2时得到空白记录,我在方法2中犯了什么错误
class DayForecast(var map: MutableMap<String, Any?>) {
var _id: Long by map
var date: Long by map
var description: String by map
var high: Int by map
var low: Int by map
var iconUrl: String by map
var cityId: Long by map
constructor(date: Long, description: String, high: Int, low: Int,
iconUrl: String, cityId: Long) : this(HashMap()) {
this.date = date
this.description = description
this.high = high
this.low = low
this.iconUrl = iconUrl
this.cityId = cityId
}
}
方法一
override fun requestForecastByZipCode(zipCode: Long, date: Long) =
forecastDbHelper.use {
val dailyRequest = "${DayForecastTable.CITY_ID} = ? AND ${DayForecastTable.DATE} >= ?"
val dailyForecast = select(DayForecastTable.NAME)
.whereSimple(dailyRequest, zipCode.toString(), date.toString())
.parseList { DayForecast(HashMap(it)) }
/* common code block */
}
fun <T : Any> SelectQueryBuilder.parseList(parser: (Map<String, Any?>) -> T):
List<T> = parseList(object : MapRowParser<T> {
override fun parseRow(columns: Map<String, Any?>): T = parser(columns)
})
方法二
override fun requestForecastByZipCode(zipCode: Long, date: Long) =
forecastDbHelper.use {
val dailyRequest = "${DayForecastTable.CITY_ID} = ? AND ${DayForecastTable.DATE} >= ?"
val dailyForecast = select(DayForecastTable.NAME)
.whereSimple(dailyRequest, zipCode.toString(), date.toString())
.exec { parseList(classParser<DayForecast>()) }
/* common code block */
}
我真的认为你应该坚持使用 'method 1' 方法,一旦你意识到 Kotlin 允许你做什么,它就会容易得多。由于我不知道您对 Kotlin 了解多少,所以我会尽量全面介绍这一点。
现有的 class SelectQueryBuilder
有(我推测)一个名为 parseList
的函数,现有函数采用 MapRowParser<T>
。 MapRowParser<T>
有一个函数 parseRow
接受一个 Map<String, Any?>
和 returns 一个 T
.
在旧的 Java 工作方式中,您将从 MapRowParser<T>
派生并覆盖 parseRow
以便它进行您想要的转换;将 Map<String, Any?>
转换为 DayForecast
(泛型 T
现在有一个类型)。此派生的 class 的一个实例被传递到现有的 parseList
函数中。您派生的 class 看起来像
class MapToDayForecastRowParser extends MapRowParser<DayForecast> {
@Override public DayForecast parseRow(Map<String, Object> map) {
// Note that Java's "Object" is more or less Kotlin's "Any?"
return new DayForecast(map); // Might need to convert the map type btw
}
}
扩展方法使得 wrap/hide/abstract 派生 class 的创建变得非常容易。扩展方法采用 lambda,也就是说,您必须将一段代码解析为新的 parseList
方法,该代码块采用 Map<String, Any?>
和 returns T
(这就是DayForecast(HashMap(it))
是干什么的,it
是一个自动命名的变量,就是Map
。扩展方法然后调用已有的parseList
方法,解析成一个匿名的class 它会创建自己。这意味着使用此扩展方法会创建一个新的匿名 class,但 Kotlin 编译器处理得很好。
一开始让我感到困惑的一个部分是 Kotlin 处理匿名的方式 class。
// Java
new MapRowParser<T>() {
@Override public T parseRow(Map<String, Object>) {
/* Map to T logic */
}
}
// Kotlin
object : MapRowParser<T> {
override fun parseRow(columns: Map<String, Any?>): T = parser(columns)
}
Kotlin 也使得处理 'lambda' 变得非常容易。它被解析为 parser
的扩展方法,然后设置为我们的匿名 class parseRow
函数的实现。如果你愿意,你也可以重用它们,所以如果你需要在很多地方做同样的解析,你可以使用命名函数。
这种新的 Kotlin 方式的最大优势在于它可以很容易地专注于您想要做的事情。使用该扩展方法,可以非常快速地重新使用它,以便在另一个查询中您可以执行 parseList{ it.getOrDefault("name", "unkown_user") }
。您现在可以轻松地思考 "If each row is a map, how do I convert that down to a value I want?".