Java Preferences API с делегированными свойствами Kotlin
Я хочу использовать Java Preferences API для хранения некоторых данных для моего приложения. Поскольку я пишу приложение на Kotlin, я хотел бы воспользоваться делегированными свойствами, чтобы получить и установить свои предпочтения.
Мои текущие настройки такие
object Pref {
val systemPref by lazy { Preferences.systemNodeForPackage(App::class.java) }
val userPref by lazy { Preferences.userNodeForPackage(App::class.java) }
// example preference
const val SETTING = "setting"
var setting: Int
get() = userPref.getInt(SETTING, -1)
set(value) = userPref.putInt(SETTING, value)
}
Но я хотел бы иметь что-то вроде этого, чтобы объявить предпочтения
var setting: Int by preference(userPref, "setting", -1)
В API есть разные методы доступа к настройкам разных типов данных. Я хотел бы использовать их, но это усложняет для меня вещи. Есть ли способ иметь такой делегат, который будет работать с любым типом данных, а также не будет преобразовывать все в строки и обратно при общении с библиотекой?
1 ответ
Вы можете заставить конструктор делегата взять
KClass
параметр, для которого вы проверяете тип, чтобы вернуть правильный тип предпочтения. Тогда напишите свой
preference
функция с
reified
type, чтобы он мог соответствующим образом создать экземпляр делегата. Что-то вроде этого должно использоваться с синтаксисом, который вы продемонстрировали:
inline fun <reified T: Any> preference(preferences: Preferences, key: String, defaultValue: T)
= PreferenceDelegate(preferences, key, defaultValue, T::class)
class PreferenceDelegate<T: Any>(
val preferences: Preferences,
val key: String,
val defaultValue: T,
val type: KClass<T>
): ReadWriteProperty<Any, T> {
@Suppress("UNCHECKED_CAST")
override fun setValue(thisRef: Any, property: KProperty<*>, value: T) {
with (preferences) {
when (type) {
Int::class -> putInt(key, value as Int)
Long::class -> putLong(key, value as Long)
Float::class -> putFloat(key, value as Float)
Boolean::class -> putBoolean(key, value as Boolean)
String::class -> put(key, value as String)
ByteArray::class -> putByteArray(key, value as ByteArray)
else -> error("Unsupported preference type $type.")
}
}
}
@Suppress("UNCHECKED_CAST")
override fun getValue(thisRef: Any, property: KProperty<*>): T {
return with (preferences) {
when (type) {
Int::class -> getInt(key, defaultValue as Int)
Long::class -> getLong(key, defaultValue as Long)
Float::class -> getFloat(key, defaultValue as Float)
Boolean::class -> getBoolean(key, defaultValue as Boolean)
String::class -> get(key, defaultValue as String)
ByteArray::class -> getByteArray(key, defaultValue as ByteArray)
else -> error("Unsupported preference type $type.")
}
} as T
}
}