5

我想将 HOCON (Typesafe Config) 文件中的以下配置读入 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 中“值”字段的数据类型。我如何声明或让 Kotlin 知道?

也不是说此 Map 中有不同的类型和可选键。

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:

看看它在其他地方的处理方式,我想我可能需要使用反射。在我有限的曝光范围内尝试一下。到目前为止没有运气。

4

2 回答 2

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
于 2016-05-08T09:41:30.927 回答
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如上所述的 a 或Pair列名到设置映射的列表,因为列名不在其设置内。

配置文件的设计也可以改变,使配置的处理更简单(即配置对象内的表名,而不是左侧键。列名也一样,添加到对象中并不是左侧键。)

于 2016-05-09T18:22:38.853 回答