我想从HOCON(Typesafe配置)文件中读取以下配置到Kotlin。
tablename: {
columns: [
{ item: { type: integer, key: true, null: false } }
{ desc: { type: varchar, length: 64 } }
{ quantity: { type: integer, null: false } }
{ price: { type: decimal, precision: 14, scale: 3 } }
]
}
实际上我想提取关键列。到目前为止,我已尝试过以下内容。
val metadata = ConfigFactory.parseFile(metafile)
val keys = metadata.getObjectList("${tablename.toLowerCase()}.columns")
.filter { it.unwrapped().values.first().get("key") == true }
但它失败并出现以下错误。
Unresolved reference. None of the following candidates is applicable because of receiver type mismatch:
@kotlin.internal.InlineOnly public operator inline fun <@kotlin.internal.OnlyInputTypes K, V> kotlin.collections.Map<out kotlin.String, ???>.get(key: kotlin.String): ??? defined in kotlin.collections
很明显,Kotlin无法理解Map中“value”字段的数据类型。我如何申报或让Kotlin知道?
此地图中还有不同的类型和可选键。
PS:我知道Kotlin有几种包装,如Konfig和Klutter。我希望如果这很容易写,我可以避免使用另一个库。
更新1:
我尝试了以下内容。
it.unwrapped().values.first().get<String, Boolean>("key")
获取以下编译器错误。
Unresolved reference. None of the following candidates is applicable because of receiver type mismatch:
@kotlin.internal.InlineOnly public operator inline fun <@kotlin.internal.OnlyInputTypes K, V> kotlin.collections.Map<out kotlin.String, kotlin.Boolean>.get(key: kotlin.String): kotlin.Boolean? defined in kotlin.collections
这个
it.unwrapped().values.first().get<String, Boolean?>("key")
带输出
Unresolved reference. None of the following candidates is applicable because of receiver type mismatch:
@kotlin.internal.InlineOnly public operator inline fun <@kotlin.internal.OnlyInputTypes K, V> kotlin.collections.Map<out kotlin.String, kotlin.Boolean?>.get(key: kotlin.String): kotlin.Boolean? defined in kotlin.collections
更新2:
看看它在其他地方处理的方式,我想我可能需要使用反射。用我有限的曝光试试吧。到目前为止没有运气。
答案 0 :(得分:8)
考虑一下你的代码,解构如下:
val keys = metadata.getObjectList("tablename.columns")
.filter {
val item:ConfigObject = it
val unwrapped:Map<String,Any?> = item.unwrapped()
val values:Collection<Any?> = unwrapped.values
val firstValue:Any? = values.first()
firstValue.get("key") == true // does not compile
}
从上面的问题应该是显而易见的。您需要使用firstValue
拥有Map
的信息来帮助编译器:
val firstValueMap = firstValue as Map<String,Any?>
firstValueMap["key"] == true
答案 1 :(得分:2)
即使你没有使用Klutter,我也为它创建了一个更新,使ConfigObject
和Config
行为统一相同。从Klutter版本1.17.1
开始(今天推送到Maven中心),您可以根据您的问题执行以下单元测试中表示的内容。
查找关键列的函数:
fun findKeyColumns(cfg: Config, tableName: String): Map<String, ConfigObject> {
return cfg.nested(tableName).value("columns").asObjectList()
.map { it.keys.single() to it.value(it.keys.single()).asObject() }
.filter {
it.second.value("key").asBoolean(false)
}
.toMap()
}
这是完整的单元测试:
// from http://stackoverflow.com/questions/37092808/reading-and-processing-hocon-in-kotlin
@Test fun testFromSo37092808() {
// === mocked configuration file
val cfg = loadConfig(StringAsConfig("""
products: {
columns: [
{ item: { type: integer, key: true, null: false } }
{ desc: { type: varchar, length: 64 } }
{ quantity: { type: integer, null: false } }
{ price: { type: decimal, precision: 14, scale: 3 } }
]
}
"""))
// === function to find which columns are key columns
fun findKeyColumns(cfg: Config, tableName: String): Map<String, ConfigObject> {
return cfg.nested(tableName).value("columns").asObjectList()
.map { it.keys.single() to it.value(it.keys.single()).asObject() }
.filter {
it.second.value("key").asBoolean(false)
}
.toMap()
}
// === sample usage
val productKeys = findKeyColumns(cfg, "products")
// we only have 1 in the test data, so grab the name and the values
val onlyColumnName = productKeys.entries.first().key
val onlyColumnObj = productKeys.entries.first().value
assertEquals ("item", onlyColumnName)
assertEquals (true, onlyColumnObj.value("key").asBoolean())
assertEquals ("integer", onlyColumnObj.value("type").asString())
assertEquals (false, onlyColumnObj.value("null").asBoolean())
}
您可以返回上面的Map
或列表名Pair
的列表到设置映射,因为列名不在其设置中。
也可以更改配置文件的设计,使配置处理更简单(即配置对象内的表名,而不是左侧键。列名相同,添加到对象而不是左侧键。)