泛型preference委托

现在我们已经是泛型专家了,为什么不扩展LongPreference为支持所有Shared Preferences支持的类型呢?我们来创建一个Preference委托:

  1. class Preference<T>(val context: Context, val name: String, val default: T)
  2. : ReadWriteProperty<Any?, T> {
  3. val prefs by lazy {
  4. context.getSharedPreferences("default", Context.MODE_PRIVATE)
  5. }
  6. override fun getValue(thisRef: Any?, property: KProperty<*>): T {
  7. return findPreference(name, default)
  8. }
  9. override fun setValue(thisRef: Any?, property: KProperty<*>, value: T) {
  10. putPreference(name, value)
  11. }
  12. ...
  13. }

这个preference与我们之前使用的非常相似。我们仅仅替换了Long为泛型类型T,然后调用了两个函数来做具体重要的工作。这些函数非常简单,尽管有些重复。它们会检查类型然后使用指定的方式来操作。比如,findPrefernce函数如下:

  1. private fun <T> findPreference(name: String, default: T): T = with(prefs) {
  2. val res: Any = when (default) {
  3. is Long -> getLong(name, default)
  4. is String -> getString(name, default)
  5. is Int -> getInt(name, default)
  6. is Boolean -> getBoolean(name, default)
  7. is Float -> getFloat(name, default)
  8. else -> throw IllegalArgumentException(
  9. "This type can be saved into Preferences")
  10. }
  11. res as T
  12. }

putPreference函数也是一样,但是在when最后通过apply,使用preferences editor保存结果:

  1. private fun <U> putPreference(name: String, value: U) = with(prefs.edit()) {
  2. when (value) {
  3. is Long -> putLong(name, value)
  4. is String -> putString(name, value)
  5. is Int -> putInt(name, value)
  6. is Boolean -> putBoolean(name, value)
  7. is Float -> putFloat(name, value)
  8. else -> throw IllegalArgumentException("This type can be saved into Pref\
  9. erences")
  10. }.apply()
  11. }

现在修改DelegateExt

  1. object DelegatesExt {
  2. ...
  3. fun preference<T : Any>(context: Context, name: String, default: T)
  4. = Preference(context, name, default)
  5. }

这章之后,用户可以访问设置界面并修改zip code。然后当我们返回主界面,forecast会自动重新刷新并显示新的信息。查看代码库中其余xi wei