主要涉及kotlin知识点有:伴生对象、类的定义、委托属性和when表达式。
由于SP支持存储多种类型的数据,所以在这里我们定义成了一个带泛型的类。第一个变量prefs使用了懒加载lazy来缓存SharedPreference对象。getValue和setValue函数使用了operator修饰,operator修饰的函数就是属性委托的时候主动调用的。
class Preference(val name: String, private val default: T) {
private val prefs: SharedPreferences by lazy { App.instance.applicationContext.getSharedPreferences(name, Context.MODE_PRIVATE) }
operator fun getValue(thisRef: Any?, property: KProperty<*>): T {
Log.i("info", "调用$this 的getValue()")
return getSharePreferences(name, default)
}
operator fun setValue(thisRef: Any?, property: KProperty<*>, value: T) {
Log.i("info", "调用$this 的setValue() value参数值为:$value")
putSharePreferences(name, value)
}
@SuppressLint("CommitPrefEdits")
private fun putSharePreferences(name: String, value: T) = with(prefs.edit()) {
when (value) {
is Long -> putLong(name, value)
is String -> putString(name, value)
is Int -> putInt(name, value)
is Boolean -> putBoolean(name, value)
is Float -> putFloat(name, value)
else -> throw IllegalArgumentException("This type of data cannot be saved!")
}.apply()
}
@Suppress("UNCHECKED_CAST")
private fun getSharePreferences(name: String, default: T): T = with(prefs) {
val res: Any = when (default) {
is Long -> getLong(name, default)
is String -> getString(name, default)
is Int -> getInt(name, default)
is Boolean -> getBoolean(name, default)
is Float -> getFloat(name, default)
else -> throw IllegalArgumentException("This type of data cannot be saved!")
}
return res as T
}
}
类中第一行代码通过调用了Application的实例来获取Context实例。在自定义的Application中,创建了一个伴生对象,用于获取Application实例。
class App : Application() {
companion object {// 伴生对象
lateinit var instance: App
private set
}
override fun onCreate() {
super.onCreate()
instance = this
}
}
使用Preference,是需要一行代码。即,private var variable by Preference(“keyName”, “10”)。
class MainActivity : AppCompatActivity() {
// 委托属性,Preference把取值和存值的操作代理给variable,我们对userId的赋值和取值最终是操作的Preference得setValue和getValue函数。
private var variable by Preference("keyName", "10")//由于已经能够推断出variable的类型是String,所以这里的variable省去了类型。
//完整写法如下:
//private var variable: String by Preference("keyName", "10")
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_main)
tvShowResult.text = variable
btnSave.setOnClickListener {
variable = "20"
Log.i("info", "保存成功")
}
btnRead.setOnClickListener {
tvShowResult.text = "new-" + variable
}
}
}
首次安装apk,SP文件中没有存值,variable的到的默认值“10”。 执行了variable =
“20”之后,再去获取值时,就变成“20”了,我们看看日志,在取值的时候执行了Preference的getValue()函数。
12-21 18:48:10.303 30573-30573/com.kv.kotlindemo I/info: 调用com.kv.kotlindemo.Preference@18a33be3 的getValue()
12-21 18:48:15.313 30573-30573/com.kv.kotlindemo I/info: 调用com.kv.kotlindemo.Preference@18a33be3 的setValue() value参数值为:20
12-21 18:48:15.313 30573-30573/com.kv.kotlindemo I/info: 保存成功
12-21 18:48:16.373 30573-30573/com.kv.kotlindemo I/info: 调用com.kv.kotlindemo.Preference@18a33be3 的getValue()